soc-2008-mxcurioni: merged changes to revision 15163
This commit is contained in:
@@ -139,6 +139,24 @@ BF_FFMPEG_LIB = ''
|
|||||||
BF_FFMPEG_INC = '${BF_FFMPEG}/include'
|
BF_FFMPEG_INC = '${BF_FFMPEG}/include'
|
||||||
BF_FFMPEG_LIBPATH='${BF_FFMPEG}/lib'
|
BF_FFMPEG_LIBPATH='${BF_FFMPEG}/lib'
|
||||||
|
|
||||||
|
WITH_BF_OPENJPEG = 'true'
|
||||||
|
BF_OPENJPEG = '#extern/libopenjpeg'
|
||||||
|
BF_OPENJPEG_LIB = ''
|
||||||
|
# Uncomment the following two lines to use system's ffmpeg
|
||||||
|
# BF_FFMPEG = '/usr'
|
||||||
|
# BF_FFMPEG_LIB = 'avformat avcodec swscale avutil'
|
||||||
|
BF_OPENJPEG_INC = '${BF_OPENJPEG}/include'
|
||||||
|
BF_OPENJPEG_LIBPATH='${BF_OPENJPEG}/lib'
|
||||||
|
|
||||||
|
WITH_BF_REDCODE = 'true'
|
||||||
|
BF_REDCODE = '#extern/libredcode'
|
||||||
|
BF_REDCODE_LIB = ''
|
||||||
|
# Uncomment the following two lines to use system's ffmpeg
|
||||||
|
# BF_FFMPEG = '/usr'
|
||||||
|
# BF_FFMPEG_LIB = 'avformat avcodec swscale avutil'
|
||||||
|
BF_REDCODE_INC = '${BF_REDCODE}/include'
|
||||||
|
BF_REDCODE_LIBPATH='${BF_REDCODE}/lib'
|
||||||
|
|
||||||
# Mesa Libs should go here if your using them as well....
|
# Mesa Libs should go here if your using them as well....
|
||||||
WITH_BF_STATICOPENGL = 'false'
|
WITH_BF_STATICOPENGL = 'false'
|
||||||
BF_OPENGL = '/usr'
|
BF_OPENGL = '/usr'
|
||||||
|
|||||||
@@ -166,7 +166,7 @@ REL_CCFLAGS = ['-O2', '-DNDEBUG']
|
|||||||
C_WARN = []
|
C_WARN = []
|
||||||
CC_WARN = []
|
CC_WARN = []
|
||||||
|
|
||||||
LLIBS = 'ws2_32 vfw32 winmm kernel32 user32 gdi32 comdlg32 advapi32 shell32 ole32 oleaut32 uuid'
|
LLIBS = 'ws2_32 vfw32 winmm kernel32 user32 gdi32 comdlg32 advapi32 shfolder shell32 ole32 oleaut32 uuid'
|
||||||
|
|
||||||
PLATFORM_LINKFLAGS = '''
|
PLATFORM_LINKFLAGS = '''
|
||||||
/SUBSYSTEM:CONSOLE
|
/SUBSYSTEM:CONSOLE
|
||||||
|
|||||||
6
extern/SConscript
vendored
6
extern/SConscript
vendored
@@ -25,5 +25,11 @@ if env['WITH_BF_FFMPEG'] and env['BF_FFMPEG_LIB'] == '':
|
|||||||
SConscript(['xvidcore/SConscript'])
|
SConscript(['xvidcore/SConscript'])
|
||||||
SConscript(['ffmpeg/SConscript'])
|
SConscript(['ffmpeg/SConscript'])
|
||||||
|
|
||||||
|
if env['WITH_BF_OPENJPEG'] and env['BF_OPENJPEG_LIB'] == '':
|
||||||
|
SConscript(['libopenjpeg/SConscript'])
|
||||||
|
|
||||||
|
if env['WITH_BF_REDCODE'] and env['BF_REDCODE_LIB'] == '':
|
||||||
|
SConscript(['libredcode/SConscript'])
|
||||||
|
|
||||||
if env['OURPLATFORM'] == 'linux2':
|
if env['OURPLATFORM'] == 'linux2':
|
||||||
SConscript(['binreloc/SConscript']);
|
SConscript(['binreloc/SConscript']);
|
||||||
|
|||||||
20
extern/libopenjpeg/SConscript
vendored
Normal file
20
extern/libopenjpeg/SConscript
vendored
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
|
Import('env')
|
||||||
|
|
||||||
|
sources = env.Glob('*.c')
|
||||||
|
incs = '.'
|
||||||
|
|
||||||
|
flags = "-Wall -O3 -ffast-math -std=c99"
|
||||||
|
|
||||||
|
oj_env = env.Copy();
|
||||||
|
oj_env.Replace(CCFLAGS = '')
|
||||||
|
oj_env.Replace(BF_DEBUG_FLAGS = '')
|
||||||
|
|
||||||
|
oj_env.BlenderLib ( libname='extern_openjpeg',
|
||||||
|
sources=sources, includes=Split(incs),
|
||||||
|
defines=[],
|
||||||
|
libtype=['core','intern','player'],
|
||||||
|
priority=[10, 10, 300], compileflags = Split(flags))
|
||||||
187
extern/libopenjpeg/bio.c
vendored
Normal file
187
extern/libopenjpeg/bio.c
vendored
Normal file
@@ -0,0 +1,187 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/** @defgroup BIO BIO - Individual bit input-output stream */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Local static functions */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Write a bit
|
||||||
|
@param bio BIO handle
|
||||||
|
@param b Bit to write (0 or 1)
|
||||||
|
*/
|
||||||
|
static void bio_putbit(opj_bio_t *bio, int b);
|
||||||
|
/**
|
||||||
|
Read a bit
|
||||||
|
@param bio BIO handle
|
||||||
|
@return Returns the read bit
|
||||||
|
*/
|
||||||
|
static int bio_getbit(opj_bio_t *bio);
|
||||||
|
/**
|
||||||
|
Write a byte
|
||||||
|
@param bio BIO handle
|
||||||
|
@return Returns 0 if successful, returns 1 otherwise
|
||||||
|
*/
|
||||||
|
static int bio_byteout(opj_bio_t *bio);
|
||||||
|
/**
|
||||||
|
Read a byte
|
||||||
|
@param bio BIO handle
|
||||||
|
@return Returns 0 if successful, returns 1 otherwise
|
||||||
|
*/
|
||||||
|
static int bio_bytein(opj_bio_t *bio);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
local functions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
static int bio_byteout(opj_bio_t *bio) {
|
||||||
|
bio->buf = (bio->buf << 8) & 0xffff;
|
||||||
|
bio->ct = bio->buf == 0xff00 ? 7 : 8;
|
||||||
|
if (bio->bp >= bio->end) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
*bio->bp++ = bio->buf >> 8;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int bio_bytein(opj_bio_t *bio) {
|
||||||
|
bio->buf = (bio->buf << 8) & 0xffff;
|
||||||
|
bio->ct = bio->buf == 0xff00 ? 7 : 8;
|
||||||
|
if (bio->bp >= bio->end) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
bio->buf |= *bio->bp++;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void bio_putbit(opj_bio_t *bio, int b) {
|
||||||
|
if (bio->ct == 0) {
|
||||||
|
bio_byteout(bio);
|
||||||
|
}
|
||||||
|
bio->ct--;
|
||||||
|
bio->buf |= b << bio->ct;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int bio_getbit(opj_bio_t *bio) {
|
||||||
|
if (bio->ct == 0) {
|
||||||
|
bio_bytein(bio);
|
||||||
|
}
|
||||||
|
bio->ct--;
|
||||||
|
return (bio->buf >> bio->ct) & 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
Bit Input/Output interface
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
opj_bio_t* bio_create(void) {
|
||||||
|
opj_bio_t *bio = (opj_bio_t*)opj_malloc(sizeof(opj_bio_t));
|
||||||
|
return bio;
|
||||||
|
}
|
||||||
|
|
||||||
|
void bio_destroy(opj_bio_t *bio) {
|
||||||
|
if(bio) {
|
||||||
|
opj_free(bio);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int bio_numbytes(opj_bio_t *bio) {
|
||||||
|
return (bio->bp - bio->start);
|
||||||
|
}
|
||||||
|
|
||||||
|
void bio_init_enc(opj_bio_t *bio, unsigned char *bp, int len) {
|
||||||
|
bio->start = bp;
|
||||||
|
bio->end = bp + len;
|
||||||
|
bio->bp = bp;
|
||||||
|
bio->buf = 0;
|
||||||
|
bio->ct = 8;
|
||||||
|
}
|
||||||
|
|
||||||
|
void bio_init_dec(opj_bio_t *bio, unsigned char *bp, int len) {
|
||||||
|
bio->start = bp;
|
||||||
|
bio->end = bp + len;
|
||||||
|
bio->bp = bp;
|
||||||
|
bio->buf = 0;
|
||||||
|
bio->ct = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void bio_write(opj_bio_t *bio, int v, int n) {
|
||||||
|
int i;
|
||||||
|
for (i = n - 1; i >= 0; i--) {
|
||||||
|
bio_putbit(bio, (v >> i) & 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int bio_read(opj_bio_t *bio, int n) {
|
||||||
|
int i, v;
|
||||||
|
v = 0;
|
||||||
|
for (i = n - 1; i >= 0; i--) {
|
||||||
|
v += bio_getbit(bio) << i;
|
||||||
|
}
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
|
||||||
|
int bio_flush(opj_bio_t *bio) {
|
||||||
|
bio->ct = 0;
|
||||||
|
if (bio_byteout(bio)) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if (bio->ct == 7) {
|
||||||
|
bio->ct = 0;
|
||||||
|
if (bio_byteout(bio)) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int bio_inalign(opj_bio_t *bio) {
|
||||||
|
bio->ct = 0;
|
||||||
|
if ((bio->buf & 0xff) == 0xff) {
|
||||||
|
if (bio_bytein(bio)) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
bio->ct = 0;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
125
extern/libopenjpeg/bio.h
vendored
Normal file
125
extern/libopenjpeg/bio.h
vendored
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __BIO_H
|
||||||
|
#define __BIO_H
|
||||||
|
/**
|
||||||
|
@file bio.h
|
||||||
|
@brief Implementation of an individual bit input-output (BIO)
|
||||||
|
|
||||||
|
The functions in BIO.C have for goal to realize an individual bit input - output.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup BIO BIO - Individual bit input-output stream */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Individual bit input-output stream (BIO)
|
||||||
|
*/
|
||||||
|
typedef struct opj_bio {
|
||||||
|
/** pointer to the start of the buffer */
|
||||||
|
unsigned char *start;
|
||||||
|
/** pointer to the end of the buffer */
|
||||||
|
unsigned char *end;
|
||||||
|
/** pointer to the present position in the buffer */
|
||||||
|
unsigned char *bp;
|
||||||
|
/** temporary place where each byte is read or written */
|
||||||
|
unsigned int buf;
|
||||||
|
/** coder : number of bits free to write. decoder : number of bits read */
|
||||||
|
int ct;
|
||||||
|
} opj_bio_t;
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Create a new BIO handle
|
||||||
|
@return Returns a new BIO handle if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_bio_t* bio_create(void);
|
||||||
|
/**
|
||||||
|
Destroy a previously created BIO handle
|
||||||
|
@param bio BIO handle to destroy
|
||||||
|
*/
|
||||||
|
void bio_destroy(opj_bio_t *bio);
|
||||||
|
/**
|
||||||
|
Number of bytes written.
|
||||||
|
@param bio BIO handle
|
||||||
|
@return Returns the number of bytes written
|
||||||
|
*/
|
||||||
|
int bio_numbytes(opj_bio_t *bio);
|
||||||
|
/**
|
||||||
|
Init encoder
|
||||||
|
@param bio BIO handle
|
||||||
|
@param bp Output buffer
|
||||||
|
@param len Output buffer length
|
||||||
|
*/
|
||||||
|
void bio_init_enc(opj_bio_t *bio, unsigned char *bp, int len);
|
||||||
|
/**
|
||||||
|
Init decoder
|
||||||
|
@param bio BIO handle
|
||||||
|
@param bp Input buffer
|
||||||
|
@param len Input buffer length
|
||||||
|
*/
|
||||||
|
void bio_init_dec(opj_bio_t *bio, unsigned char *bp, int len);
|
||||||
|
/**
|
||||||
|
Write bits
|
||||||
|
@param bio BIO handle
|
||||||
|
@param v Value of bits
|
||||||
|
@param n Number of bits to write
|
||||||
|
*/
|
||||||
|
void bio_write(opj_bio_t *bio, int v, int n);
|
||||||
|
/**
|
||||||
|
Read bits
|
||||||
|
@param bio BIO handle
|
||||||
|
@param n Number of bits to read
|
||||||
|
@return Returns the corresponding read number
|
||||||
|
*/
|
||||||
|
int bio_read(opj_bio_t *bio, int n);
|
||||||
|
/**
|
||||||
|
Flush bits
|
||||||
|
@param bio BIO handle
|
||||||
|
@return Returns 1 if successful, returns 0 otherwise
|
||||||
|
*/
|
||||||
|
int bio_flush(opj_bio_t *bio);
|
||||||
|
/**
|
||||||
|
Passes the ending bits (coming from flushing)
|
||||||
|
@param bio BIO handle
|
||||||
|
@return Returns 1 if successful, returns 0 otherwise
|
||||||
|
*/
|
||||||
|
int bio_inalign(opj_bio_t *bio);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __BIO_H */
|
||||||
|
|
||||||
191
extern/libopenjpeg/cio.c
vendored
Normal file
191
extern/libopenjpeg/cio.c
vendored
Normal file
@@ -0,0 +1,191 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
opj_cio_t* OPJ_CALLCONV opj_cio_open(opj_common_ptr cinfo, unsigned char *buffer, int length) {
|
||||||
|
opj_cp_t *cp = NULL;
|
||||||
|
opj_cio_t *cio = (opj_cio_t*)opj_malloc(sizeof(opj_cio_t));
|
||||||
|
if(!cio) return NULL;
|
||||||
|
cio->cinfo = cinfo;
|
||||||
|
if(buffer && length) {
|
||||||
|
/* wrap a user buffer containing the encoded image */
|
||||||
|
cio->openmode = OPJ_STREAM_READ;
|
||||||
|
cio->buffer = buffer;
|
||||||
|
cio->length = length;
|
||||||
|
}
|
||||||
|
else if(!buffer && !length && cinfo) {
|
||||||
|
/* allocate a buffer for the encoded image */
|
||||||
|
cio->openmode = OPJ_STREAM_WRITE;
|
||||||
|
switch(cinfo->codec_format) {
|
||||||
|
case CODEC_J2K:
|
||||||
|
cp = ((opj_j2k_t*)cinfo->j2k_handle)->cp;
|
||||||
|
break;
|
||||||
|
case CODEC_JP2:
|
||||||
|
cp = ((opj_jp2_t*)cinfo->jp2_handle)->j2k->cp;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
opj_free(cio);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
cio->length = (unsigned int) (0.1625 * cp->img_size + 2000); /* 0.1625 = 1.3/8 and 2000 bytes as a minimum for headers */
|
||||||
|
cio->buffer = (unsigned char *)opj_malloc(cio->length);
|
||||||
|
if(!cio->buffer) {
|
||||||
|
opj_event_msg(cio->cinfo, EVT_ERROR, "Error allocating memory for compressed bitstream\n");
|
||||||
|
opj_free(cio);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
opj_free(cio);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Initialize byte IO */
|
||||||
|
cio->start = cio->buffer;
|
||||||
|
cio->end = cio->buffer + cio->length;
|
||||||
|
cio->bp = cio->buffer;
|
||||||
|
|
||||||
|
return cio;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OPJ_CALLCONV opj_cio_close(opj_cio_t *cio) {
|
||||||
|
if(cio) {
|
||||||
|
if(cio->openmode == OPJ_STREAM_WRITE) {
|
||||||
|
/* destroy the allocated buffer */
|
||||||
|
opj_free(cio->buffer);
|
||||||
|
}
|
||||||
|
/* destroy the cio */
|
||||||
|
opj_free(cio);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get position in byte stream.
|
||||||
|
*/
|
||||||
|
int OPJ_CALLCONV cio_tell(opj_cio_t *cio) {
|
||||||
|
return cio->bp - cio->start;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set position in byte stream.
|
||||||
|
*
|
||||||
|
* pos : position, in number of bytes, from the beginning of the stream
|
||||||
|
*/
|
||||||
|
void OPJ_CALLCONV cio_seek(opj_cio_t *cio, int pos) {
|
||||||
|
cio->bp = cio->start + pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Number of bytes left before the end of the stream.
|
||||||
|
*/
|
||||||
|
int cio_numbytesleft(opj_cio_t *cio) {
|
||||||
|
return cio->end - cio->bp;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get pointer to the current position in the stream.
|
||||||
|
*/
|
||||||
|
unsigned char *cio_getbp(opj_cio_t *cio) {
|
||||||
|
return cio->bp;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Write a byte.
|
||||||
|
*/
|
||||||
|
bool cio_byteout(opj_cio_t *cio, unsigned char v) {
|
||||||
|
if (cio->bp >= cio->end) {
|
||||||
|
opj_event_msg(cio->cinfo, EVT_ERROR, "write error\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
*cio->bp++ = v;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read a byte.
|
||||||
|
*/
|
||||||
|
unsigned char cio_bytein(opj_cio_t *cio) {
|
||||||
|
if (cio->bp >= cio->end) {
|
||||||
|
opj_event_msg(cio->cinfo, EVT_ERROR, "read error: passed the end of the codestream (start = %d, current = %d, end = %d\n", cio->start, cio->bp, cio->end);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return *cio->bp++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Write some bytes.
|
||||||
|
*
|
||||||
|
* v : value to write
|
||||||
|
* n : number of bytes to write
|
||||||
|
*/
|
||||||
|
unsigned int cio_write(opj_cio_t *cio, unsigned int v, int n) {
|
||||||
|
int i;
|
||||||
|
for (i = n - 1; i >= 0; i--) {
|
||||||
|
if( !cio_byteout(cio, (unsigned char) ((v >> (i << 3)) & 0xff)) )
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read some bytes.
|
||||||
|
*
|
||||||
|
* n : number of bytes to read
|
||||||
|
*
|
||||||
|
* return : value of the n bytes read
|
||||||
|
*/
|
||||||
|
unsigned int cio_read(opj_cio_t *cio, int n) {
|
||||||
|
int i;
|
||||||
|
unsigned int v;
|
||||||
|
v = 0;
|
||||||
|
for (i = n - 1; i >= 0; i--) {
|
||||||
|
v += cio_bytein(cio) << (i << 3);
|
||||||
|
}
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Skip some bytes.
|
||||||
|
*
|
||||||
|
* n : number of bytes to skip
|
||||||
|
*/
|
||||||
|
void cio_skip(opj_cio_t *cio, int n) {
|
||||||
|
cio->bp += n;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
86
extern/libopenjpeg/cio.h
vendored
Normal file
86
extern/libopenjpeg/cio.h
vendored
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __CIO_H
|
||||||
|
#define __CIO_H
|
||||||
|
/**
|
||||||
|
@file cio.h
|
||||||
|
@brief Implementation of a byte input-output process (CIO)
|
||||||
|
|
||||||
|
The functions in CIO.C have for goal to realize a byte input / output process.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup CIO CIO - byte input-output stream */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Exported functions (see also openjpeg.h) */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Number of bytes left before the end of the stream
|
||||||
|
@param cio CIO handle
|
||||||
|
@return Returns the number of bytes before the end of the stream
|
||||||
|
*/
|
||||||
|
int cio_numbytesleft(opj_cio_t *cio);
|
||||||
|
/**
|
||||||
|
Get pointer to the current position in the stream
|
||||||
|
@param cio CIO handle
|
||||||
|
@return Returns a pointer to the current position
|
||||||
|
*/
|
||||||
|
unsigned char *cio_getbp(opj_cio_t *cio);
|
||||||
|
/**
|
||||||
|
Write some bytes
|
||||||
|
@param cio CIO handle
|
||||||
|
@param v Value to write
|
||||||
|
@param n Number of bytes to write
|
||||||
|
@return Returns the number of bytes written or 0 if an error occured
|
||||||
|
*/
|
||||||
|
unsigned int cio_write(opj_cio_t *cio, unsigned int v, int n);
|
||||||
|
/**
|
||||||
|
Read some bytes
|
||||||
|
@param cio CIO handle
|
||||||
|
@param n Number of bytes to read
|
||||||
|
@return Returns the value of the n bytes read
|
||||||
|
*/
|
||||||
|
unsigned int cio_read(opj_cio_t *cio, int n);
|
||||||
|
/**
|
||||||
|
Skip some bytes
|
||||||
|
@param cio CIO handle
|
||||||
|
@param n Number of bytes to skip
|
||||||
|
*/
|
||||||
|
void cio_skip(opj_cio_t *cio, int n);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __CIO_H */
|
||||||
|
|
||||||
825
extern/libopenjpeg/dwt.c
vendored
Normal file
825
extern/libopenjpeg/dwt.c
vendored
Normal file
@@ -0,0 +1,825 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* Copyright (c) 2007, Jonathan Ballard <dzonatas@dzonux.net>
|
||||||
|
* Copyright (c) 2007, Callum Lerwick <seg@haxxed.com>
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef __SSE__
|
||||||
|
#include <xmmintrin.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "opj_includes.h"
|
||||||
|
|
||||||
|
/** @defgroup DWT DWT - Implementation of a discrete wavelet transform */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
#define WS(i) v->mem[(i)*2]
|
||||||
|
#define WD(i) v->mem[(1+(i)*2)]
|
||||||
|
|
||||||
|
/** @name Local data structures */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
typedef struct dwt_local {
|
||||||
|
int* mem;
|
||||||
|
int dn;
|
||||||
|
int sn;
|
||||||
|
int cas;
|
||||||
|
} dwt_t;
|
||||||
|
|
||||||
|
typedef union {
|
||||||
|
float f[4];
|
||||||
|
} v4;
|
||||||
|
|
||||||
|
typedef struct v4dwt_local {
|
||||||
|
v4* wavelet ;
|
||||||
|
int dn ;
|
||||||
|
int sn ;
|
||||||
|
int cas ;
|
||||||
|
} v4dwt_t ;
|
||||||
|
|
||||||
|
static const float dwt_alpha = 1.586134342f; // 12994
|
||||||
|
static const float dwt_beta = 0.052980118f; // 434
|
||||||
|
static const float dwt_gamma = -0.882911075f; // -7233
|
||||||
|
static const float dwt_delta = -0.443506852f; // -3633
|
||||||
|
|
||||||
|
static const float K = 1.230174105f; // 10078
|
||||||
|
/* FIXME: What is this constant? */
|
||||||
|
static const float c13318 = 1.625732422f;
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Virtual function type for wavelet transform in 1-D
|
||||||
|
*/
|
||||||
|
typedef void (*DWT1DFN)(dwt_t* v);
|
||||||
|
|
||||||
|
/** @name Local static functions */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Forward lazy transform (horizontal)
|
||||||
|
*/
|
||||||
|
static void dwt_deinterleave_h(int *a, int *b, int dn, int sn, int cas);
|
||||||
|
/**
|
||||||
|
Forward lazy transform (vertical)
|
||||||
|
*/
|
||||||
|
static void dwt_deinterleave_v(int *a, int *b, int dn, int sn, int x, int cas);
|
||||||
|
/**
|
||||||
|
Inverse lazy transform (horizontal)
|
||||||
|
*/
|
||||||
|
static void dwt_interleave_h(dwt_t* h, int *a);
|
||||||
|
/**
|
||||||
|
Inverse lazy transform (vertical)
|
||||||
|
*/
|
||||||
|
static void dwt_interleave_v(dwt_t* v, int *a, int x);
|
||||||
|
/**
|
||||||
|
Forward 5-3 wavelet transform in 1-D
|
||||||
|
*/
|
||||||
|
static void dwt_encode_1(int *a, int dn, int sn, int cas);
|
||||||
|
/**
|
||||||
|
Inverse 5-3 wavelet transform in 1-D
|
||||||
|
*/
|
||||||
|
static void dwt_decode_1(dwt_t *v);
|
||||||
|
/**
|
||||||
|
Forward 9-7 wavelet transform in 1-D
|
||||||
|
*/
|
||||||
|
static void dwt_encode_1_real(int *a, int dn, int sn, int cas);
|
||||||
|
/**
|
||||||
|
Explicit calculation of the Quantization Stepsizes
|
||||||
|
*/
|
||||||
|
static void dwt_encode_stepsize(int stepsize, int numbps, opj_stepsize_t *bandno_stepsize);
|
||||||
|
/**
|
||||||
|
Inverse wavelet transform in 2-D.
|
||||||
|
*/
|
||||||
|
static void dwt_decode_tile(opj_tcd_tilecomp_t* tilec, int i, DWT1DFN fn);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#define S(i) a[(i)*2]
|
||||||
|
#define D(i) a[(1+(i)*2)]
|
||||||
|
#define S_(i) ((i)<0?S(0):((i)>=sn?S(sn-1):S(i)))
|
||||||
|
#define D_(i) ((i)<0?D(0):((i)>=dn?D(dn-1):D(i)))
|
||||||
|
/* new */
|
||||||
|
#define SS_(i) ((i)<0?S(0):((i)>=dn?S(dn-1):S(i)))
|
||||||
|
#define DD_(i) ((i)<0?D(0):((i)>=sn?D(sn-1):D(i)))
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* This table contains the norms of the 5-3 wavelets for different bands. */
|
||||||
|
/* </summary> */
|
||||||
|
static const double dwt_norms[4][10] = {
|
||||||
|
{1.000, 1.500, 2.750, 5.375, 10.68, 21.34, 42.67, 85.33, 170.7, 341.3},
|
||||||
|
{1.038, 1.592, 2.919, 5.703, 11.33, 22.64, 45.25, 90.48, 180.9},
|
||||||
|
{1.038, 1.592, 2.919, 5.703, 11.33, 22.64, 45.25, 90.48, 180.9},
|
||||||
|
{.7186, .9218, 1.586, 3.043, 6.019, 12.01, 24.00, 47.97, 95.93}
|
||||||
|
};
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* This table contains the norms of the 9-7 wavelets for different bands. */
|
||||||
|
/* </summary> */
|
||||||
|
static const double dwt_norms_real[4][10] = {
|
||||||
|
{1.000, 1.965, 4.177, 8.403, 16.90, 33.84, 67.69, 135.3, 270.6, 540.9},
|
||||||
|
{2.022, 3.989, 8.355, 17.04, 34.27, 68.63, 137.3, 274.6, 549.0},
|
||||||
|
{2.022, 3.989, 8.355, 17.04, 34.27, 68.63, 137.3, 274.6, 549.0},
|
||||||
|
{2.080, 3.865, 8.307, 17.18, 34.71, 69.59, 139.3, 278.6, 557.2}
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
local functions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Forward lazy transform (horizontal). */
|
||||||
|
/* </summary> */
|
||||||
|
static void dwt_deinterleave_h(int *a, int *b, int dn, int sn, int cas) {
|
||||||
|
int i;
|
||||||
|
for (i=0; i<sn; i++) b[i]=a[2*i+cas];
|
||||||
|
for (i=0; i<dn; i++) b[sn+i]=a[(2*i+1-cas)];
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Forward lazy transform (vertical). */
|
||||||
|
/* </summary> */
|
||||||
|
static void dwt_deinterleave_v(int *a, int *b, int dn, int sn, int x, int cas) {
|
||||||
|
int i;
|
||||||
|
for (i=0; i<sn; i++) b[i*x]=a[2*i+cas];
|
||||||
|
for (i=0; i<dn; i++) b[(sn+i)*x]=a[(2*i+1-cas)];
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse lazy transform (horizontal). */
|
||||||
|
/* </summary> */
|
||||||
|
static void dwt_interleave_h(dwt_t* h, int *a) {
|
||||||
|
int *ai = a;
|
||||||
|
int *bi = h->mem + h->cas;
|
||||||
|
int i = h->sn;
|
||||||
|
while( i-- ) {
|
||||||
|
*bi = *(ai++);
|
||||||
|
bi += 2;
|
||||||
|
}
|
||||||
|
ai = a + h->sn;
|
||||||
|
bi = h->mem + 1 - h->cas;
|
||||||
|
i = h->dn ;
|
||||||
|
while( i-- ) {
|
||||||
|
*bi = *(ai++);
|
||||||
|
bi += 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse lazy transform (vertical). */
|
||||||
|
/* </summary> */
|
||||||
|
static void dwt_interleave_v(dwt_t* v, int *a, int x) {
|
||||||
|
int *ai = a;
|
||||||
|
int *bi = v->mem + v->cas;
|
||||||
|
int i = v->sn;
|
||||||
|
while( i-- ) {
|
||||||
|
*bi = *ai;
|
||||||
|
bi += 2;
|
||||||
|
ai += x;
|
||||||
|
}
|
||||||
|
ai = a + (v->sn * x);
|
||||||
|
bi = v->mem + 1 - v->cas;
|
||||||
|
i = v->dn ;
|
||||||
|
while( i-- ) {
|
||||||
|
*bi = *ai;
|
||||||
|
bi += 2;
|
||||||
|
ai += x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Forward 5-3 wavelet transform in 1-D. */
|
||||||
|
/* </summary> */
|
||||||
|
static void dwt_encode_1(int *a, int dn, int sn, int cas) {
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!cas) {
|
||||||
|
if ((dn > 0) || (sn > 1)) { /* NEW : CASE ONE ELEMENT */
|
||||||
|
for (i = 0; i < dn; i++) D(i) -= (S_(i) + S_(i + 1)) >> 1;
|
||||||
|
for (i = 0; i < sn; i++) S(i) += (D_(i - 1) + D_(i) + 2) >> 2;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (!sn && dn == 1) /* NEW : CASE ONE ELEMENT */
|
||||||
|
S(0) *= 2;
|
||||||
|
else {
|
||||||
|
for (i = 0; i < dn; i++) S(i) -= (DD_(i) + DD_(i - 1)) >> 1;
|
||||||
|
for (i = 0; i < sn; i++) D(i) += (SS_(i) + SS_(i + 1) + 2) >> 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse 5-3 wavelet transform in 1-D. */
|
||||||
|
/* </summary> */
|
||||||
|
static void dwt_decode_1_(int *a, int dn, int sn, int cas) {
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!cas) {
|
||||||
|
if ((dn > 0) || (sn > 1)) { /* NEW : CASE ONE ELEMENT */
|
||||||
|
for (i = 0; i < sn; i++) S(i) -= (D_(i - 1) + D_(i) + 2) >> 2;
|
||||||
|
for (i = 0; i < dn; i++) D(i) += (S_(i) + S_(i + 1)) >> 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (!sn && dn == 1) /* NEW : CASE ONE ELEMENT */
|
||||||
|
S(0) /= 2;
|
||||||
|
else {
|
||||||
|
for (i = 0; i < sn; i++) D(i) -= (SS_(i) + SS_(i + 1) + 2) >> 2;
|
||||||
|
for (i = 0; i < dn; i++) S(i) += (DD_(i) + DD_(i - 1)) >> 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse 5-3 wavelet transform in 1-D. */
|
||||||
|
/* </summary> */
|
||||||
|
static void dwt_decode_1(dwt_t *v) {
|
||||||
|
dwt_decode_1_(v->mem, v->dn, v->sn, v->cas);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Forward 9-7 wavelet transform in 1-D. */
|
||||||
|
/* </summary> */
|
||||||
|
static void dwt_encode_1_real(int *a, int dn, int sn, int cas) {
|
||||||
|
int i;
|
||||||
|
if (!cas) {
|
||||||
|
if ((dn > 0) || (sn > 1)) { /* NEW : CASE ONE ELEMENT */
|
||||||
|
for (i = 0; i < dn; i++)
|
||||||
|
D(i) -= fix_mul(S_(i) + S_(i + 1), 12993);
|
||||||
|
for (i = 0; i < sn; i++)
|
||||||
|
S(i) -= fix_mul(D_(i - 1) + D_(i), 434);
|
||||||
|
for (i = 0; i < dn; i++)
|
||||||
|
D(i) += fix_mul(S_(i) + S_(i + 1), 7233);
|
||||||
|
for (i = 0; i < sn; i++)
|
||||||
|
S(i) += fix_mul(D_(i - 1) + D_(i), 3633);
|
||||||
|
for (i = 0; i < dn; i++)
|
||||||
|
D(i) = fix_mul(D(i), 5038); /*5038 */
|
||||||
|
for (i = 0; i < sn; i++)
|
||||||
|
S(i) = fix_mul(S(i), 6659); /*6660 */
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if ((sn > 0) || (dn > 1)) { /* NEW : CASE ONE ELEMENT */
|
||||||
|
for (i = 0; i < dn; i++)
|
||||||
|
S(i) -= fix_mul(DD_(i) + DD_(i - 1), 12993);
|
||||||
|
for (i = 0; i < sn; i++)
|
||||||
|
D(i) -= fix_mul(SS_(i) + SS_(i + 1), 434);
|
||||||
|
for (i = 0; i < dn; i++)
|
||||||
|
S(i) += fix_mul(DD_(i) + DD_(i - 1), 7233);
|
||||||
|
for (i = 0; i < sn; i++)
|
||||||
|
D(i) += fix_mul(SS_(i) + SS_(i + 1), 3633);
|
||||||
|
for (i = 0; i < dn; i++)
|
||||||
|
S(i) = fix_mul(S(i), 5038); /*5038 */
|
||||||
|
for (i = 0; i < sn; i++)
|
||||||
|
D(i) = fix_mul(D(i), 6659); /*6660 */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void dwt_encode_stepsize(int stepsize, int numbps, opj_stepsize_t *bandno_stepsize) {
|
||||||
|
int p, n;
|
||||||
|
p = int_floorlog2(stepsize) - 13;
|
||||||
|
n = 11 - int_floorlog2(stepsize);
|
||||||
|
bandno_stepsize->mant = (n < 0 ? stepsize >> -n : stepsize << n) & 0x7ff;
|
||||||
|
bandno_stepsize->expn = numbps - p;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
DWT interface
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Forward 5-3 wavelet transform in 2-D. */
|
||||||
|
/* </summary> */
|
||||||
|
void dwt_encode(opj_tcd_tilecomp_t * tilec) {
|
||||||
|
int i, j, k;
|
||||||
|
int *a = NULL;
|
||||||
|
int *aj = NULL;
|
||||||
|
int *bj = NULL;
|
||||||
|
int w, l;
|
||||||
|
|
||||||
|
w = tilec->x1-tilec->x0;
|
||||||
|
l = tilec->numresolutions-1;
|
||||||
|
a = tilec->data;
|
||||||
|
|
||||||
|
for (i = 0; i < l; i++) {
|
||||||
|
int rw; /* width of the resolution level computed */
|
||||||
|
int rh; /* height of the resolution level computed */
|
||||||
|
int rw1; /* width of the resolution level once lower than computed one */
|
||||||
|
int rh1; /* height of the resolution level once lower than computed one */
|
||||||
|
int cas_col; /* 0 = non inversion on horizontal filtering 1 = inversion between low-pass and high-pass filtering */
|
||||||
|
int cas_row; /* 0 = non inversion on vertical filtering 1 = inversion between low-pass and high-pass filtering */
|
||||||
|
int dn, sn;
|
||||||
|
|
||||||
|
rw = tilec->resolutions[l - i].x1 - tilec->resolutions[l - i].x0;
|
||||||
|
rh = tilec->resolutions[l - i].y1 - tilec->resolutions[l - i].y0;
|
||||||
|
rw1= tilec->resolutions[l - i - 1].x1 - tilec->resolutions[l - i - 1].x0;
|
||||||
|
rh1= tilec->resolutions[l - i - 1].y1 - tilec->resolutions[l - i - 1].y0;
|
||||||
|
|
||||||
|
cas_row = tilec->resolutions[l - i].x0 % 2;
|
||||||
|
cas_col = tilec->resolutions[l - i].y0 % 2;
|
||||||
|
|
||||||
|
sn = rh1;
|
||||||
|
dn = rh - rh1;
|
||||||
|
bj = (int*)opj_malloc(rh * sizeof(int));
|
||||||
|
for (j = 0; j < rw; j++) {
|
||||||
|
aj = a + j;
|
||||||
|
for (k = 0; k < rh; k++) bj[k] = aj[k*w];
|
||||||
|
dwt_encode_1(bj, dn, sn, cas_col);
|
||||||
|
dwt_deinterleave_v(bj, aj, dn, sn, w, cas_col);
|
||||||
|
}
|
||||||
|
opj_free(bj);
|
||||||
|
|
||||||
|
sn = rw1;
|
||||||
|
dn = rw - rw1;
|
||||||
|
bj = (int*)opj_malloc(rw * sizeof(int));
|
||||||
|
for (j = 0; j < rh; j++) {
|
||||||
|
aj = a + j * w;
|
||||||
|
for (k = 0; k < rw; k++) bj[k] = aj[k];
|
||||||
|
dwt_encode_1(bj, dn, sn, cas_row);
|
||||||
|
dwt_deinterleave_h(bj, aj, dn, sn, cas_row);
|
||||||
|
}
|
||||||
|
opj_free(bj);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse 5-3 wavelet transform in 2-D. */
|
||||||
|
/* </summary> */
|
||||||
|
void dwt_decode(opj_tcd_tilecomp_t* tilec, int numres) {
|
||||||
|
dwt_decode_tile(tilec, numres, &dwt_decode_1);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Get gain of 5-3 wavelet transform. */
|
||||||
|
/* </summary> */
|
||||||
|
int dwt_getgain(int orient) {
|
||||||
|
if (orient == 0)
|
||||||
|
return 0;
|
||||||
|
if (orient == 1 || orient == 2)
|
||||||
|
return 1;
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Get norm of 5-3 wavelet. */
|
||||||
|
/* </summary> */
|
||||||
|
double dwt_getnorm(int level, int orient) {
|
||||||
|
return dwt_norms[orient][level];
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Forward 9-7 wavelet transform in 2-D. */
|
||||||
|
/* </summary> */
|
||||||
|
|
||||||
|
void dwt_encode_real(opj_tcd_tilecomp_t * tilec) {
|
||||||
|
int i, j, k;
|
||||||
|
int *a = NULL;
|
||||||
|
int *aj = NULL;
|
||||||
|
int *bj = NULL;
|
||||||
|
int w, l;
|
||||||
|
|
||||||
|
w = tilec->x1-tilec->x0;
|
||||||
|
l = tilec->numresolutions-1;
|
||||||
|
a = tilec->data;
|
||||||
|
|
||||||
|
for (i = 0; i < l; i++) {
|
||||||
|
int rw; /* width of the resolution level computed */
|
||||||
|
int rh; /* height of the resolution level computed */
|
||||||
|
int rw1; /* width of the resolution level once lower than computed one */
|
||||||
|
int rh1; /* height of the resolution level once lower than computed one */
|
||||||
|
int cas_col; /* 0 = non inversion on horizontal filtering 1 = inversion between low-pass and high-pass filtering */
|
||||||
|
int cas_row; /* 0 = non inversion on vertical filtering 1 = inversion between low-pass and high-pass filtering */
|
||||||
|
int dn, sn;
|
||||||
|
|
||||||
|
rw = tilec->resolutions[l - i].x1 - tilec->resolutions[l - i].x0;
|
||||||
|
rh = tilec->resolutions[l - i].y1 - tilec->resolutions[l - i].y0;
|
||||||
|
rw1= tilec->resolutions[l - i - 1].x1 - tilec->resolutions[l - i - 1].x0;
|
||||||
|
rh1= tilec->resolutions[l - i - 1].y1 - tilec->resolutions[l - i - 1].y0;
|
||||||
|
|
||||||
|
cas_row = tilec->resolutions[l - i].x0 % 2;
|
||||||
|
cas_col = tilec->resolutions[l - i].y0 % 2;
|
||||||
|
|
||||||
|
sn = rh1;
|
||||||
|
dn = rh - rh1;
|
||||||
|
bj = (int*)opj_malloc(rh * sizeof(int));
|
||||||
|
for (j = 0; j < rw; j++) {
|
||||||
|
aj = a + j;
|
||||||
|
for (k = 0; k < rh; k++) bj[k] = aj[k*w];
|
||||||
|
dwt_encode_1_real(bj, dn, sn, cas_col);
|
||||||
|
dwt_deinterleave_v(bj, aj, dn, sn, w, cas_col);
|
||||||
|
}
|
||||||
|
opj_free(bj);
|
||||||
|
|
||||||
|
sn = rw1;
|
||||||
|
dn = rw - rw1;
|
||||||
|
bj = (int*)opj_malloc(rw * sizeof(int));
|
||||||
|
for (j = 0; j < rh; j++) {
|
||||||
|
aj = a + j * w;
|
||||||
|
for (k = 0; k < rw; k++) bj[k] = aj[k];
|
||||||
|
dwt_encode_1_real(bj, dn, sn, cas_row);
|
||||||
|
dwt_deinterleave_h(bj, aj, dn, sn, cas_row);
|
||||||
|
}
|
||||||
|
opj_free(bj);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Get gain of 9-7 wavelet transform. */
|
||||||
|
/* </summary> */
|
||||||
|
int dwt_getgain_real(int orient) {
|
||||||
|
(void)orient;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Get norm of 9-7 wavelet. */
|
||||||
|
/* </summary> */
|
||||||
|
double dwt_getnorm_real(int level, int orient) {
|
||||||
|
return dwt_norms_real[orient][level];
|
||||||
|
}
|
||||||
|
|
||||||
|
void dwt_calc_explicit_stepsizes(opj_tccp_t * tccp, int prec) {
|
||||||
|
int numbands, bandno;
|
||||||
|
numbands = 3 * tccp->numresolutions - 2;
|
||||||
|
for (bandno = 0; bandno < numbands; bandno++) {
|
||||||
|
double stepsize;
|
||||||
|
int resno, level, orient, gain;
|
||||||
|
|
||||||
|
resno = (bandno == 0) ? 0 : ((bandno - 1) / 3 + 1);
|
||||||
|
orient = (bandno == 0) ? 0 : ((bandno - 1) % 3 + 1);
|
||||||
|
level = tccp->numresolutions - 1 - resno;
|
||||||
|
gain = (tccp->qmfbid == 0) ? 0 : ((orient == 0) ? 0 : (((orient == 1) || (orient == 2)) ? 1 : 2));
|
||||||
|
if (tccp->qntsty == J2K_CCP_QNTSTY_NOQNT) {
|
||||||
|
stepsize = 1.0;
|
||||||
|
} else {
|
||||||
|
double norm = dwt_norms_real[orient][level];
|
||||||
|
stepsize = (1 << (gain)) / norm;
|
||||||
|
}
|
||||||
|
dwt_encode_stepsize((int) floor(stepsize * 8192.0), prec + gain, &tccp->stepsizes[bandno]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Determine maximum computed resolution level for inverse wavelet transform */
|
||||||
|
/* </summary> */
|
||||||
|
static int dwt_decode_max_resolution(opj_tcd_resolution_t* restrict r, int i) {
|
||||||
|
int mr = 1;
|
||||||
|
int w;
|
||||||
|
while( --i ) {
|
||||||
|
r++;
|
||||||
|
if( mr < ( w = r->x1 - r->x0 ) )
|
||||||
|
mr = w ;
|
||||||
|
if( mr < ( w = r->y1 - r->y0 ) )
|
||||||
|
mr = w ;
|
||||||
|
}
|
||||||
|
return mr ;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse wavelet transform in 2-D. */
|
||||||
|
/* </summary> */
|
||||||
|
static void dwt_decode_tile(opj_tcd_tilecomp_t* tilec, int numres, DWT1DFN dwt_1D) {
|
||||||
|
dwt_t h;
|
||||||
|
dwt_t v;
|
||||||
|
|
||||||
|
opj_tcd_resolution_t* tr = tilec->resolutions;
|
||||||
|
|
||||||
|
int rw = tr->x1 - tr->x0; /* width of the resolution level computed */
|
||||||
|
int rh = tr->y1 - tr->y0; /* height of the resolution level computed */
|
||||||
|
|
||||||
|
int w = tilec->x1 - tilec->x0;
|
||||||
|
|
||||||
|
h.mem = opj_aligned_malloc(dwt_decode_max_resolution(tr, numres) * sizeof(int));
|
||||||
|
v.mem = h.mem;
|
||||||
|
|
||||||
|
while( --numres) {
|
||||||
|
int * restrict tiledp = tilec->data;
|
||||||
|
int j;
|
||||||
|
|
||||||
|
++tr;
|
||||||
|
h.sn = rw;
|
||||||
|
v.sn = rh;
|
||||||
|
|
||||||
|
rw = tr->x1 - tr->x0;
|
||||||
|
rh = tr->y1 - tr->y0;
|
||||||
|
|
||||||
|
h.dn = rw - h.sn;
|
||||||
|
h.cas = tr->x0 % 2;
|
||||||
|
|
||||||
|
for(j = 0; j < rh; ++j) {
|
||||||
|
dwt_interleave_h(&h, &tiledp[j*w]);
|
||||||
|
(dwt_1D)(&h);
|
||||||
|
memcpy(&tiledp[j*w], h.mem, rw * sizeof(int));
|
||||||
|
}
|
||||||
|
|
||||||
|
v.dn = rh - v.sn;
|
||||||
|
v.cas = tr->y0 % 2;
|
||||||
|
|
||||||
|
for(j = 0; j < rw; ++j){
|
||||||
|
int k;
|
||||||
|
dwt_interleave_v(&v, &tiledp[j], w);
|
||||||
|
(dwt_1D)(&v);
|
||||||
|
for(k = 0; k < rh; ++k) {
|
||||||
|
tiledp[k * w + j] = v.mem[k];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
opj_aligned_free(h.mem);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void v4dwt_interleave_h(v4dwt_t* restrict w, float* restrict a, int x, int size){
|
||||||
|
float* restrict bi = (float*) (w->wavelet + w->cas);
|
||||||
|
int count = w->sn;
|
||||||
|
int i, k;
|
||||||
|
for(k = 0; k < 2; ++k){
|
||||||
|
for(i = 0; i < count; ++i){
|
||||||
|
int j = i;
|
||||||
|
bi[i*8 ] = a[j];
|
||||||
|
j += x;
|
||||||
|
if(j > size) continue;
|
||||||
|
bi[i*8 + 1] = a[j];
|
||||||
|
j += x;
|
||||||
|
if(j > size) continue;
|
||||||
|
bi[i*8 + 2] = a[j];
|
||||||
|
j += x;
|
||||||
|
if(j > size) continue;
|
||||||
|
bi[i*8 + 3] = a[j];
|
||||||
|
}
|
||||||
|
bi = (float*) (w->wavelet + 1 - w->cas);
|
||||||
|
a += w->sn;
|
||||||
|
size -= w->sn;
|
||||||
|
count = w->dn;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void v4dwt_interleave_v(v4dwt_t* restrict v , float* restrict a , int x){
|
||||||
|
v4* restrict bi = v->wavelet + v->cas;
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < v->sn; ++i){
|
||||||
|
memcpy(&bi[i*2], &a[i*x], 4 * sizeof(float));
|
||||||
|
}
|
||||||
|
a += v->sn * x;
|
||||||
|
bi = v->wavelet + 1 - v->cas;
|
||||||
|
for(i = 0; i < v->dn; ++i){
|
||||||
|
memcpy(&bi[i*2], &a[i*x], 4 * sizeof(float));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef __SSE__
|
||||||
|
|
||||||
|
static void v4dwt_decode_step1_sse(v4* w, int count, const __m128 c){
|
||||||
|
__m128* restrict vw = (__m128*) w;
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < count; ++i){
|
||||||
|
__m128 tmp = vw[i*2];
|
||||||
|
vw[i*2] = tmp * c;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void v4dwt_decode_step2_sse(v4* l, v4* w, int k, int m, __m128 c){
|
||||||
|
__m128* restrict vl = (__m128*) l;
|
||||||
|
__m128* restrict vw = (__m128*) w;
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < m; ++i){
|
||||||
|
__m128 tmp1 = vl[ 0];
|
||||||
|
__m128 tmp2 = vw[-1];
|
||||||
|
__m128 tmp3 = vw[ 0];
|
||||||
|
vw[-1] = tmp2 + ((tmp1 + tmp3) * c);
|
||||||
|
vl = vw;
|
||||||
|
vw += 2;
|
||||||
|
}
|
||||||
|
if(m >= k){
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
c += c;
|
||||||
|
c *= vl[0];
|
||||||
|
for(; m < k; ++m){
|
||||||
|
__m128 tmp = vw[-1];
|
||||||
|
vw[-1] = tmp + c;
|
||||||
|
vw += 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static void v4dwt_decode_step1(v4* w, int count, const float c){
|
||||||
|
float* restrict fw = (float*) w;
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < count; ++i){
|
||||||
|
float tmp1 = fw[i*8 ];
|
||||||
|
float tmp2 = fw[i*8 + 1];
|
||||||
|
float tmp3 = fw[i*8 + 2];
|
||||||
|
float tmp4 = fw[i*8 + 3];
|
||||||
|
fw[i*8 ] = tmp1 * c;
|
||||||
|
fw[i*8 + 1] = tmp2 * c;
|
||||||
|
fw[i*8 + 2] = tmp3 * c;
|
||||||
|
fw[i*8 + 3] = tmp4 * c;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void v4dwt_decode_step2(v4* l, v4* w, int k, int m, float c){
|
||||||
|
float* restrict fl = (float*) l;
|
||||||
|
float* restrict fw = (float*) w;
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < m; ++i){
|
||||||
|
float tmp1_1 = fl[0];
|
||||||
|
float tmp1_2 = fl[1];
|
||||||
|
float tmp1_3 = fl[2];
|
||||||
|
float tmp1_4 = fl[3];
|
||||||
|
float tmp2_1 = fw[-4];
|
||||||
|
float tmp2_2 = fw[-3];
|
||||||
|
float tmp2_3 = fw[-2];
|
||||||
|
float tmp2_4 = fw[-1];
|
||||||
|
float tmp3_1 = fw[0];
|
||||||
|
float tmp3_2 = fw[1];
|
||||||
|
float tmp3_3 = fw[2];
|
||||||
|
float tmp3_4 = fw[3];
|
||||||
|
fw[-4] = tmp2_1 + ((tmp1_1 + tmp3_1) * c);
|
||||||
|
fw[-3] = tmp2_2 + ((tmp1_2 + tmp3_2) * c);
|
||||||
|
fw[-2] = tmp2_3 + ((tmp1_3 + tmp3_3) * c);
|
||||||
|
fw[-1] = tmp2_4 + ((tmp1_4 + tmp3_4) * c);
|
||||||
|
fl = fw;
|
||||||
|
fw += 8;
|
||||||
|
}
|
||||||
|
if(m < k){
|
||||||
|
float c1;
|
||||||
|
float c2;
|
||||||
|
float c3;
|
||||||
|
float c4;
|
||||||
|
c += c;
|
||||||
|
c1 = fl[0] * c;
|
||||||
|
c2 = fl[1] * c;
|
||||||
|
c3 = fl[2] * c;
|
||||||
|
c4 = fl[3] * c;
|
||||||
|
for(; m < k; ++m){
|
||||||
|
float tmp1 = fw[-4];
|
||||||
|
float tmp2 = fw[-3];
|
||||||
|
float tmp3 = fw[-2];
|
||||||
|
float tmp4 = fw[-1];
|
||||||
|
fw[-4] = tmp1 + c1;
|
||||||
|
fw[-3] = tmp2 + c2;
|
||||||
|
fw[-2] = tmp3 + c3;
|
||||||
|
fw[-1] = tmp4 + c4;
|
||||||
|
fw += 8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse 9-7 wavelet transform in 1-D. */
|
||||||
|
/* </summary> */
|
||||||
|
static void v4dwt_decode(v4dwt_t* restrict dwt){
|
||||||
|
int a, b;
|
||||||
|
if(dwt->cas == 0) {
|
||||||
|
if(!((dwt->dn > 0) || (dwt->sn > 1))){
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
a = 0;
|
||||||
|
b = 1;
|
||||||
|
}else{
|
||||||
|
if(!((dwt->sn > 0) || (dwt->dn > 1))) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
a = 1;
|
||||||
|
b = 0;
|
||||||
|
}
|
||||||
|
#ifdef __SSE__
|
||||||
|
v4dwt_decode_step1_sse(dwt->wavelet+a, dwt->sn, _mm_set1_ps(K));
|
||||||
|
v4dwt_decode_step1_sse(dwt->wavelet+b, dwt->dn, _mm_set1_ps(c13318));
|
||||||
|
v4dwt_decode_step2_sse(dwt->wavelet+b, dwt->wavelet+a+1, dwt->sn, int_min(dwt->sn, dwt->dn-a), _mm_set1_ps(dwt_delta));
|
||||||
|
v4dwt_decode_step2_sse(dwt->wavelet+a, dwt->wavelet+b+1, dwt->dn, int_min(dwt->dn, dwt->sn-b), _mm_set1_ps(dwt_gamma));
|
||||||
|
v4dwt_decode_step2_sse(dwt->wavelet+b, dwt->wavelet+a+1, dwt->sn, int_min(dwt->sn, dwt->dn-a), _mm_set1_ps(dwt_beta));
|
||||||
|
v4dwt_decode_step2_sse(dwt->wavelet+a, dwt->wavelet+b+1, dwt->dn, int_min(dwt->dn, dwt->sn-b), _mm_set1_ps(dwt_alpha));
|
||||||
|
#else
|
||||||
|
v4dwt_decode_step1(dwt->wavelet+a, dwt->sn, K);
|
||||||
|
v4dwt_decode_step1(dwt->wavelet+b, dwt->dn, c13318);
|
||||||
|
v4dwt_decode_step2(dwt->wavelet+b, dwt->wavelet+a+1, dwt->sn, int_min(dwt->sn, dwt->dn-a), dwt_delta);
|
||||||
|
v4dwt_decode_step2(dwt->wavelet+a, dwt->wavelet+b+1, dwt->dn, int_min(dwt->dn, dwt->sn-b), dwt_gamma);
|
||||||
|
v4dwt_decode_step2(dwt->wavelet+b, dwt->wavelet+a+1, dwt->sn, int_min(dwt->sn, dwt->dn-a), dwt_beta);
|
||||||
|
v4dwt_decode_step2(dwt->wavelet+a, dwt->wavelet+b+1, dwt->dn, int_min(dwt->dn, dwt->sn-b), dwt_alpha);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse 9-7 wavelet transform in 2-D. */
|
||||||
|
/* </summary> */
|
||||||
|
void dwt_decode_real(opj_tcd_tilecomp_t* restrict tilec, int numres){
|
||||||
|
v4dwt_t h;
|
||||||
|
v4dwt_t v;
|
||||||
|
|
||||||
|
opj_tcd_resolution_t* res = tilec->resolutions;
|
||||||
|
|
||||||
|
int rw = res->x1 - res->x0; /* width of the resolution level computed */
|
||||||
|
int rh = res->y1 - res->y0; /* height of the resolution level computed */
|
||||||
|
|
||||||
|
int w = tilec->x1 - tilec->x0;
|
||||||
|
|
||||||
|
h.wavelet = (v4*) opj_aligned_malloc((dwt_decode_max_resolution(res, numres)+5) * sizeof(v4));
|
||||||
|
v.wavelet = h.wavelet;
|
||||||
|
|
||||||
|
while( --numres) {
|
||||||
|
float * restrict aj = (float*) tilec->data;
|
||||||
|
int bufsize = (tilec->x1 - tilec->x0) * (tilec->y1 - tilec->y0);
|
||||||
|
int j;
|
||||||
|
|
||||||
|
h.sn = rw;
|
||||||
|
v.sn = rh;
|
||||||
|
|
||||||
|
++res;
|
||||||
|
|
||||||
|
rw = res->x1 - res->x0; /* width of the resolution level computed */
|
||||||
|
rh = res->y1 - res->y0; /* height of the resolution level computed */
|
||||||
|
|
||||||
|
h.dn = rw - h.sn;
|
||||||
|
h.cas = res->x0 % 2;
|
||||||
|
|
||||||
|
for(j = rh; j > 0; j -= 4){
|
||||||
|
v4dwt_interleave_h(&h, aj, w, bufsize);
|
||||||
|
v4dwt_decode(&h);
|
||||||
|
if(j >= 4){
|
||||||
|
int k;
|
||||||
|
for(k = rw; --k >= 0;){
|
||||||
|
aj[k ] = h.wavelet[k].f[0];
|
||||||
|
aj[k+w ] = h.wavelet[k].f[1];
|
||||||
|
aj[k+w*2] = h.wavelet[k].f[2];
|
||||||
|
aj[k+w*3] = h.wavelet[k].f[3];
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
int k;
|
||||||
|
for(k = rw; --k >= 0;){
|
||||||
|
switch(j) {
|
||||||
|
case 3: aj[k+w*2] = h.wavelet[k].f[2];
|
||||||
|
case 2: aj[k+w ] = h.wavelet[k].f[1];
|
||||||
|
case 1: aj[k ] = h.wavelet[k].f[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
aj += w*4;
|
||||||
|
bufsize -= w*4;
|
||||||
|
}
|
||||||
|
|
||||||
|
v.dn = rh - v.sn;
|
||||||
|
v.cas = res->y0 % 2;
|
||||||
|
|
||||||
|
aj = (float*) tilec->data;
|
||||||
|
for(j = rw; j > 0; j -= 4){
|
||||||
|
v4dwt_interleave_v(&v, aj, w);
|
||||||
|
v4dwt_decode(&v);
|
||||||
|
if(j >= 4){
|
||||||
|
int k;
|
||||||
|
for(k = 0; k < rh; ++k){
|
||||||
|
memcpy(&aj[k*w], &v.wavelet[k], 4 * sizeof(float));
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
int k;
|
||||||
|
for(k = 0; k < rh; ++k){
|
||||||
|
memcpy(&aj[k*w], &v.wavelet[k], j * sizeof(float));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
aj += 4;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
opj_aligned_free(h.wavelet);
|
||||||
|
}
|
||||||
|
|
||||||
113
extern/libopenjpeg/dwt.h
vendored
Normal file
113
extern/libopenjpeg/dwt.h
vendored
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __DWT_H
|
||||||
|
#define __DWT_H
|
||||||
|
/**
|
||||||
|
@file dwt.h
|
||||||
|
@brief Implementation of a discrete wavelet transform (DWT)
|
||||||
|
|
||||||
|
The functions in DWT.C have for goal to realize forward and inverse discret wavelet
|
||||||
|
transform with filter 5-3 (reversible) and filter 9-7 (irreversible). The functions in
|
||||||
|
DWT.C are used by some function in TCD.C.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup DWT DWT - Implementation of a discrete wavelet transform */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Forward 5-3 wavelet tranform in 2-D.
|
||||||
|
Apply a reversible DWT transform to a component of an image.
|
||||||
|
@param tilec Tile component information (current tile)
|
||||||
|
*/
|
||||||
|
void dwt_encode(opj_tcd_tilecomp_t * tilec);
|
||||||
|
/**
|
||||||
|
Inverse 5-3 wavelet tranform in 2-D.
|
||||||
|
Apply a reversible inverse DWT transform to a component of an image.
|
||||||
|
@param tilec Tile component information (current tile)
|
||||||
|
@param numres Number of resolution levels to decode
|
||||||
|
*/
|
||||||
|
void dwt_decode(opj_tcd_tilecomp_t* tilec, int numres);
|
||||||
|
/**
|
||||||
|
Get the gain of a subband for the reversible 5-3 DWT.
|
||||||
|
@param orient Number that identifies the subband (0->LL, 1->HL, 2->LH, 3->HH)
|
||||||
|
@return Returns 0 if orient = 0, returns 1 if orient = 1 or 2, returns 2 otherwise
|
||||||
|
*/
|
||||||
|
int dwt_getgain(int orient);
|
||||||
|
/**
|
||||||
|
Get the norm of a wavelet function of a subband at a specified level for the reversible 5-3 DWT.
|
||||||
|
@param level Level of the wavelet function
|
||||||
|
@param orient Band of the wavelet function
|
||||||
|
@return Returns the norm of the wavelet function
|
||||||
|
*/
|
||||||
|
double dwt_getnorm(int level, int orient);
|
||||||
|
/**
|
||||||
|
Forward 9-7 wavelet transform in 2-D.
|
||||||
|
Apply an irreversible DWT transform to a component of an image.
|
||||||
|
@param tilec Tile component information (current tile)
|
||||||
|
*/
|
||||||
|
void dwt_encode_real(opj_tcd_tilecomp_t * tilec);
|
||||||
|
/**
|
||||||
|
Inverse 9-7 wavelet transform in 2-D.
|
||||||
|
Apply an irreversible inverse DWT transform to a component of an image.
|
||||||
|
@param tilec Tile component information (current tile)
|
||||||
|
@param numres Number of resolution levels to decode
|
||||||
|
*/
|
||||||
|
void dwt_decode_real(opj_tcd_tilecomp_t* tilec, int numres);
|
||||||
|
/**
|
||||||
|
Get the gain of a subband for the irreversible 9-7 DWT.
|
||||||
|
@param orient Number that identifies the subband (0->LL, 1->HL, 2->LH, 3->HH)
|
||||||
|
@return Returns the gain of the 9-7 wavelet transform
|
||||||
|
*/
|
||||||
|
int dwt_getgain_real(int orient);
|
||||||
|
/**
|
||||||
|
Get the norm of a wavelet function of a subband at a specified level for the irreversible 9-7 DWT
|
||||||
|
@param level Level of the wavelet function
|
||||||
|
@param orient Band of the wavelet function
|
||||||
|
@return Returns the norm of the 9-7 wavelet
|
||||||
|
*/
|
||||||
|
double dwt_getnorm_real(int level, int orient);
|
||||||
|
/**
|
||||||
|
Explicit calculation of the Quantization Stepsizes
|
||||||
|
@param tccp Tile-component coding parameters
|
||||||
|
@param prec Precint analyzed
|
||||||
|
*/
|
||||||
|
void dwt_calc_explicit_stepsizes(opj_tccp_t * tccp, int prec);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __DWT_H */
|
||||||
121
extern/libopenjpeg/event.c
vendored
Normal file
121
extern/libopenjpeg/event.c
vendored
Normal file
@@ -0,0 +1,121 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2005, Herv<72> Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/* ==========================================================
|
||||||
|
Utility functions
|
||||||
|
==========================================================*/
|
||||||
|
|
||||||
|
#if !defined(_MSC_VER) && !defined(__MINGW32__)
|
||||||
|
static char*
|
||||||
|
i2a(unsigned i, char *a, unsigned r) {
|
||||||
|
if (i/r > 0) a = i2a(i/r,a,r);
|
||||||
|
*a = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"[i%r];
|
||||||
|
return a+1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Transforms integer i into an ascii string and stores the result in a;
|
||||||
|
string is encoded in the base indicated by r.
|
||||||
|
@param i Number to be converted
|
||||||
|
@param a String result
|
||||||
|
@param r Base of value; must be in the range 2 - 36
|
||||||
|
@return Returns a
|
||||||
|
*/
|
||||||
|
static char *
|
||||||
|
_itoa(int i, char *a, int r) {
|
||||||
|
r = ((r < 2) || (r > 36)) ? 10 : r;
|
||||||
|
if(i < 0) {
|
||||||
|
*a = '-';
|
||||||
|
*i2a(-i, a+1, r) = 0;
|
||||||
|
}
|
||||||
|
else *i2a(i, a, r) = 0;
|
||||||
|
return a;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* !WIN32 */
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
opj_event_mgr_t* OPJ_CALLCONV opj_set_event_mgr(opj_common_ptr cinfo, opj_event_mgr_t *event_mgr, void *context) {
|
||||||
|
if(cinfo) {
|
||||||
|
opj_event_mgr_t *previous = cinfo->event_mgr;
|
||||||
|
cinfo->event_mgr = event_mgr;
|
||||||
|
cinfo->client_data = context;
|
||||||
|
return previous;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool opj_event_msg(opj_common_ptr cinfo, int event_type, const char *fmt, ...) {
|
||||||
|
#define MSG_SIZE 512 /* 512 bytes should be more than enough for a short message */
|
||||||
|
opj_msg_callback msg_handler = NULL;
|
||||||
|
|
||||||
|
opj_event_mgr_t *event_mgr = cinfo->event_mgr;
|
||||||
|
if(event_mgr != NULL) {
|
||||||
|
switch(event_type) {
|
||||||
|
case EVT_ERROR:
|
||||||
|
msg_handler = event_mgr->error_handler;
|
||||||
|
break;
|
||||||
|
case EVT_WARNING:
|
||||||
|
msg_handler = event_mgr->warning_handler;
|
||||||
|
break;
|
||||||
|
case EVT_INFO:
|
||||||
|
msg_handler = event_mgr->info_handler;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if(msg_handler == NULL) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((fmt != NULL) && (event_mgr != NULL)) {
|
||||||
|
va_list arg;
|
||||||
|
int str_length/*, i, j*/; /* UniPG */
|
||||||
|
char message[MSG_SIZE];
|
||||||
|
memset(message, 0, MSG_SIZE);
|
||||||
|
/* initialize the optional parameter list */
|
||||||
|
va_start(arg, fmt);
|
||||||
|
/* check the length of the format string */
|
||||||
|
str_length = (strlen(fmt) > MSG_SIZE) ? MSG_SIZE : strlen(fmt);
|
||||||
|
/* parse the format string and put the result in 'message' */
|
||||||
|
vsprintf(message, fmt, arg); /* UniPG */
|
||||||
|
/* deinitialize the optional parameter list */
|
||||||
|
va_end(arg);
|
||||||
|
|
||||||
|
/* output the message to the user program */
|
||||||
|
msg_handler(message, cinfo->client_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
58
extern/libopenjpeg/event.h
vendored
Normal file
58
extern/libopenjpeg/event.h
vendored
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2005, Herv<72> Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __EVENT_H
|
||||||
|
#define __EVENT_H
|
||||||
|
/**
|
||||||
|
@file event.h
|
||||||
|
@brief Implementation of a event callback system
|
||||||
|
|
||||||
|
The functions in EVENT.C have for goal to send output messages (errors, warnings, debug) to the user.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define EVT_ERROR 1 /**< Error event type */
|
||||||
|
#define EVT_WARNING 2 /**< Warning event type */
|
||||||
|
#define EVT_INFO 4 /**< Debug event type */
|
||||||
|
|
||||||
|
/** @defgroup EVENT EVENT - Implementation of a event callback system */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Exported functions (see also openjpeg.h) */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Write formatted data to a string and send the string to a user callback.
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@param event_type Event type or callback to use to send the message
|
||||||
|
@param fmt Format-control string (plus optionnal arguments)
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
bool opj_event_msg(opj_common_ptr cinfo, int event_type, const char *fmt, ...);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __EVENT_H */
|
||||||
64
extern/libopenjpeg/fix.h
vendored
Normal file
64
extern/libopenjpeg/fix.h
vendored
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __FIX_H
|
||||||
|
#define __FIX_H
|
||||||
|
|
||||||
|
#if defined(_MSC_VER) || defined(__BORLANDC__)
|
||||||
|
#define int64 __int64
|
||||||
|
#else
|
||||||
|
#define int64 long long
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
@file fix.h
|
||||||
|
@brief Implementation of operations of specific multiplication (FIX)
|
||||||
|
|
||||||
|
The functions in FIX.H have for goal to realize specific multiplication.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup FIX FIX - Implementation of operations of specific multiplication */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Multiply two fixed-precision rational numbers.
|
||||||
|
@param a
|
||||||
|
@param b
|
||||||
|
@return Returns a * b
|
||||||
|
*/
|
||||||
|
static INLINE int fix_mul(int a, int b) {
|
||||||
|
int64 temp = (int64) a * (int64) b ;
|
||||||
|
temp += temp & 4096;
|
||||||
|
return (int) (temp >> 13) ;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __FIX_H */
|
||||||
89
extern/libopenjpeg/image.c
vendored
Normal file
89
extern/libopenjpeg/image.c
vendored
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2005, Herv<72> Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
opj_image_t* opj_image_create0(void) {
|
||||||
|
opj_image_t *image = (opj_image_t*)opj_malloc(sizeof(opj_image_t));
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
opj_image_t* OPJ_CALLCONV opj_image_create(int numcmpts, opj_image_cmptparm_t *cmptparms, OPJ_COLOR_SPACE clrspc) {
|
||||||
|
int compno;
|
||||||
|
opj_image_t *image = NULL;
|
||||||
|
|
||||||
|
image = (opj_image_t*) opj_calloc(1, sizeof(opj_image_t));
|
||||||
|
if(image) {
|
||||||
|
image->color_space = clrspc;
|
||||||
|
image->numcomps = numcmpts;
|
||||||
|
/* allocate memory for the per-component information */
|
||||||
|
image->comps = (opj_image_comp_t*)opj_malloc(image->numcomps * sizeof(opj_image_comp_t));
|
||||||
|
if(!image->comps) {
|
||||||
|
fprintf(stderr,"Unable to allocate memory for image.\n");
|
||||||
|
opj_image_destroy(image);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
/* create the individual image components */
|
||||||
|
for(compno = 0; compno < numcmpts; compno++) {
|
||||||
|
opj_image_comp_t *comp = &image->comps[compno];
|
||||||
|
comp->dx = cmptparms[compno].dx;
|
||||||
|
comp->dy = cmptparms[compno].dy;
|
||||||
|
comp->w = cmptparms[compno].w;
|
||||||
|
comp->h = cmptparms[compno].h;
|
||||||
|
comp->x0 = cmptparms[compno].x0;
|
||||||
|
comp->y0 = cmptparms[compno].y0;
|
||||||
|
comp->prec = cmptparms[compno].prec;
|
||||||
|
comp->bpp = cmptparms[compno].bpp;
|
||||||
|
comp->sgnd = cmptparms[compno].sgnd;
|
||||||
|
comp->data = (int*) opj_calloc(comp->w * comp->h, sizeof(int));
|
||||||
|
if(!comp->data) {
|
||||||
|
fprintf(stderr,"Unable to allocate memory for image.\n");
|
||||||
|
opj_image_destroy(image);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OPJ_CALLCONV opj_image_destroy(opj_image_t *image) {
|
||||||
|
int i;
|
||||||
|
if(image) {
|
||||||
|
if(image->comps) {
|
||||||
|
/* image components */
|
||||||
|
for(i = 0; i < image->numcomps; i++) {
|
||||||
|
opj_image_comp_t *image_comp = &image->comps[i];
|
||||||
|
if(image_comp->data) {
|
||||||
|
opj_free(image_comp->data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
opj_free(image->comps);
|
||||||
|
}
|
||||||
|
opj_free(image);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
48
extern/libopenjpeg/image.h
vendored
Normal file
48
extern/libopenjpeg/image.h
vendored
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2005, Herv<72> Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __IMAGE_H
|
||||||
|
#define __IMAGE_H
|
||||||
|
/**
|
||||||
|
@file image.h
|
||||||
|
@brief Implementation of operations on images (IMAGE)
|
||||||
|
|
||||||
|
The functions in IMAGE.C have for goal to realize operations on images.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup IMAGE IMAGE - Implementation of operations on images */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Create an empty image
|
||||||
|
@todo this function should be removed
|
||||||
|
@return returns an empty image if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_image_t* opj_image_create0(void);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __IMAGE_H */
|
||||||
|
|
||||||
119
extern/libopenjpeg/int.h
vendored
Normal file
119
extern/libopenjpeg/int.h
vendored
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __INT_H
|
||||||
|
#define __INT_H
|
||||||
|
/**
|
||||||
|
@file int.h
|
||||||
|
@brief Implementation of operations on integers (INT)
|
||||||
|
|
||||||
|
The functions in INT.H have for goal to realize operations on integers.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup INT INT - Implementation of operations on integers */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Exported functions (see also openjpeg.h) */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Get the minimum of two integers
|
||||||
|
@return Returns a if a < b else b
|
||||||
|
*/
|
||||||
|
static INLINE int int_min(int a, int b) {
|
||||||
|
return a < b ? a : b;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
Get the maximum of two integers
|
||||||
|
@return Returns a if a > b else b
|
||||||
|
*/
|
||||||
|
static INLINE int int_max(int a, int b) {
|
||||||
|
return (a > b) ? a : b;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
Clamp an integer inside an interval
|
||||||
|
@return
|
||||||
|
<ul>
|
||||||
|
<li>Returns a if (min < a < max)
|
||||||
|
<li>Returns max if (a > max)
|
||||||
|
<li>Returns min if (a < min)
|
||||||
|
</ul>
|
||||||
|
*/
|
||||||
|
static INLINE int int_clamp(int a, int min, int max) {
|
||||||
|
if (a < min)
|
||||||
|
return min;
|
||||||
|
if (a > max)
|
||||||
|
return max;
|
||||||
|
return a;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
@return Get absolute value of integer
|
||||||
|
*/
|
||||||
|
static INLINE int int_abs(int a) {
|
||||||
|
return a < 0 ? -a : a;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
Divide an integer and round upwards
|
||||||
|
@return Returns a divided by b
|
||||||
|
*/
|
||||||
|
static INLINE int int_ceildiv(int a, int b) {
|
||||||
|
return (a + b - 1) / b;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
Divide an integer by a power of 2 and round upwards
|
||||||
|
@return Returns a divided by 2^b
|
||||||
|
*/
|
||||||
|
static INLINE int int_ceildivpow2(int a, int b) {
|
||||||
|
return (a + (1 << b) - 1) >> b;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
Divide an integer by a power of 2 and round downwards
|
||||||
|
@return Returns a divided by 2^b
|
||||||
|
*/
|
||||||
|
static INLINE int int_floordivpow2(int a, int b) {
|
||||||
|
return a >> b;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
Get logarithm of an integer and round downwards
|
||||||
|
@return Returns log2(a)
|
||||||
|
*/
|
||||||
|
static INLINE int int_floorlog2(int a) {
|
||||||
|
int l;
|
||||||
|
for (l = 0; a > 1; l++) {
|
||||||
|
a >>= 1;
|
||||||
|
}
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif
|
||||||
2498
extern/libopenjpeg/j2k.c
vendored
Normal file
2498
extern/libopenjpeg/j2k.c
vendored
Normal file
File diff suppressed because it is too large
Load Diff
446
extern/libopenjpeg/j2k.h
vendored
Normal file
446
extern/libopenjpeg/j2k.h
vendored
Normal file
@@ -0,0 +1,446 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* Copyright (c) 2006-2007, Parvatha Elangovan
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __J2K_H
|
||||||
|
#define __J2K_H
|
||||||
|
/**
|
||||||
|
@file j2k.h
|
||||||
|
@brief The JPEG-2000 Codestream Reader/Writer (J2K)
|
||||||
|
|
||||||
|
The functions in J2K.C have for goal to read/write the several parts of the codestream: markers and data.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup J2K J2K - JPEG-2000 codestream reader/writer */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
#define J2K_CP_CSTY_PRT 0x01
|
||||||
|
#define J2K_CP_CSTY_SOP 0x02
|
||||||
|
#define J2K_CP_CSTY_EPH 0x04
|
||||||
|
#define J2K_CCP_CSTY_PRT 0x01
|
||||||
|
#define J2K_CCP_CBLKSTY_LAZY 0x01
|
||||||
|
#define J2K_CCP_CBLKSTY_RESET 0x02
|
||||||
|
#define J2K_CCP_CBLKSTY_TERMALL 0x04
|
||||||
|
#define J2K_CCP_CBLKSTY_VSC 0x08
|
||||||
|
#define J2K_CCP_CBLKSTY_PTERM 0x10
|
||||||
|
#define J2K_CCP_CBLKSTY_SEGSYM 0x20
|
||||||
|
#define J2K_CCP_QNTSTY_NOQNT 0
|
||||||
|
#define J2K_CCP_QNTSTY_SIQNT 1
|
||||||
|
#define J2K_CCP_QNTSTY_SEQNT 2
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
#define J2K_MS_SOC 0xff4f /**< SOC marker value */
|
||||||
|
#define J2K_MS_SOT 0xff90 /**< SOT marker value */
|
||||||
|
#define J2K_MS_SOD 0xff93 /**< SOD marker value */
|
||||||
|
#define J2K_MS_EOC 0xffd9 /**< EOC marker value */
|
||||||
|
#define J2K_MS_SIZ 0xff51 /**< SIZ marker value */
|
||||||
|
#define J2K_MS_COD 0xff52 /**< COD marker value */
|
||||||
|
#define J2K_MS_COC 0xff53 /**< COC marker value */
|
||||||
|
#define J2K_MS_RGN 0xff5e /**< RGN marker value */
|
||||||
|
#define J2K_MS_QCD 0xff5c /**< QCD marker value */
|
||||||
|
#define J2K_MS_QCC 0xff5d /**< QCC marker value */
|
||||||
|
#define J2K_MS_POC 0xff5f /**< POC marker value */
|
||||||
|
#define J2K_MS_TLM 0xff55 /**< TLM marker value */
|
||||||
|
#define J2K_MS_PLM 0xff57 /**< PLM marker value */
|
||||||
|
#define J2K_MS_PLT 0xff58 /**< PLT marker value */
|
||||||
|
#define J2K_MS_PPM 0xff60 /**< PPM marker value */
|
||||||
|
#define J2K_MS_PPT 0xff61 /**< PPT marker value */
|
||||||
|
#define J2K_MS_SOP 0xff91 /**< SOP marker value */
|
||||||
|
#define J2K_MS_EPH 0xff92 /**< EPH marker value */
|
||||||
|
#define J2K_MS_CRG 0xff63 /**< CRG marker value */
|
||||||
|
#define J2K_MS_COM 0xff64 /**< COM marker value */
|
||||||
|
/* UniPG>> */
|
||||||
|
#ifdef USE_JPWL
|
||||||
|
#define J2K_MS_EPC 0xff68 /**< EPC marker value (Part 11: JPEG 2000 for Wireless) */
|
||||||
|
#define J2K_MS_EPB 0xff66 /**< EPB marker value (Part 11: JPEG 2000 for Wireless) */
|
||||||
|
#define J2K_MS_ESD 0xff67 /**< ESD marker value (Part 11: JPEG 2000 for Wireless) */
|
||||||
|
#define J2K_MS_RED 0xff69 /**< RED marker value (Part 11: JPEG 2000 for Wireless) */
|
||||||
|
#endif /* USE_JPWL */
|
||||||
|
#ifdef USE_JPSEC
|
||||||
|
#define J2K_MS_SEC 0xff65 /**< SEC marker value (Part 8: Secure JPEG 2000) */
|
||||||
|
#define J2K_MS_INSEC 0xff94 /**< INSEC marker value (Part 8: Secure JPEG 2000) */
|
||||||
|
#endif /* USE_JPSEC */
|
||||||
|
/* <<UniPG */
|
||||||
|
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
Values that specify the status of the decoding process when decoding the main header.
|
||||||
|
These values may be combined with a | operator.
|
||||||
|
*/
|
||||||
|
typedef enum J2K_STATUS {
|
||||||
|
J2K_STATE_MHSOC = 0x0001, /**< a SOC marker is expected */
|
||||||
|
J2K_STATE_MHSIZ = 0x0002, /**< a SIZ marker is expected */
|
||||||
|
J2K_STATE_MH = 0x0004, /**< the decoding process is in the main header */
|
||||||
|
J2K_STATE_TPHSOT = 0x0008, /**< the decoding process is in a tile part header and expects a SOT marker */
|
||||||
|
J2K_STATE_TPH = 0x0010, /**< the decoding process is in a tile part header */
|
||||||
|
J2K_STATE_MT = 0x0020, /**< the EOC marker has just been read */
|
||||||
|
J2K_STATE_NEOC = 0x0040, /**< the decoding process must not expect a EOC marker because the codestream is truncated */
|
||||||
|
J2K_STATE_ERR = 0x0080 /**< the decoding process has encountered an error */
|
||||||
|
} J2K_STATUS;
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
T2 encoding mode
|
||||||
|
*/
|
||||||
|
typedef enum T2_MODE {
|
||||||
|
THRESH_CALC = 0, /** Function called in Rate allocation process*/
|
||||||
|
FINAL_PASS = 1 /** Function called in Tier 2 process*/
|
||||||
|
}J2K_T2_MODE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Quantization stepsize
|
||||||
|
*/
|
||||||
|
typedef struct opj_stepsize {
|
||||||
|
/** exponent */
|
||||||
|
int expn;
|
||||||
|
/** mantissa */
|
||||||
|
int mant;
|
||||||
|
} opj_stepsize_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tile-component coding parameters
|
||||||
|
*/
|
||||||
|
typedef struct opj_tccp {
|
||||||
|
/** coding style */
|
||||||
|
int csty;
|
||||||
|
/** number of resolutions */
|
||||||
|
int numresolutions;
|
||||||
|
/** code-blocks width */
|
||||||
|
int cblkw;
|
||||||
|
/** code-blocks height */
|
||||||
|
int cblkh;
|
||||||
|
/** code-block coding style */
|
||||||
|
int cblksty;
|
||||||
|
/** discrete wavelet transform identifier */
|
||||||
|
int qmfbid;
|
||||||
|
/** quantisation style */
|
||||||
|
int qntsty;
|
||||||
|
/** stepsizes used for quantization */
|
||||||
|
opj_stepsize_t stepsizes[J2K_MAXBANDS];
|
||||||
|
/** number of guard bits */
|
||||||
|
int numgbits;
|
||||||
|
/** Region Of Interest shift */
|
||||||
|
int roishift;
|
||||||
|
/** precinct width */
|
||||||
|
int prcw[J2K_MAXRLVLS];
|
||||||
|
/** precinct height */
|
||||||
|
int prch[J2K_MAXRLVLS];
|
||||||
|
} opj_tccp_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tile coding parameters :
|
||||||
|
this structure is used to store coding/decoding parameters common to all
|
||||||
|
tiles (information like COD, COC in main header)
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcp {
|
||||||
|
/** 1 : first part-tile of a tile */
|
||||||
|
int first;
|
||||||
|
/** coding style */
|
||||||
|
int csty;
|
||||||
|
/** progression order */
|
||||||
|
OPJ_PROG_ORDER prg;
|
||||||
|
/** number of layers */
|
||||||
|
int numlayers;
|
||||||
|
/** multi-component transform identifier */
|
||||||
|
int mct;
|
||||||
|
/** rates of layers */
|
||||||
|
float rates[100];
|
||||||
|
/** number of progression order changes */
|
||||||
|
int numpocs;
|
||||||
|
/** indicates if a POC marker has been used O:NO, 1:YES */
|
||||||
|
int POC;
|
||||||
|
/** progression order changes */
|
||||||
|
opj_poc_t pocs[32];
|
||||||
|
/** packet header store there for futur use in t2_decode_packet */
|
||||||
|
unsigned char *ppt_data;
|
||||||
|
/** pointer remaining on the first byte of the first header if ppt is used */
|
||||||
|
unsigned char *ppt_data_first;
|
||||||
|
/** If ppt == 1 --> there was a PPT marker for the present tile */
|
||||||
|
int ppt;
|
||||||
|
/** used in case of multiple marker PPT (number of info already stored) */
|
||||||
|
int ppt_store;
|
||||||
|
/** ppmbug1 */
|
||||||
|
int ppt_len;
|
||||||
|
/** add fixed_quality */
|
||||||
|
float distoratio[100];
|
||||||
|
/** tile-component coding parameters */
|
||||||
|
opj_tccp_t *tccps;
|
||||||
|
} opj_tcp_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Coding parameters
|
||||||
|
*/
|
||||||
|
typedef struct opj_cp {
|
||||||
|
/** Digital cinema profile*/
|
||||||
|
OPJ_CINEMA_MODE cinema;
|
||||||
|
/** Maximum rate for each component. If == 0, component size limitation is not considered */
|
||||||
|
int max_comp_size;
|
||||||
|
/** Size of the image in bits*/
|
||||||
|
int img_size;
|
||||||
|
/** Rsiz*/
|
||||||
|
OPJ_RSIZ_CAPABILITIES rsiz;
|
||||||
|
/** Enabling Tile part generation*/
|
||||||
|
char tp_on;
|
||||||
|
/** Flag determining tile part generation*/
|
||||||
|
char tp_flag;
|
||||||
|
/** Position of tile part flag in progression order*/
|
||||||
|
int tp_pos;
|
||||||
|
/** allocation by rate/distortion */
|
||||||
|
int disto_alloc;
|
||||||
|
/** allocation by fixed layer */
|
||||||
|
int fixed_alloc;
|
||||||
|
/** add fixed_quality */
|
||||||
|
int fixed_quality;
|
||||||
|
/** if != 0, then original dimension divided by 2^(reduce); if == 0 or not used, image is decoded to the full resolution */
|
||||||
|
int reduce;
|
||||||
|
/** if != 0, then only the first "layer" layers are decoded; if == 0 or not used, all the quality layers are decoded */
|
||||||
|
int layer;
|
||||||
|
/** if == NO_LIMITATION, decode entire codestream; if == LIMIT_TO_MAIN_HEADER then only decode the main header */
|
||||||
|
OPJ_LIMIT_DECODING limit_decoding;
|
||||||
|
/** XTOsiz */
|
||||||
|
int tx0;
|
||||||
|
/** YTOsiz */
|
||||||
|
int ty0;
|
||||||
|
/** XTsiz */
|
||||||
|
int tdx;
|
||||||
|
/** YTsiz */
|
||||||
|
int tdy;
|
||||||
|
/** comment for coding */
|
||||||
|
char *comment;
|
||||||
|
/** number of tiles in width */
|
||||||
|
int tw;
|
||||||
|
/** number of tiles in heigth */
|
||||||
|
int th;
|
||||||
|
/** ID number of the tiles present in the codestream */
|
||||||
|
int *tileno;
|
||||||
|
/** size of the vector tileno */
|
||||||
|
int tileno_size;
|
||||||
|
/** packet header store there for futur use in t2_decode_packet */
|
||||||
|
unsigned char *ppm_data;
|
||||||
|
/** pointer remaining on the first byte of the first header if ppm is used */
|
||||||
|
unsigned char *ppm_data_first;
|
||||||
|
/** if ppm == 1 --> there was a PPM marker for the present tile */
|
||||||
|
int ppm;
|
||||||
|
/** use in case of multiple marker PPM (number of info already store) */
|
||||||
|
int ppm_store;
|
||||||
|
/** use in case of multiple marker PPM (case on non-finished previous info) */
|
||||||
|
int ppm_previous;
|
||||||
|
/** ppmbug1 */
|
||||||
|
int ppm_len;
|
||||||
|
/** tile coding parameters */
|
||||||
|
opj_tcp_t *tcps;
|
||||||
|
/** fixed layer */
|
||||||
|
int *matrice;
|
||||||
|
/* UniPG>> */
|
||||||
|
#ifdef USE_JPWL
|
||||||
|
/** enables writing of EPC in MH, thus activating JPWL */
|
||||||
|
bool epc_on;
|
||||||
|
/** enables writing of EPB, in case of activated JPWL */
|
||||||
|
bool epb_on;
|
||||||
|
/** enables writing of ESD, in case of activated JPWL */
|
||||||
|
bool esd_on;
|
||||||
|
/** enables writing of informative techniques of ESD, in case of activated JPWL */
|
||||||
|
bool info_on;
|
||||||
|
/** enables writing of RED, in case of activated JPWL */
|
||||||
|
bool red_on;
|
||||||
|
/** error protection method for MH (0,1,16,32,37-128) */
|
||||||
|
int hprot_MH;
|
||||||
|
/** tile number of header protection specification (>=0) */
|
||||||
|
int hprot_TPH_tileno[JPWL_MAX_NO_TILESPECS];
|
||||||
|
/** error protection methods for TPHs (0,1,16,32,37-128) */
|
||||||
|
int hprot_TPH[JPWL_MAX_NO_TILESPECS];
|
||||||
|
/** tile number of packet protection specification (>=0) */
|
||||||
|
int pprot_tileno[JPWL_MAX_NO_PACKSPECS];
|
||||||
|
/** packet number of packet protection specification (>=0) */
|
||||||
|
int pprot_packno[JPWL_MAX_NO_PACKSPECS];
|
||||||
|
/** error protection methods for packets (0,1,16,32,37-128) */
|
||||||
|
int pprot[JPWL_MAX_NO_PACKSPECS];
|
||||||
|
/** enables writing of ESD, (0/2/4 bytes) */
|
||||||
|
int sens_size;
|
||||||
|
/** sensitivity addressing size (0=auto/2/4 bytes) */
|
||||||
|
int sens_addr;
|
||||||
|
/** sensitivity range (0-3) */
|
||||||
|
int sens_range;
|
||||||
|
/** sensitivity method for MH (-1,0-7) */
|
||||||
|
int sens_MH;
|
||||||
|
/** tile number of sensitivity specification (>=0) */
|
||||||
|
int sens_TPH_tileno[JPWL_MAX_NO_TILESPECS];
|
||||||
|
/** sensitivity methods for TPHs (-1,0-7) */
|
||||||
|
int sens_TPH[JPWL_MAX_NO_TILESPECS];
|
||||||
|
/** enables JPWL correction at the decoder */
|
||||||
|
bool correct;
|
||||||
|
/** expected number of components at the decoder */
|
||||||
|
int exp_comps;
|
||||||
|
/** maximum number of tiles at the decoder */
|
||||||
|
int max_tiles;
|
||||||
|
#endif /* USE_JPWL */
|
||||||
|
/* <<UniPG */
|
||||||
|
} opj_cp_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
JPEG-2000 codestream reader/writer
|
||||||
|
*/
|
||||||
|
typedef struct opj_j2k {
|
||||||
|
/** codec context */
|
||||||
|
opj_common_ptr cinfo;
|
||||||
|
|
||||||
|
/** locate in which part of the codestream the decoder is (main header, tile header, end) */
|
||||||
|
int state;
|
||||||
|
/** number of the tile curently concern by coding/decoding */
|
||||||
|
int curtileno;
|
||||||
|
/** Tile part number*/
|
||||||
|
int tp_num;
|
||||||
|
/** Tilepart number currently coding*/
|
||||||
|
int cur_tp_num;
|
||||||
|
/** Total number of tileparts of the current tile*/
|
||||||
|
int *cur_totnum_tp;
|
||||||
|
/**
|
||||||
|
locate the start position of the TLM marker
|
||||||
|
after encoding the tilepart, a jump (in j2k_write_sod) is done to the TLM marker to store the value of its length.
|
||||||
|
*/
|
||||||
|
int tlm_start;
|
||||||
|
/** Total num of tile parts in whole image = num tiles* num tileparts in each tile*/
|
||||||
|
/** used in TLMmarker*/
|
||||||
|
int totnum_tp;
|
||||||
|
/**
|
||||||
|
locate the position of the end of the tile in the codestream,
|
||||||
|
used to detect a truncated codestream (in j2k_read_sod)
|
||||||
|
*/
|
||||||
|
unsigned char *eot;
|
||||||
|
/**
|
||||||
|
locate the start position of the SOT marker of the current coded tile:
|
||||||
|
after encoding the tile, a jump (in j2k_write_sod) is done to the SOT marker to store the value of its length.
|
||||||
|
*/
|
||||||
|
int sot_start;
|
||||||
|
int sod_start;
|
||||||
|
/**
|
||||||
|
as the J2K-file is written in several parts during encoding,
|
||||||
|
it enables to make the right correction in position return by cio_tell
|
||||||
|
*/
|
||||||
|
int pos_correction;
|
||||||
|
/** array used to store the data of each tile */
|
||||||
|
unsigned char **tile_data;
|
||||||
|
/** array used to store the length of each tile */
|
||||||
|
int *tile_len;
|
||||||
|
/**
|
||||||
|
decompression only :
|
||||||
|
store decoding parameters common to all tiles (information like COD, COC in main header)
|
||||||
|
*/
|
||||||
|
opj_tcp_t *default_tcp;
|
||||||
|
/** pointer to the encoded / decoded image */
|
||||||
|
opj_image_t *image;
|
||||||
|
/** pointer to the coding parameters */
|
||||||
|
opj_cp_t *cp;
|
||||||
|
/** helper used to write the index file */
|
||||||
|
opj_codestream_info_t *cstr_info;
|
||||||
|
/** pointer to the byte i/o stream */
|
||||||
|
opj_cio_t *cio;
|
||||||
|
} opj_j2k_t;
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Creates a J2K decompression structure
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@return Returns a handle to a J2K decompressor if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_j2k_t* j2k_create_decompress(opj_common_ptr cinfo);
|
||||||
|
/**
|
||||||
|
Destroy a J2K decompressor handle
|
||||||
|
@param j2k J2K decompressor handle to destroy
|
||||||
|
*/
|
||||||
|
void j2k_destroy_decompress(opj_j2k_t *j2k);
|
||||||
|
/**
|
||||||
|
Setup the decoder decoding parameters using user parameters.
|
||||||
|
Decoding parameters are returned in j2k->cp.
|
||||||
|
@param j2k J2K decompressor handle
|
||||||
|
@param parameters decompression parameters
|
||||||
|
*/
|
||||||
|
void j2k_setup_decoder(opj_j2k_t *j2k, opj_dparameters_t *parameters);
|
||||||
|
/**
|
||||||
|
Decode an image from a JPEG-2000 codestream
|
||||||
|
@param j2k J2K decompressor handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@param cstr_info Codestream information structure if required, NULL otherwise
|
||||||
|
@return Returns a decoded image if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_image_t* j2k_decode(opj_j2k_t *j2k, opj_cio_t *cio, opj_codestream_info_t *cstr_info);
|
||||||
|
/**
|
||||||
|
Decode an image form a JPT-stream (JPEG 2000, JPIP)
|
||||||
|
@param j2k J2K decompressor handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@param cstr_info Codestream information structure if required, NULL otherwise
|
||||||
|
@return Returns a decoded image if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_image_t* j2k_decode_jpt_stream(opj_j2k_t *j2k, opj_cio_t *cio, opj_codestream_info_t *cstr_info);
|
||||||
|
/**
|
||||||
|
Creates a J2K compression structure
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@return Returns a handle to a J2K compressor if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_j2k_t* j2k_create_compress(opj_common_ptr cinfo);
|
||||||
|
/**
|
||||||
|
Destroy a J2K compressor handle
|
||||||
|
@param j2k J2K compressor handle to destroy
|
||||||
|
*/
|
||||||
|
void j2k_destroy_compress(opj_j2k_t *j2k);
|
||||||
|
/**
|
||||||
|
Setup the encoder parameters using the current image and using user parameters.
|
||||||
|
Coding parameters are returned in j2k->cp.
|
||||||
|
@param j2k J2K compressor handle
|
||||||
|
@param parameters compression parameters
|
||||||
|
@param image input filled image
|
||||||
|
*/
|
||||||
|
void j2k_setup_encoder(opj_j2k_t *j2k, opj_cparameters_t *parameters, opj_image_t *image);
|
||||||
|
/**
|
||||||
|
Converts an enum type progression order to string type
|
||||||
|
*/
|
||||||
|
char *j2k_convert_progression_order(OPJ_PROG_ORDER prg_order);
|
||||||
|
/**
|
||||||
|
Encode an image into a JPEG-2000 codestream
|
||||||
|
@param j2k J2K compressor handle
|
||||||
|
@param cio Output buffer stream
|
||||||
|
@param image Image to encode
|
||||||
|
@param cstr_info Codestream information structure if required, NULL otherwise
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
bool j2k_encode(opj_j2k_t *j2k, opj_cio_t *cio, opj_image_t *image, opj_codestream_info_t *cstr_info);
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __J2K_H */
|
||||||
59
extern/libopenjpeg/j2k_lib.c
vendored
Normal file
59
extern/libopenjpeg/j2k_lib.c
vendored
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2005, Herv<72> Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef WIN32
|
||||||
|
#include <windows.h>
|
||||||
|
#else
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <sys/resource.h>
|
||||||
|
#include <sys/times.h>
|
||||||
|
#endif /* WIN32 */
|
||||||
|
#include "opj_includes.h"
|
||||||
|
|
||||||
|
double opj_clock(void) {
|
||||||
|
#ifdef WIN32
|
||||||
|
/* WIN32: use QueryPerformance (very accurate) */
|
||||||
|
LARGE_INTEGER freq , t ;
|
||||||
|
/* freq is the clock speed of the CPU */
|
||||||
|
QueryPerformanceFrequency(&freq) ;
|
||||||
|
/* cout << "freq = " << ((double) freq.QuadPart) << endl; */
|
||||||
|
/* t is the high resolution performance counter (see MSDN) */
|
||||||
|
QueryPerformanceCounter ( & t ) ;
|
||||||
|
return ( t.QuadPart /(double) freq.QuadPart ) ;
|
||||||
|
#else
|
||||||
|
/* Unix or Linux: use resource usage */
|
||||||
|
struct rusage t;
|
||||||
|
double procTime;
|
||||||
|
/* (1) Get the rusage data structure at this moment (man getrusage) */
|
||||||
|
getrusage(0,&t);
|
||||||
|
/* (2) What is the elapsed time ? - CPU time = User time + System time */
|
||||||
|
/* (2a) Get the seconds */
|
||||||
|
procTime = t.ru_utime.tv_sec + t.ru_stime.tv_sec;
|
||||||
|
/* (2b) More precisely! Get the microseconds part ! */
|
||||||
|
return ( procTime + (t.ru_utime.tv_usec + t.ru_stime.tv_usec) * 1e-6 ) ;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
54
extern/libopenjpeg/j2k_lib.h
vendored
Normal file
54
extern/libopenjpeg/j2k_lib.h
vendored
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2005, Herv<72> Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __J2K_LIB_H
|
||||||
|
#define __J2K_LIB_H
|
||||||
|
/**
|
||||||
|
@file j2k_lib.h
|
||||||
|
@brief Internal functions
|
||||||
|
|
||||||
|
The functions in J2K_LIB.C are internal utilities mainly used for timing.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup MISC MISC - Miscellaneous internal functions */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
Difference in successive opj_clock() calls tells you the elapsed time
|
||||||
|
@return Returns time in seconds
|
||||||
|
*/
|
||||||
|
double opj_clock(void);
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __J2K_LIB_H */
|
||||||
|
|
||||||
710
extern/libopenjpeg/jp2.c
vendored
Normal file
710
extern/libopenjpeg/jp2.c
vendored
Normal file
@@ -0,0 +1,710 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/** @defgroup JP2 JP2 - JPEG-2000 file format reader/writer */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Local static functions */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Read box headers
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@param cio Input stream
|
||||||
|
@param box
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
static bool jp2_read_boxhdr(opj_common_ptr cinfo, opj_cio_t *cio, opj_jp2_box_t *box);
|
||||||
|
/*static void jp2_write_url(opj_cio_t *cio, char *Idx_file);*/
|
||||||
|
/**
|
||||||
|
Read the IHDR box - Image Header box
|
||||||
|
@param jp2 JP2 handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
static bool jp2_read_ihdr(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
static void jp2_write_ihdr(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
static void jp2_write_bpcc(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
static bool jp2_read_bpcc(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
static void jp2_write_colr(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
static bool jp2_read_colr(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
/**
|
||||||
|
Write the FTYP box - File type box
|
||||||
|
@param jp2 JP2 handle
|
||||||
|
@param cio Output buffer stream
|
||||||
|
*/
|
||||||
|
static void jp2_write_ftyp(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
/**
|
||||||
|
Read the FTYP box - File type box
|
||||||
|
@param jp2 JP2 handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
static bool jp2_read_ftyp(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
static int jp2_write_jp2c(opj_jp2_t *jp2, opj_cio_t *cio, opj_image_t *image, opj_codestream_info_t *cstr_info);
|
||||||
|
static bool jp2_read_jp2c(opj_jp2_t *jp2, opj_cio_t *cio, unsigned int *j2k_codestream_length, unsigned int *j2k_codestream_offset);
|
||||||
|
static void jp2_write_jp(opj_cio_t *cio);
|
||||||
|
/**
|
||||||
|
Read the JP box - JPEG 2000 signature
|
||||||
|
@param jp2 JP2 handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
static bool jp2_read_jp(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
/**
|
||||||
|
Decode the structure of a JP2 file
|
||||||
|
@param jp2 JP2 handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
static bool jp2_read_struct(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static bool jp2_read_boxhdr(opj_common_ptr cinfo, opj_cio_t *cio, opj_jp2_box_t *box) {
|
||||||
|
box->init_pos = cio_tell(cio);
|
||||||
|
box->length = cio_read(cio, 4);
|
||||||
|
box->type = cio_read(cio, 4);
|
||||||
|
if (box->length == 1) {
|
||||||
|
if (cio_read(cio, 4) != 0) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Cannot handle box sizes higher than 2^32\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
box->length = cio_read(cio, 4);
|
||||||
|
if (box->length == 0)
|
||||||
|
box->length = cio_numbytesleft(cio) + 12;
|
||||||
|
}
|
||||||
|
else if (box->length == 0) {
|
||||||
|
box->length = cio_numbytesleft(cio) + 8;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
static void jp2_write_url(opj_cio_t *cio, char *Idx_file) {
|
||||||
|
unsigned int i;
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
box.init_pos = cio_tell(cio);
|
||||||
|
cio_skip(cio, 4);
|
||||||
|
cio_write(cio, JP2_URL, 4); /* DBTL */
|
||||||
|
cio_write(cio, 0, 1); /* VERS */
|
||||||
|
cio_write(cio, 0, 3); /* FLAG */
|
||||||
|
|
||||||
|
if(Idx_file) {
|
||||||
|
for (i = 0; i < strlen(Idx_file); i++) {
|
||||||
|
cio_write(cio, Idx_file[i], 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
box.length = cio_tell(cio) - box.init_pos;
|
||||||
|
cio_seek(cio, box.init_pos);
|
||||||
|
cio_write(cio, box.length, 4); /* L */
|
||||||
|
cio_seek(cio, box.init_pos + box.length);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static bool jp2_read_ihdr(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
opj_common_ptr cinfo = jp2->cinfo;
|
||||||
|
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
if (JP2_IHDR != box.type) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Expected IHDR Marker\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
jp2->h = cio_read(cio, 4); /* HEIGHT */
|
||||||
|
jp2->w = cio_read(cio, 4); /* WIDTH */
|
||||||
|
jp2->numcomps = cio_read(cio, 2); /* NC */
|
||||||
|
jp2->comps = (opj_jp2_comps_t*) opj_malloc(jp2->numcomps * sizeof(opj_jp2_comps_t));
|
||||||
|
|
||||||
|
jp2->bpc = cio_read(cio, 1); /* BPC */
|
||||||
|
|
||||||
|
jp2->C = cio_read(cio, 1); /* C */
|
||||||
|
jp2->UnkC = cio_read(cio, 1); /* UnkC */
|
||||||
|
jp2->IPR = cio_read(cio, 1); /* IPR */
|
||||||
|
|
||||||
|
if (cio_tell(cio) - box.init_pos != box.length) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Error with IHDR Box\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void jp2_write_ihdr(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
box.init_pos = cio_tell(cio);
|
||||||
|
cio_skip(cio, 4);
|
||||||
|
cio_write(cio, JP2_IHDR, 4); /* IHDR */
|
||||||
|
|
||||||
|
cio_write(cio, jp2->h, 4); /* HEIGHT */
|
||||||
|
cio_write(cio, jp2->w, 4); /* WIDTH */
|
||||||
|
cio_write(cio, jp2->numcomps, 2); /* NC */
|
||||||
|
|
||||||
|
cio_write(cio, jp2->bpc, 1); /* BPC */
|
||||||
|
|
||||||
|
cio_write(cio, jp2->C, 1); /* C : Always 7 */
|
||||||
|
cio_write(cio, jp2->UnkC, 1); /* UnkC, colorspace unknown */
|
||||||
|
cio_write(cio, jp2->IPR, 1); /* IPR, no intellectual property */
|
||||||
|
|
||||||
|
box.length = cio_tell(cio) - box.init_pos;
|
||||||
|
cio_seek(cio, box.init_pos);
|
||||||
|
cio_write(cio, box.length, 4); /* L */
|
||||||
|
cio_seek(cio, box.init_pos + box.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void jp2_write_bpcc(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
unsigned int i;
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
box.init_pos = cio_tell(cio);
|
||||||
|
cio_skip(cio, 4);
|
||||||
|
cio_write(cio, JP2_BPCC, 4); /* BPCC */
|
||||||
|
|
||||||
|
for (i = 0; i < jp2->numcomps; i++) {
|
||||||
|
cio_write(cio, jp2->comps[i].bpcc, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
box.length = cio_tell(cio) - box.init_pos;
|
||||||
|
cio_seek(cio, box.init_pos);
|
||||||
|
cio_write(cio, box.length, 4); /* L */
|
||||||
|
cio_seek(cio, box.init_pos + box.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static bool jp2_read_bpcc(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
unsigned int i;
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
opj_common_ptr cinfo = jp2->cinfo;
|
||||||
|
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
if (JP2_BPCC != box.type) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Expected BPCC Marker\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < jp2->numcomps; i++) {
|
||||||
|
jp2->comps[i].bpcc = cio_read(cio, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cio_tell(cio) - box.init_pos != box.length) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Error with BPCC Box\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void jp2_write_colr(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
box.init_pos = cio_tell(cio);
|
||||||
|
cio_skip(cio, 4);
|
||||||
|
cio_write(cio, JP2_COLR, 4); /* COLR */
|
||||||
|
|
||||||
|
cio_write(cio, jp2->meth, 1); /* METH */
|
||||||
|
cio_write(cio, jp2->precedence, 1); /* PRECEDENCE */
|
||||||
|
cio_write(cio, jp2->approx, 1); /* APPROX */
|
||||||
|
|
||||||
|
if (jp2->meth == 1) {
|
||||||
|
cio_write(cio, jp2->enumcs, 4); /* EnumCS */
|
||||||
|
} else {
|
||||||
|
cio_write(cio, 0, 1); /* PROFILE (??) */
|
||||||
|
}
|
||||||
|
|
||||||
|
box.length = cio_tell(cio) - box.init_pos;
|
||||||
|
cio_seek(cio, box.init_pos);
|
||||||
|
cio_write(cio, box.length, 4); /* L */
|
||||||
|
cio_seek(cio, box.init_pos + box.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool jp2_read_colr(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
int skip_len;
|
||||||
|
|
||||||
|
opj_common_ptr cinfo = jp2->cinfo;
|
||||||
|
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
do {
|
||||||
|
if (JP2_COLR != box.type) {
|
||||||
|
cio_skip(cio, box.length - 8);
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
}
|
||||||
|
} while(JP2_COLR != box.type);
|
||||||
|
|
||||||
|
jp2->meth = cio_read(cio, 1); /* METH */
|
||||||
|
jp2->precedence = cio_read(cio, 1); /* PRECEDENCE */
|
||||||
|
jp2->approx = cio_read(cio, 1); /* APPROX */
|
||||||
|
|
||||||
|
if (jp2->meth == 1) {
|
||||||
|
jp2->enumcs = cio_read(cio, 4); /* EnumCS */
|
||||||
|
} else {
|
||||||
|
/* skip PROFILE */
|
||||||
|
skip_len = box.init_pos + box.length - cio_tell(cio);
|
||||||
|
if (skip_len < 0) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Error with JP2H box size\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
cio_skip(cio, box.init_pos + box.length - cio_tell(cio));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cio_tell(cio) - box.init_pos != box.length) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Error with BPCC Box\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void jp2_write_jp2h(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
box.init_pos = cio_tell(cio);
|
||||||
|
cio_skip(cio, 4);
|
||||||
|
cio_write(cio, JP2_JP2H, 4); /* JP2H */
|
||||||
|
|
||||||
|
jp2_write_ihdr(jp2, cio);
|
||||||
|
|
||||||
|
if (jp2->bpc == 255) {
|
||||||
|
jp2_write_bpcc(jp2, cio);
|
||||||
|
}
|
||||||
|
jp2_write_colr(jp2, cio);
|
||||||
|
|
||||||
|
box.length = cio_tell(cio) - box.init_pos;
|
||||||
|
cio_seek(cio, box.init_pos);
|
||||||
|
cio_write(cio, box.length, 4); /* L */
|
||||||
|
cio_seek(cio, box.init_pos + box.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool jp2_read_jp2h(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
int skip_len;
|
||||||
|
|
||||||
|
opj_common_ptr cinfo = jp2->cinfo;
|
||||||
|
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
do {
|
||||||
|
if (JP2_JP2H != box.type) {
|
||||||
|
if (box.type == JP2_JP2C) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Expected JP2H Marker\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
cio_skip(cio, box.length - 8);
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
}
|
||||||
|
} while(JP2_JP2H != box.type);
|
||||||
|
|
||||||
|
if (!jp2_read_ihdr(jp2, cio))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (jp2->bpc == 255) {
|
||||||
|
if (!jp2_read_bpcc(jp2, cio))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!jp2_read_colr(jp2, cio))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
skip_len = box.init_pos + box.length - cio_tell(cio);
|
||||||
|
if (skip_len < 0) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Error with JP2H Box\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
cio_skip(cio, box.init_pos + box.length - cio_tell(cio));
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void jp2_write_ftyp(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
unsigned int i;
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
box.init_pos = cio_tell(cio);
|
||||||
|
cio_skip(cio, 4);
|
||||||
|
cio_write(cio, JP2_FTYP, 4); /* FTYP */
|
||||||
|
|
||||||
|
cio_write(cio, jp2->brand, 4); /* BR */
|
||||||
|
cio_write(cio, jp2->minversion, 4); /* MinV */
|
||||||
|
|
||||||
|
for (i = 0; i < jp2->numcl; i++) {
|
||||||
|
cio_write(cio, jp2->cl[i], 4); /* CL */
|
||||||
|
}
|
||||||
|
|
||||||
|
box.length = cio_tell(cio) - box.init_pos;
|
||||||
|
cio_seek(cio, box.init_pos);
|
||||||
|
cio_write(cio, box.length, 4); /* L */
|
||||||
|
cio_seek(cio, box.init_pos + box.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool jp2_read_ftyp(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
int i;
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
opj_common_ptr cinfo = jp2->cinfo;
|
||||||
|
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
|
||||||
|
if (JP2_FTYP != box.type) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Expected FTYP Marker\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
jp2->brand = cio_read(cio, 4); /* BR */
|
||||||
|
jp2->minversion = cio_read(cio, 4); /* MinV */
|
||||||
|
jp2->numcl = (box.length - 16) / 4;
|
||||||
|
jp2->cl = (unsigned int *) opj_malloc(jp2->numcl * sizeof(unsigned int));
|
||||||
|
|
||||||
|
for (i = 0; i < (int)jp2->numcl; i++) {
|
||||||
|
jp2->cl[i] = cio_read(cio, 4); /* CLi */
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cio_tell(cio) - box.init_pos != box.length) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Error with FTYP Box\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int jp2_write_jp2c(opj_jp2_t *jp2, opj_cio_t *cio, opj_image_t *image, opj_codestream_info_t *cstr_info) {
|
||||||
|
unsigned int j2k_codestream_offset, j2k_codestream_length;
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
opj_j2k_t *j2k = jp2->j2k;
|
||||||
|
|
||||||
|
box.init_pos = cio_tell(cio);
|
||||||
|
cio_skip(cio, 4);
|
||||||
|
cio_write(cio, JP2_JP2C, 4); /* JP2C */
|
||||||
|
|
||||||
|
/* J2K encoding */
|
||||||
|
j2k_codestream_offset = cio_tell(cio);
|
||||||
|
if(!j2k_encode(j2k, cio, image, cstr_info)) {
|
||||||
|
opj_event_msg(j2k->cinfo, EVT_ERROR, "Failed to encode image\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
j2k_codestream_length = cio_tell(cio) - j2k_codestream_offset;
|
||||||
|
|
||||||
|
jp2->j2k_codestream_offset = j2k_codestream_offset;
|
||||||
|
jp2->j2k_codestream_length = j2k_codestream_length;
|
||||||
|
|
||||||
|
box.length = 8 + jp2->j2k_codestream_length;
|
||||||
|
cio_seek(cio, box.init_pos);
|
||||||
|
cio_write(cio, box.length, 4); /* L */
|
||||||
|
cio_seek(cio, box.init_pos + box.length);
|
||||||
|
|
||||||
|
return box.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool jp2_read_jp2c(opj_jp2_t *jp2, opj_cio_t *cio, unsigned int *j2k_codestream_length, unsigned int *j2k_codestream_offset) {
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
opj_common_ptr cinfo = jp2->cinfo;
|
||||||
|
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
do {
|
||||||
|
if(JP2_JP2C != box.type) {
|
||||||
|
cio_skip(cio, box.length - 8);
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
}
|
||||||
|
} while(JP2_JP2C != box.type);
|
||||||
|
|
||||||
|
*j2k_codestream_offset = cio_tell(cio);
|
||||||
|
*j2k_codestream_length = box.length - 8;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void jp2_write_jp(opj_cio_t *cio) {
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
box.init_pos = cio_tell(cio);
|
||||||
|
cio_skip(cio, 4);
|
||||||
|
cio_write(cio, JP2_JP, 4); /* JP2 signature */
|
||||||
|
cio_write(cio, 0x0d0a870a, 4);
|
||||||
|
|
||||||
|
box.length = cio_tell(cio) - box.init_pos;
|
||||||
|
cio_seek(cio, box.init_pos);
|
||||||
|
cio_write(cio, box.length, 4); /* L */
|
||||||
|
cio_seek(cio, box.init_pos + box.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool jp2_read_jp(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
opj_jp2_box_t box;
|
||||||
|
|
||||||
|
opj_common_ptr cinfo = jp2->cinfo;
|
||||||
|
|
||||||
|
jp2_read_boxhdr(cinfo, cio, &box);
|
||||||
|
if (JP2_JP != box.type) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Expected JP Marker\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (0x0d0a870a != cio_read(cio, 4)) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Error with JP Marker\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (cio_tell(cio) - box.init_pos != box.length) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Error with JP Box size\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static bool jp2_read_struct(opj_jp2_t *jp2, opj_cio_t *cio) {
|
||||||
|
if (!jp2_read_jp(jp2, cio))
|
||||||
|
return false;
|
||||||
|
if (!jp2_read_ftyp(jp2, cio))
|
||||||
|
return false;
|
||||||
|
if (!jp2_read_jp2h(jp2, cio))
|
||||||
|
return false;
|
||||||
|
if (!jp2_read_jp2c(jp2, cio, &jp2->j2k_codestream_length, &jp2->j2k_codestream_offset))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/* JP2 decoder interface */
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
opj_jp2_t* jp2_create_decompress(opj_common_ptr cinfo) {
|
||||||
|
opj_jp2_t *jp2 = (opj_jp2_t*) opj_calloc(1, sizeof(opj_jp2_t));
|
||||||
|
if(jp2) {
|
||||||
|
jp2->cinfo = cinfo;
|
||||||
|
/* create the J2K codec */
|
||||||
|
jp2->j2k = j2k_create_decompress(cinfo);
|
||||||
|
if(jp2->j2k == NULL) {
|
||||||
|
jp2_destroy_decompress(jp2);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return jp2;
|
||||||
|
}
|
||||||
|
|
||||||
|
void jp2_destroy_decompress(opj_jp2_t *jp2) {
|
||||||
|
if(jp2) {
|
||||||
|
/* destroy the J2K codec */
|
||||||
|
j2k_destroy_decompress(jp2->j2k);
|
||||||
|
|
||||||
|
if(jp2->comps) {
|
||||||
|
opj_free(jp2->comps);
|
||||||
|
}
|
||||||
|
if(jp2->cl) {
|
||||||
|
opj_free(jp2->cl);
|
||||||
|
}
|
||||||
|
opj_free(jp2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void jp2_setup_decoder(opj_jp2_t *jp2, opj_dparameters_t *parameters) {
|
||||||
|
/* setup the J2K codec */
|
||||||
|
j2k_setup_decoder(jp2->j2k, parameters);
|
||||||
|
/* further JP2 initializations go here */
|
||||||
|
}
|
||||||
|
|
||||||
|
opj_image_t* jp2_decode(opj_jp2_t *jp2, opj_cio_t *cio, opj_codestream_info_t *cstr_info) {
|
||||||
|
opj_common_ptr cinfo;
|
||||||
|
opj_image_t *image = NULL;
|
||||||
|
|
||||||
|
if(!jp2 || !cio) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
cinfo = jp2->cinfo;
|
||||||
|
|
||||||
|
/* JP2 decoding */
|
||||||
|
if(!jp2_read_struct(jp2, cio)) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Failed to decode jp2 structure\n");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* J2K decoding */
|
||||||
|
image = j2k_decode(jp2->j2k, cio, cstr_info);
|
||||||
|
if(!image) {
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Failed to decode J2K image\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set Image Color Space */
|
||||||
|
if (jp2->enumcs == 16)
|
||||||
|
image->color_space = CLRSPC_SRGB;
|
||||||
|
else if (jp2->enumcs == 17)
|
||||||
|
image->color_space = CLRSPC_GRAY;
|
||||||
|
else if (jp2->enumcs == 18)
|
||||||
|
image->color_space = CLRSPC_SYCC;
|
||||||
|
else
|
||||||
|
image->color_space = CLRSPC_UNKNOWN;
|
||||||
|
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/* JP2 encoder interface */
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
opj_jp2_t* jp2_create_compress(opj_common_ptr cinfo) {
|
||||||
|
opj_jp2_t *jp2 = (opj_jp2_t*)opj_malloc(sizeof(opj_jp2_t));
|
||||||
|
if(jp2) {
|
||||||
|
jp2->cinfo = cinfo;
|
||||||
|
/* create the J2K codec */
|
||||||
|
jp2->j2k = j2k_create_compress(cinfo);
|
||||||
|
if(jp2->j2k == NULL) {
|
||||||
|
jp2_destroy_compress(jp2);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return jp2;
|
||||||
|
}
|
||||||
|
|
||||||
|
void jp2_destroy_compress(opj_jp2_t *jp2) {
|
||||||
|
if(jp2) {
|
||||||
|
/* destroy the J2K codec */
|
||||||
|
j2k_destroy_compress(jp2->j2k);
|
||||||
|
|
||||||
|
if(jp2->comps) {
|
||||||
|
opj_free(jp2->comps);
|
||||||
|
}
|
||||||
|
if(jp2->cl) {
|
||||||
|
opj_free(jp2->cl);
|
||||||
|
}
|
||||||
|
opj_free(jp2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void jp2_setup_encoder(opj_jp2_t *jp2, opj_cparameters_t *parameters, opj_image_t *image) {
|
||||||
|
int i;
|
||||||
|
int depth_0, sign;
|
||||||
|
|
||||||
|
if(!jp2 || !parameters || !image)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* setup the J2K codec */
|
||||||
|
/* ------------------- */
|
||||||
|
|
||||||
|
/* Check if number of components respects standard */
|
||||||
|
if (image->numcomps < 1 || image->numcomps > 16384) {
|
||||||
|
opj_event_msg(jp2->cinfo, EVT_ERROR, "Invalid number of components specified while setting up JP2 encoder\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
j2k_setup_encoder(jp2->j2k, parameters, image);
|
||||||
|
|
||||||
|
/* setup the JP2 codec */
|
||||||
|
/* ------------------- */
|
||||||
|
|
||||||
|
/* Profile box */
|
||||||
|
|
||||||
|
jp2->brand = JP2_JP2; /* BR */
|
||||||
|
jp2->minversion = 0; /* MinV */
|
||||||
|
jp2->numcl = 1;
|
||||||
|
jp2->cl = (unsigned int*) opj_malloc(jp2->numcl * sizeof(unsigned int));
|
||||||
|
jp2->cl[0] = JP2_JP2; /* CL0 : JP2 */
|
||||||
|
|
||||||
|
/* Image Header box */
|
||||||
|
|
||||||
|
jp2->numcomps = image->numcomps; /* NC */
|
||||||
|
jp2->comps = (opj_jp2_comps_t*) opj_malloc(jp2->numcomps * sizeof(opj_jp2_comps_t));
|
||||||
|
jp2->h = image->y1 - image->y0; /* HEIGHT */
|
||||||
|
jp2->w = image->x1 - image->x0; /* WIDTH */
|
||||||
|
/* BPC */
|
||||||
|
depth_0 = image->comps[0].prec - 1;
|
||||||
|
sign = image->comps[0].sgnd;
|
||||||
|
jp2->bpc = depth_0 + (sign << 7);
|
||||||
|
for (i = 1; i < image->numcomps; i++) {
|
||||||
|
int depth = image->comps[i].prec - 1;
|
||||||
|
sign = image->comps[i].sgnd;
|
||||||
|
if (depth_0 != depth)
|
||||||
|
jp2->bpc = 255;
|
||||||
|
}
|
||||||
|
jp2->C = 7; /* C : Always 7 */
|
||||||
|
jp2->UnkC = 0; /* UnkC, colorspace specified in colr box */
|
||||||
|
jp2->IPR = 0; /* IPR, no intellectual property */
|
||||||
|
|
||||||
|
/* BitsPerComponent box */
|
||||||
|
|
||||||
|
for (i = 0; i < image->numcomps; i++) {
|
||||||
|
jp2->comps[i].bpcc = image->comps[i].prec - 1 + (image->comps[i].sgnd << 7);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Colour Specification box */
|
||||||
|
|
||||||
|
if ((image->numcomps == 1 || image->numcomps == 3) && (jp2->bpc != 255)) {
|
||||||
|
jp2->meth = 1; /* METH: Enumerated colourspace */
|
||||||
|
} else {
|
||||||
|
jp2->meth = 2; /* METH: Restricted ICC profile */
|
||||||
|
}
|
||||||
|
if (jp2->meth == 1) {
|
||||||
|
if (image->color_space == 1)
|
||||||
|
jp2->enumcs = 16; /* sRGB as defined by IEC 61966<36>2<EFBFBD>1 */
|
||||||
|
else if (image->color_space == 2)
|
||||||
|
jp2->enumcs = 17; /* greyscale */
|
||||||
|
else if (image->color_space == 3)
|
||||||
|
jp2->enumcs = 18; /* YUV */
|
||||||
|
} else {
|
||||||
|
jp2->enumcs = 0; /* PROFILE (??) */
|
||||||
|
}
|
||||||
|
jp2->precedence = 0; /* PRECEDENCE */
|
||||||
|
jp2->approx = 0; /* APPROX */
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
bool jp2_encode(opj_jp2_t *jp2, opj_cio_t *cio, opj_image_t *image, opj_codestream_info_t *cstr_info) {
|
||||||
|
|
||||||
|
/* JP2 encoding */
|
||||||
|
|
||||||
|
/* JPEG 2000 Signature box */
|
||||||
|
jp2_write_jp(cio);
|
||||||
|
/* File Type box */
|
||||||
|
jp2_write_ftyp(jp2, cio);
|
||||||
|
/* JP2 Header box */
|
||||||
|
jp2_write_jp2h(jp2, cio);
|
||||||
|
|
||||||
|
/* J2K encoding */
|
||||||
|
|
||||||
|
if(!jp2_write_jp2c(jp2, cio, image, cstr_info)) {
|
||||||
|
opj_event_msg(jp2->cinfo, EVT_ERROR, "Failed to encode image\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
177
extern/libopenjpeg/jp2.h
vendored
Normal file
177
extern/libopenjpeg/jp2.h
vendored
Normal file
@@ -0,0 +1,177 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __JP2_H
|
||||||
|
#define __JP2_H
|
||||||
|
/**
|
||||||
|
@file jp2.h
|
||||||
|
@brief The JPEG-2000 file format Reader/Writer (JP2)
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup JP2 JP2 - JPEG-2000 file format reader/writer */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
#define JPIP_JPIP 0x6a706970
|
||||||
|
|
||||||
|
#define JP2_JP 0x6a502020 /**< JPEG 2000 signature box */
|
||||||
|
#define JP2_FTYP 0x66747970 /**< File type box */
|
||||||
|
#define JP2_JP2H 0x6a703268 /**< JP2 header box */
|
||||||
|
#define JP2_IHDR 0x69686472 /**< Image header box */
|
||||||
|
#define JP2_COLR 0x636f6c72 /**< Colour specification box */
|
||||||
|
#define JP2_JP2C 0x6a703263 /**< Contiguous codestream box */
|
||||||
|
#define JP2_URL 0x75726c20 /**< URL box */
|
||||||
|
#define JP2_DBTL 0x6474626c /**< ??? */
|
||||||
|
#define JP2_BPCC 0x62706363 /**< Bits per component box */
|
||||||
|
#define JP2_JP2 0x6a703220 /**< File type fields */
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
JP2 component
|
||||||
|
*/
|
||||||
|
typedef struct opj_jp2_comps {
|
||||||
|
int depth;
|
||||||
|
int sgnd;
|
||||||
|
int bpcc;
|
||||||
|
} opj_jp2_comps_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
JPEG-2000 file format reader/writer
|
||||||
|
*/
|
||||||
|
typedef struct opj_jp2 {
|
||||||
|
/** codec context */
|
||||||
|
opj_common_ptr cinfo;
|
||||||
|
/** handle to the J2K codec */
|
||||||
|
opj_j2k_t *j2k;
|
||||||
|
unsigned int w;
|
||||||
|
unsigned int h;
|
||||||
|
unsigned int numcomps;
|
||||||
|
unsigned int bpc;
|
||||||
|
unsigned int C;
|
||||||
|
unsigned int UnkC;
|
||||||
|
unsigned int IPR;
|
||||||
|
unsigned int meth;
|
||||||
|
unsigned int approx;
|
||||||
|
unsigned int enumcs;
|
||||||
|
unsigned int precedence;
|
||||||
|
unsigned int brand;
|
||||||
|
unsigned int minversion;
|
||||||
|
unsigned int numcl;
|
||||||
|
unsigned int *cl;
|
||||||
|
opj_jp2_comps_t *comps;
|
||||||
|
unsigned int j2k_codestream_offset;
|
||||||
|
unsigned int j2k_codestream_length;
|
||||||
|
} opj_jp2_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
JP2 Box
|
||||||
|
*/
|
||||||
|
typedef struct opj_jp2_box {
|
||||||
|
int length;
|
||||||
|
int type;
|
||||||
|
int init_pos;
|
||||||
|
} opj_jp2_box_t;
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Write the JP2H box - JP2 Header box (used in MJ2)
|
||||||
|
@param jp2 JP2 handle
|
||||||
|
@param cio Output buffer stream
|
||||||
|
*/
|
||||||
|
void jp2_write_jp2h(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
/**
|
||||||
|
Read the JP2H box - JP2 Header box (used in MJ2)
|
||||||
|
@param jp2 JP2 handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
bool jp2_read_jp2h(opj_jp2_t *jp2, opj_cio_t *cio);
|
||||||
|
/**
|
||||||
|
Creates a JP2 decompression structure
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@return Returns a handle to a JP2 decompressor if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_jp2_t* jp2_create_decompress(opj_common_ptr cinfo);
|
||||||
|
/**
|
||||||
|
Destroy a JP2 decompressor handle
|
||||||
|
@param jp2 JP2 decompressor handle to destroy
|
||||||
|
*/
|
||||||
|
void jp2_destroy_decompress(opj_jp2_t *jp2);
|
||||||
|
/**
|
||||||
|
Setup the decoder decoding parameters using user parameters.
|
||||||
|
Decoding parameters are returned in jp2->j2k->cp.
|
||||||
|
@param jp2 JP2 decompressor handle
|
||||||
|
@param parameters decompression parameters
|
||||||
|
*/
|
||||||
|
void jp2_setup_decoder(opj_jp2_t *jp2, opj_dparameters_t *parameters);
|
||||||
|
/**
|
||||||
|
Decode an image from a JPEG-2000 file stream
|
||||||
|
@param jp2 JP2 decompressor handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@param cstr_info Codestream information structure if required, NULL otherwise
|
||||||
|
@return Returns a decoded image if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_image_t* jp2_decode(opj_jp2_t *jp2, opj_cio_t *cio, opj_codestream_info_t *cstr_info);
|
||||||
|
/**
|
||||||
|
Creates a JP2 compression structure
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@return Returns a handle to a JP2 compressor if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_jp2_t* jp2_create_compress(opj_common_ptr cinfo);
|
||||||
|
/**
|
||||||
|
Destroy a JP2 compressor handle
|
||||||
|
@param jp2 JP2 compressor handle to destroy
|
||||||
|
*/
|
||||||
|
void jp2_destroy_compress(opj_jp2_t *jp2);
|
||||||
|
/**
|
||||||
|
Setup the encoder parameters using the current image and using user parameters.
|
||||||
|
Coding parameters are returned in jp2->j2k->cp.
|
||||||
|
@param jp2 JP2 compressor handle
|
||||||
|
@param parameters compression parameters
|
||||||
|
@param image input filled image
|
||||||
|
*/
|
||||||
|
void jp2_setup_encoder(opj_jp2_t *jp2, opj_cparameters_t *parameters, opj_image_t *image);
|
||||||
|
/**
|
||||||
|
Encode an image into a JPEG-2000 file stream
|
||||||
|
@param jp2 JP2 compressor handle
|
||||||
|
@param cio Output buffer stream
|
||||||
|
@param image Image to encode
|
||||||
|
@param cstr_info Codestream information structure if required, NULL otherwise
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
bool jp2_encode(opj_jp2_t *jp2, opj_cio_t *cio, opj_image_t *image, opj_codestream_info_t *cstr_info);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __JP2_H */
|
||||||
|
|
||||||
155
extern/libopenjpeg/jpt.c
vendored
Normal file
155
extern/libopenjpeg/jpt.c
vendored
Normal file
@@ -0,0 +1,155 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read the information contains in VBAS [JPP/JPT stream message header]
|
||||||
|
* Store information (7 bits) in value
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
unsigned int jpt_read_VBAS_info(opj_cio_t *cio, unsigned int value) {
|
||||||
|
unsigned char elmt;
|
||||||
|
|
||||||
|
elmt = cio_read(cio, 1);
|
||||||
|
while ((elmt >> 7) == 1) {
|
||||||
|
value = (value << 7);
|
||||||
|
value |= (elmt & 0x7f);
|
||||||
|
elmt = cio_read(cio, 1);
|
||||||
|
}
|
||||||
|
value = (value << 7);
|
||||||
|
value |= (elmt & 0x7f);
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Initialize the value of the message header structure
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
void jpt_init_msg_header(opj_jpt_msg_header_t * header) {
|
||||||
|
header->Id = 0; /* In-class Identifier */
|
||||||
|
header->last_byte = 0; /* Last byte information */
|
||||||
|
header->Class_Id = 0; /* Class Identifier */
|
||||||
|
header->CSn_Id = 0; /* CSn : index identifier */
|
||||||
|
header->Msg_offset = 0; /* Message offset */
|
||||||
|
header->Msg_length = 0; /* Message length */
|
||||||
|
header->Layer_nb = 0; /* Auxiliary for JPP case */
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Re-initialize the value of the message header structure
|
||||||
|
*
|
||||||
|
* Only parameters always present in message header
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
void jpt_reinit_msg_header(opj_jpt_msg_header_t * header) {
|
||||||
|
header->Id = 0; /* In-class Identifier */
|
||||||
|
header->last_byte = 0; /* Last byte information */
|
||||||
|
header->Msg_offset = 0; /* Message offset */
|
||||||
|
header->Msg_length = 0; /* Message length */
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read the message header for a JPP/JPT - stream
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
void jpt_read_msg_header(opj_common_ptr cinfo, opj_cio_t *cio, opj_jpt_msg_header_t *header) {
|
||||||
|
unsigned char elmt, Class = 0, CSn = 0;
|
||||||
|
jpt_reinit_msg_header(header);
|
||||||
|
|
||||||
|
/* ------------- */
|
||||||
|
/* VBAS : Bin-ID */
|
||||||
|
/* ------------- */
|
||||||
|
elmt = cio_read(cio, 1);
|
||||||
|
|
||||||
|
/* See for Class and CSn */
|
||||||
|
switch ((elmt >> 5) & 0x03) {
|
||||||
|
case 0:
|
||||||
|
opj_event_msg(cinfo, EVT_ERROR, "Forbidden value encounter in message header !!\n");
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
Class = 0;
|
||||||
|
CSn = 0;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
Class = 1;
|
||||||
|
CSn = 0;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
Class = 1;
|
||||||
|
CSn = 1;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* see information on bits 'c' [p 10 : A.2.1 general, ISO/IEC FCD 15444-9] */
|
||||||
|
if (((elmt >> 4) & 0x01) == 1)
|
||||||
|
header->last_byte = 1;
|
||||||
|
|
||||||
|
/* In-class identifier */
|
||||||
|
header->Id |= (elmt & 0x0f);
|
||||||
|
if ((elmt >> 7) == 1)
|
||||||
|
header->Id = jpt_read_VBAS_info(cio, header->Id);
|
||||||
|
|
||||||
|
/* ------------ */
|
||||||
|
/* VBAS : Class */
|
||||||
|
/* ------------ */
|
||||||
|
if (Class == 1) {
|
||||||
|
header->Class_Id = 0;
|
||||||
|
header->Class_Id = jpt_read_VBAS_info(cio, header->Class_Id);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ---------- */
|
||||||
|
/* VBAS : CSn */
|
||||||
|
/* ---------- */
|
||||||
|
if (CSn == 1) {
|
||||||
|
header->CSn_Id = 0;
|
||||||
|
header->CSn_Id = jpt_read_VBAS_info(cio, header->CSn_Id);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ----------------- */
|
||||||
|
/* VBAS : Msg_offset */
|
||||||
|
/* ----------------- */
|
||||||
|
header->Msg_offset = jpt_read_VBAS_info(cio, header->Msg_offset);
|
||||||
|
|
||||||
|
/* ----------------- */
|
||||||
|
/* VBAS : Msg_length */
|
||||||
|
/* ----------------- */
|
||||||
|
header->Msg_length = jpt_read_VBAS_info(cio, header->Msg_length);
|
||||||
|
|
||||||
|
/* ---------- */
|
||||||
|
/* VBAS : Aux */
|
||||||
|
/* ---------- */
|
||||||
|
if ((header->Class_Id & 0x01) == 1) {
|
||||||
|
header->Layer_nb = 0;
|
||||||
|
header->Layer_nb = jpt_read_VBAS_info(cio, header->Layer_nb);
|
||||||
|
}
|
||||||
|
}
|
||||||
75
extern/libopenjpeg/jpt.h
vendored
Normal file
75
extern/libopenjpeg/jpt.h
vendored
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __JPT_H
|
||||||
|
#define __JPT_H
|
||||||
|
/**
|
||||||
|
@file jpt.h
|
||||||
|
@brief JPT-stream reader (JPEG 2000, JPIP)
|
||||||
|
|
||||||
|
JPT-stream functions are implemented in J2K.C.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Message Header JPT stream structure
|
||||||
|
*/
|
||||||
|
typedef struct opj_jpt_msg_header {
|
||||||
|
/** In-class Identifier */
|
||||||
|
unsigned int Id;
|
||||||
|
/** Last byte information */
|
||||||
|
unsigned int last_byte;
|
||||||
|
/** Class Identifier */
|
||||||
|
unsigned int Class_Id;
|
||||||
|
/** CSn : index identifier */
|
||||||
|
unsigned int CSn_Id;
|
||||||
|
/** Message offset */
|
||||||
|
unsigned int Msg_offset;
|
||||||
|
/** Message length */
|
||||||
|
unsigned int Msg_length;
|
||||||
|
/** Auxiliary for JPP case */
|
||||||
|
unsigned int Layer_nb;
|
||||||
|
} opj_jpt_msg_header_t;
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
Initialize the value of the message header structure
|
||||||
|
@param header Message header structure
|
||||||
|
*/
|
||||||
|
void jpt_init_msg_header(opj_jpt_msg_header_t * header);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Read the message header for a JPP/JPT - stream
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@param cio CIO handle
|
||||||
|
@param header Message header structure
|
||||||
|
*/
|
||||||
|
void jpt_read_msg_header(opj_common_ptr cinfo, opj_cio_t *cio, opj_jpt_msg_header_t *header);
|
||||||
|
|
||||||
|
#endif
|
||||||
30
extern/libopenjpeg/license.txt
vendored
Normal file
30
extern/libopenjpeg/license.txt
vendored
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
148
extern/libopenjpeg/mct.c
vendored
Normal file
148
extern/libopenjpeg/mct.c
vendored
Normal file
@@ -0,0 +1,148 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* This table contains the norms of the basis function of the reversible MCT. */
|
||||||
|
/* </summary> */
|
||||||
|
static const double mct_norms[3] = { 1.732, .8292, .8292 };
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* This table contains the norms of the basis function of the irreversible MCT. */
|
||||||
|
/* </summary> */
|
||||||
|
static const double mct_norms_real[3] = { 1.732, 1.805, 1.573 };
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Foward reversible MCT. */
|
||||||
|
/* </summary> */
|
||||||
|
void mct_encode(
|
||||||
|
int* restrict c0,
|
||||||
|
int* restrict c1,
|
||||||
|
int* restrict c2,
|
||||||
|
int n)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < n; ++i) {
|
||||||
|
int r = c0[i];
|
||||||
|
int g = c1[i];
|
||||||
|
int b = c2[i];
|
||||||
|
int y = (r + (g * 2) + b) >> 2;
|
||||||
|
int u = b - g;
|
||||||
|
int v = r - g;
|
||||||
|
c0[i] = y;
|
||||||
|
c1[i] = u;
|
||||||
|
c2[i] = v;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse reversible MCT. */
|
||||||
|
/* </summary> */
|
||||||
|
void mct_decode(
|
||||||
|
int* restrict c0,
|
||||||
|
int* restrict c1,
|
||||||
|
int* restrict c2,
|
||||||
|
int n)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < n; ++i) {
|
||||||
|
int y = c0[i];
|
||||||
|
int u = c1[i];
|
||||||
|
int v = c2[i];
|
||||||
|
int g = y - ((u + v) >> 2);
|
||||||
|
int r = v + g;
|
||||||
|
int b = u + g;
|
||||||
|
c0[i] = r;
|
||||||
|
c1[i] = g;
|
||||||
|
c2[i] = b;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Get norm of basis function of reversible MCT. */
|
||||||
|
/* </summary> */
|
||||||
|
double mct_getnorm(int compno) {
|
||||||
|
return mct_norms[compno];
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Foward irreversible MCT. */
|
||||||
|
/* </summary> */
|
||||||
|
void mct_encode_real(
|
||||||
|
int* restrict c0,
|
||||||
|
int* restrict c1,
|
||||||
|
int* restrict c2,
|
||||||
|
int n)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < n; ++i) {
|
||||||
|
int r = c0[i];
|
||||||
|
int g = c1[i];
|
||||||
|
int b = c2[i];
|
||||||
|
int y = fix_mul(r, 2449) + fix_mul(g, 4809) + fix_mul(b, 934);
|
||||||
|
int u = -fix_mul(r, 1382) - fix_mul(g, 2714) + fix_mul(b, 4096);
|
||||||
|
int v = fix_mul(r, 4096) - fix_mul(g, 3430) - fix_mul(b, 666);
|
||||||
|
c0[i] = y;
|
||||||
|
c1[i] = u;
|
||||||
|
c2[i] = v;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Inverse irreversible MCT. */
|
||||||
|
/* </summary> */
|
||||||
|
void mct_decode_real(
|
||||||
|
float* restrict c0,
|
||||||
|
float* restrict c1,
|
||||||
|
float* restrict c2,
|
||||||
|
int n)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < n; ++i) {
|
||||||
|
float y = c0[i];
|
||||||
|
float u = c1[i];
|
||||||
|
float v = c2[i];
|
||||||
|
float r = y + (v * 1.402f);
|
||||||
|
float g = y - (u * 0.34413f) - (v * (0.71414f));
|
||||||
|
float b = y + (u * 1.772f);
|
||||||
|
c0[i] = r;
|
||||||
|
c1[i] = g;
|
||||||
|
c2[i] = b;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* Get norm of basis function of irreversible MCT. */
|
||||||
|
/* </summary> */
|
||||||
|
double mct_getnorm_real(int compno) {
|
||||||
|
return mct_norms_real[compno];
|
||||||
|
}
|
||||||
98
extern/libopenjpeg/mct.h
vendored
Normal file
98
extern/libopenjpeg/mct.h
vendored
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __MCT_H
|
||||||
|
#define __MCT_H
|
||||||
|
/**
|
||||||
|
@file mct.h
|
||||||
|
@brief Implementation of a multi-component transforms (MCT)
|
||||||
|
|
||||||
|
The functions in MCT.C have for goal to realize reversible and irreversible multicomponent
|
||||||
|
transform. The functions in MCT.C are used by some function in TCD.C.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup MCT MCT - Implementation of a multi-component transform */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Apply a reversible multi-component transform to an image
|
||||||
|
@param c0 Samples for red component
|
||||||
|
@param c1 Samples for green component
|
||||||
|
@param c2 Samples blue component
|
||||||
|
@param n Number of samples for each component
|
||||||
|
*/
|
||||||
|
void mct_encode(int *c0, int *c1, int *c2, int n);
|
||||||
|
/**
|
||||||
|
Apply a reversible multi-component inverse transform to an image
|
||||||
|
@param c0 Samples for luminance component
|
||||||
|
@param c1 Samples for red chrominance component
|
||||||
|
@param c2 Samples for blue chrominance component
|
||||||
|
@param n Number of samples for each component
|
||||||
|
*/
|
||||||
|
void mct_decode(int *c0, int *c1, int *c2, int n);
|
||||||
|
/**
|
||||||
|
Get norm of the basis function used for the reversible multi-component transform
|
||||||
|
@param compno Number of the component (0->Y, 1->U, 2->V)
|
||||||
|
@return
|
||||||
|
*/
|
||||||
|
double mct_getnorm(int compno);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Apply an irreversible multi-component transform to an image
|
||||||
|
@param c0 Samples for red component
|
||||||
|
@param c1 Samples for green component
|
||||||
|
@param c2 Samples blue component
|
||||||
|
@param n Number of samples for each component
|
||||||
|
*/
|
||||||
|
void mct_encode_real(int *c0, int *c1, int *c2, int n);
|
||||||
|
/**
|
||||||
|
Apply an irreversible multi-component inverse transform to an image
|
||||||
|
@param c0 Samples for luminance component
|
||||||
|
@param c1 Samples for red chrominance component
|
||||||
|
@param c2 Samples for blue chrominance component
|
||||||
|
@param n Number of samples for each component
|
||||||
|
*/
|
||||||
|
void mct_decode_real(float* c0, float* c1, float* c2, int n);
|
||||||
|
/**
|
||||||
|
Get norm of the basis function used for the irreversible multi-component transform
|
||||||
|
@param compno Number of the component (0->Y, 1->U, 2->V)
|
||||||
|
@return
|
||||||
|
*/
|
||||||
|
double mct_getnorm_real(int compno);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __MCT_H */
|
||||||
538
extern/libopenjpeg/mqc.c
vendored
Normal file
538
extern/libopenjpeg/mqc.c
vendored
Normal file
@@ -0,0 +1,538 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/** @defgroup MQC MQC - Implementation of an MQ-Coder */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Local static functions */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Output a byte, doing bit-stuffing if necessary.
|
||||||
|
After a 0xff byte, the next byte must be smaller than 0x90.
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
static void mqc_byteout(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Renormalize mqc->a and mqc->c while encoding, so that mqc->a stays between 0x8000 and 0x10000
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
static void mqc_renorme(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Encode the most probable symbol
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
static void mqc_codemps(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Encode the most least symbol
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
static void mqc_codelps(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Fill mqc->c with 1's for flushing
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
static void mqc_setbits(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
FIXME: documentation ???
|
||||||
|
@param mqc MQC handle
|
||||||
|
@return
|
||||||
|
*/
|
||||||
|
static int mqc_mpsexchange(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
FIXME: documentation ???
|
||||||
|
@param mqc MQC handle
|
||||||
|
@return
|
||||||
|
*/
|
||||||
|
static int mqc_lpsexchange(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Input a byte
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
static void mqc_bytein(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Renormalize mqc->a and mqc->c while decoding
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
static void mqc_renormd(opj_mqc_t *mqc);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/* <summary> */
|
||||||
|
/* This array defines all the possible states for a context. */
|
||||||
|
/* </summary> */
|
||||||
|
static opj_mqc_state_t mqc_states[47 * 2] = {
|
||||||
|
{0x5601, 0, &mqc_states[2], &mqc_states[3]},
|
||||||
|
{0x5601, 1, &mqc_states[3], &mqc_states[2]},
|
||||||
|
{0x3401, 0, &mqc_states[4], &mqc_states[12]},
|
||||||
|
{0x3401, 1, &mqc_states[5], &mqc_states[13]},
|
||||||
|
{0x1801, 0, &mqc_states[6], &mqc_states[18]},
|
||||||
|
{0x1801, 1, &mqc_states[7], &mqc_states[19]},
|
||||||
|
{0x0ac1, 0, &mqc_states[8], &mqc_states[24]},
|
||||||
|
{0x0ac1, 1, &mqc_states[9], &mqc_states[25]},
|
||||||
|
{0x0521, 0, &mqc_states[10], &mqc_states[58]},
|
||||||
|
{0x0521, 1, &mqc_states[11], &mqc_states[59]},
|
||||||
|
{0x0221, 0, &mqc_states[76], &mqc_states[66]},
|
||||||
|
{0x0221, 1, &mqc_states[77], &mqc_states[67]},
|
||||||
|
{0x5601, 0, &mqc_states[14], &mqc_states[13]},
|
||||||
|
{0x5601, 1, &mqc_states[15], &mqc_states[12]},
|
||||||
|
{0x5401, 0, &mqc_states[16], &mqc_states[28]},
|
||||||
|
{0x5401, 1, &mqc_states[17], &mqc_states[29]},
|
||||||
|
{0x4801, 0, &mqc_states[18], &mqc_states[28]},
|
||||||
|
{0x4801, 1, &mqc_states[19], &mqc_states[29]},
|
||||||
|
{0x3801, 0, &mqc_states[20], &mqc_states[28]},
|
||||||
|
{0x3801, 1, &mqc_states[21], &mqc_states[29]},
|
||||||
|
{0x3001, 0, &mqc_states[22], &mqc_states[34]},
|
||||||
|
{0x3001, 1, &mqc_states[23], &mqc_states[35]},
|
||||||
|
{0x2401, 0, &mqc_states[24], &mqc_states[36]},
|
||||||
|
{0x2401, 1, &mqc_states[25], &mqc_states[37]},
|
||||||
|
{0x1c01, 0, &mqc_states[26], &mqc_states[40]},
|
||||||
|
{0x1c01, 1, &mqc_states[27], &mqc_states[41]},
|
||||||
|
{0x1601, 0, &mqc_states[58], &mqc_states[42]},
|
||||||
|
{0x1601, 1, &mqc_states[59], &mqc_states[43]},
|
||||||
|
{0x5601, 0, &mqc_states[30], &mqc_states[29]},
|
||||||
|
{0x5601, 1, &mqc_states[31], &mqc_states[28]},
|
||||||
|
{0x5401, 0, &mqc_states[32], &mqc_states[28]},
|
||||||
|
{0x5401, 1, &mqc_states[33], &mqc_states[29]},
|
||||||
|
{0x5101, 0, &mqc_states[34], &mqc_states[30]},
|
||||||
|
{0x5101, 1, &mqc_states[35], &mqc_states[31]},
|
||||||
|
{0x4801, 0, &mqc_states[36], &mqc_states[32]},
|
||||||
|
{0x4801, 1, &mqc_states[37], &mqc_states[33]},
|
||||||
|
{0x3801, 0, &mqc_states[38], &mqc_states[34]},
|
||||||
|
{0x3801, 1, &mqc_states[39], &mqc_states[35]},
|
||||||
|
{0x3401, 0, &mqc_states[40], &mqc_states[36]},
|
||||||
|
{0x3401, 1, &mqc_states[41], &mqc_states[37]},
|
||||||
|
{0x3001, 0, &mqc_states[42], &mqc_states[38]},
|
||||||
|
{0x3001, 1, &mqc_states[43], &mqc_states[39]},
|
||||||
|
{0x2801, 0, &mqc_states[44], &mqc_states[38]},
|
||||||
|
{0x2801, 1, &mqc_states[45], &mqc_states[39]},
|
||||||
|
{0x2401, 0, &mqc_states[46], &mqc_states[40]},
|
||||||
|
{0x2401, 1, &mqc_states[47], &mqc_states[41]},
|
||||||
|
{0x2201, 0, &mqc_states[48], &mqc_states[42]},
|
||||||
|
{0x2201, 1, &mqc_states[49], &mqc_states[43]},
|
||||||
|
{0x1c01, 0, &mqc_states[50], &mqc_states[44]},
|
||||||
|
{0x1c01, 1, &mqc_states[51], &mqc_states[45]},
|
||||||
|
{0x1801, 0, &mqc_states[52], &mqc_states[46]},
|
||||||
|
{0x1801, 1, &mqc_states[53], &mqc_states[47]},
|
||||||
|
{0x1601, 0, &mqc_states[54], &mqc_states[48]},
|
||||||
|
{0x1601, 1, &mqc_states[55], &mqc_states[49]},
|
||||||
|
{0x1401, 0, &mqc_states[56], &mqc_states[50]},
|
||||||
|
{0x1401, 1, &mqc_states[57], &mqc_states[51]},
|
||||||
|
{0x1201, 0, &mqc_states[58], &mqc_states[52]},
|
||||||
|
{0x1201, 1, &mqc_states[59], &mqc_states[53]},
|
||||||
|
{0x1101, 0, &mqc_states[60], &mqc_states[54]},
|
||||||
|
{0x1101, 1, &mqc_states[61], &mqc_states[55]},
|
||||||
|
{0x0ac1, 0, &mqc_states[62], &mqc_states[56]},
|
||||||
|
{0x0ac1, 1, &mqc_states[63], &mqc_states[57]},
|
||||||
|
{0x09c1, 0, &mqc_states[64], &mqc_states[58]},
|
||||||
|
{0x09c1, 1, &mqc_states[65], &mqc_states[59]},
|
||||||
|
{0x08a1, 0, &mqc_states[66], &mqc_states[60]},
|
||||||
|
{0x08a1, 1, &mqc_states[67], &mqc_states[61]},
|
||||||
|
{0x0521, 0, &mqc_states[68], &mqc_states[62]},
|
||||||
|
{0x0521, 1, &mqc_states[69], &mqc_states[63]},
|
||||||
|
{0x0441, 0, &mqc_states[70], &mqc_states[64]},
|
||||||
|
{0x0441, 1, &mqc_states[71], &mqc_states[65]},
|
||||||
|
{0x02a1, 0, &mqc_states[72], &mqc_states[66]},
|
||||||
|
{0x02a1, 1, &mqc_states[73], &mqc_states[67]},
|
||||||
|
{0x0221, 0, &mqc_states[74], &mqc_states[68]},
|
||||||
|
{0x0221, 1, &mqc_states[75], &mqc_states[69]},
|
||||||
|
{0x0141, 0, &mqc_states[76], &mqc_states[70]},
|
||||||
|
{0x0141, 1, &mqc_states[77], &mqc_states[71]},
|
||||||
|
{0x0111, 0, &mqc_states[78], &mqc_states[72]},
|
||||||
|
{0x0111, 1, &mqc_states[79], &mqc_states[73]},
|
||||||
|
{0x0085, 0, &mqc_states[80], &mqc_states[74]},
|
||||||
|
{0x0085, 1, &mqc_states[81], &mqc_states[75]},
|
||||||
|
{0x0049, 0, &mqc_states[82], &mqc_states[76]},
|
||||||
|
{0x0049, 1, &mqc_states[83], &mqc_states[77]},
|
||||||
|
{0x0025, 0, &mqc_states[84], &mqc_states[78]},
|
||||||
|
{0x0025, 1, &mqc_states[85], &mqc_states[79]},
|
||||||
|
{0x0015, 0, &mqc_states[86], &mqc_states[80]},
|
||||||
|
{0x0015, 1, &mqc_states[87], &mqc_states[81]},
|
||||||
|
{0x0009, 0, &mqc_states[88], &mqc_states[82]},
|
||||||
|
{0x0009, 1, &mqc_states[89], &mqc_states[83]},
|
||||||
|
{0x0005, 0, &mqc_states[90], &mqc_states[84]},
|
||||||
|
{0x0005, 1, &mqc_states[91], &mqc_states[85]},
|
||||||
|
{0x0001, 0, &mqc_states[90], &mqc_states[86]},
|
||||||
|
{0x0001, 1, &mqc_states[91], &mqc_states[87]},
|
||||||
|
{0x5601, 0, &mqc_states[92], &mqc_states[92]},
|
||||||
|
{0x5601, 1, &mqc_states[93], &mqc_states[93]},
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
local functions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
static void mqc_byteout(opj_mqc_t *mqc) {
|
||||||
|
if (*mqc->bp == 0xff) {
|
||||||
|
mqc->bp++;
|
||||||
|
*mqc->bp = mqc->c >> 20;
|
||||||
|
mqc->c &= 0xfffff;
|
||||||
|
mqc->ct = 7;
|
||||||
|
} else {
|
||||||
|
if ((mqc->c & 0x8000000) == 0) { /* ((mqc->c&0x8000000)==0) CHANGE */
|
||||||
|
mqc->bp++;
|
||||||
|
*mqc->bp = mqc->c >> 19;
|
||||||
|
mqc->c &= 0x7ffff;
|
||||||
|
mqc->ct = 8;
|
||||||
|
} else {
|
||||||
|
(*mqc->bp)++;
|
||||||
|
if (*mqc->bp == 0xff) {
|
||||||
|
mqc->c &= 0x7ffffff;
|
||||||
|
mqc->bp++;
|
||||||
|
*mqc->bp = mqc->c >> 20;
|
||||||
|
mqc->c &= 0xfffff;
|
||||||
|
mqc->ct = 7;
|
||||||
|
} else {
|
||||||
|
mqc->bp++;
|
||||||
|
*mqc->bp = mqc->c >> 19;
|
||||||
|
mqc->c &= 0x7ffff;
|
||||||
|
mqc->ct = 8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mqc_renorme(opj_mqc_t *mqc) {
|
||||||
|
do {
|
||||||
|
mqc->a <<= 1;
|
||||||
|
mqc->c <<= 1;
|
||||||
|
mqc->ct--;
|
||||||
|
if (mqc->ct == 0) {
|
||||||
|
mqc_byteout(mqc);
|
||||||
|
}
|
||||||
|
} while ((mqc->a & 0x8000) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mqc_codemps(opj_mqc_t *mqc) {
|
||||||
|
mqc->a -= (*mqc->curctx)->qeval;
|
||||||
|
if ((mqc->a & 0x8000) == 0) {
|
||||||
|
if (mqc->a < (*mqc->curctx)->qeval) {
|
||||||
|
mqc->a = (*mqc->curctx)->qeval;
|
||||||
|
} else {
|
||||||
|
mqc->c += (*mqc->curctx)->qeval;
|
||||||
|
}
|
||||||
|
*mqc->curctx = (*mqc->curctx)->nmps;
|
||||||
|
mqc_renorme(mqc);
|
||||||
|
} else {
|
||||||
|
mqc->c += (*mqc->curctx)->qeval;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mqc_codelps(opj_mqc_t *mqc) {
|
||||||
|
mqc->a -= (*mqc->curctx)->qeval;
|
||||||
|
if (mqc->a < (*mqc->curctx)->qeval) {
|
||||||
|
mqc->c += (*mqc->curctx)->qeval;
|
||||||
|
} else {
|
||||||
|
mqc->a = (*mqc->curctx)->qeval;
|
||||||
|
}
|
||||||
|
*mqc->curctx = (*mqc->curctx)->nlps;
|
||||||
|
mqc_renorme(mqc);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mqc_setbits(opj_mqc_t *mqc) {
|
||||||
|
unsigned int tempc = mqc->c + mqc->a;
|
||||||
|
mqc->c |= 0xffff;
|
||||||
|
if (mqc->c >= tempc) {
|
||||||
|
mqc->c -= 0x8000;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mqc_mpsexchange(opj_mqc_t *mqc) {
|
||||||
|
int d;
|
||||||
|
if (mqc->a < (*mqc->curctx)->qeval) {
|
||||||
|
d = 1 - (*mqc->curctx)->mps;
|
||||||
|
*mqc->curctx = (*mqc->curctx)->nlps;
|
||||||
|
} else {
|
||||||
|
d = (*mqc->curctx)->mps;
|
||||||
|
*mqc->curctx = (*mqc->curctx)->nmps;
|
||||||
|
}
|
||||||
|
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mqc_lpsexchange(opj_mqc_t *mqc) {
|
||||||
|
int d;
|
||||||
|
if (mqc->a < (*mqc->curctx)->qeval) {
|
||||||
|
mqc->a = (*mqc->curctx)->qeval;
|
||||||
|
d = (*mqc->curctx)->mps;
|
||||||
|
*mqc->curctx = (*mqc->curctx)->nmps;
|
||||||
|
} else {
|
||||||
|
mqc->a = (*mqc->curctx)->qeval;
|
||||||
|
d = 1 - (*mqc->curctx)->mps;
|
||||||
|
*mqc->curctx = (*mqc->curctx)->nlps;
|
||||||
|
}
|
||||||
|
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mqc_bytein(opj_mqc_t *mqc) {
|
||||||
|
if (mqc->bp != mqc->end) {
|
||||||
|
unsigned int c;
|
||||||
|
if (mqc->bp + 1 != mqc->end) {
|
||||||
|
c = *(mqc->bp + 1);
|
||||||
|
} else {
|
||||||
|
c = 0xff;
|
||||||
|
}
|
||||||
|
if (*mqc->bp == 0xff) {
|
||||||
|
if (c > 0x8f) {
|
||||||
|
mqc->c += 0xff00;
|
||||||
|
mqc->ct = 8;
|
||||||
|
} else {
|
||||||
|
mqc->bp++;
|
||||||
|
mqc->c += c << 9;
|
||||||
|
mqc->ct = 7;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
mqc->bp++;
|
||||||
|
mqc->c += c << 8;
|
||||||
|
mqc->ct = 8;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
mqc->c += 0xff00;
|
||||||
|
mqc->ct = 8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mqc_renormd(opj_mqc_t *mqc) {
|
||||||
|
do {
|
||||||
|
if (mqc->ct == 0) {
|
||||||
|
mqc_bytein(mqc);
|
||||||
|
}
|
||||||
|
mqc->a <<= 1;
|
||||||
|
mqc->c <<= 1;
|
||||||
|
mqc->ct--;
|
||||||
|
} while (mqc->a < 0x8000);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
MQ-Coder interface
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
opj_mqc_t* mqc_create(void) {
|
||||||
|
opj_mqc_t *mqc = (opj_mqc_t*)opj_malloc(sizeof(opj_mqc_t));
|
||||||
|
return mqc;
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_destroy(opj_mqc_t *mqc) {
|
||||||
|
if(mqc) {
|
||||||
|
opj_free(mqc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int mqc_numbytes(opj_mqc_t *mqc) {
|
||||||
|
return mqc->bp - mqc->start;
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_init_enc(opj_mqc_t *mqc, unsigned char *bp) {
|
||||||
|
mqc_setcurctx(mqc, 0);
|
||||||
|
mqc->a = 0x8000;
|
||||||
|
mqc->c = 0;
|
||||||
|
mqc->bp = bp - 1;
|
||||||
|
mqc->ct = 12;
|
||||||
|
if (*mqc->bp == 0xff) {
|
||||||
|
mqc->ct = 13;
|
||||||
|
}
|
||||||
|
mqc->start = bp;
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_encode(opj_mqc_t *mqc, int d) {
|
||||||
|
if ((*mqc->curctx)->mps == d) {
|
||||||
|
mqc_codemps(mqc);
|
||||||
|
} else {
|
||||||
|
mqc_codelps(mqc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_flush(opj_mqc_t *mqc) {
|
||||||
|
mqc_setbits(mqc);
|
||||||
|
mqc->c <<= mqc->ct;
|
||||||
|
mqc_byteout(mqc);
|
||||||
|
mqc->c <<= mqc->ct;
|
||||||
|
mqc_byteout(mqc);
|
||||||
|
|
||||||
|
if (*mqc->bp != 0xff) {
|
||||||
|
mqc->bp++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_bypass_init_enc(opj_mqc_t *mqc) {
|
||||||
|
mqc->c = 0;
|
||||||
|
mqc->ct = 8;
|
||||||
|
/*if (*mqc->bp == 0xff) {
|
||||||
|
mqc->ct = 7;
|
||||||
|
} */
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_bypass_enc(opj_mqc_t *mqc, int d) {
|
||||||
|
mqc->ct--;
|
||||||
|
mqc->c = mqc->c + (d << mqc->ct);
|
||||||
|
if (mqc->ct == 0) {
|
||||||
|
mqc->bp++;
|
||||||
|
*mqc->bp = mqc->c;
|
||||||
|
mqc->ct = 8;
|
||||||
|
if (*mqc->bp == 0xff) {
|
||||||
|
mqc->ct = 7;
|
||||||
|
}
|
||||||
|
mqc->c = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int mqc_bypass_flush_enc(opj_mqc_t *mqc) {
|
||||||
|
unsigned char bit_padding;
|
||||||
|
|
||||||
|
bit_padding = 0;
|
||||||
|
|
||||||
|
if (mqc->ct != 0) {
|
||||||
|
while (mqc->ct > 0) {
|
||||||
|
mqc->ct--;
|
||||||
|
mqc->c += bit_padding << mqc->ct;
|
||||||
|
bit_padding = (bit_padding + 1) & 0x01;
|
||||||
|
}
|
||||||
|
mqc->bp++;
|
||||||
|
*mqc->bp = mqc->c;
|
||||||
|
mqc->ct = 8;
|
||||||
|
mqc->c = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_reset_enc(opj_mqc_t *mqc) {
|
||||||
|
mqc_resetstates(mqc);
|
||||||
|
mqc_setstate(mqc, T1_CTXNO_UNI, 0, 46);
|
||||||
|
mqc_setstate(mqc, T1_CTXNO_AGG, 0, 3);
|
||||||
|
mqc_setstate(mqc, T1_CTXNO_ZC, 0, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
int mqc_restart_enc(opj_mqc_t *mqc) {
|
||||||
|
int correction = 1;
|
||||||
|
|
||||||
|
/* <flush part> */
|
||||||
|
int n = 27 - 15 - mqc->ct;
|
||||||
|
mqc->c <<= mqc->ct;
|
||||||
|
while (n > 0) {
|
||||||
|
mqc_byteout(mqc);
|
||||||
|
n -= mqc->ct;
|
||||||
|
mqc->c <<= mqc->ct;
|
||||||
|
}
|
||||||
|
mqc_byteout(mqc);
|
||||||
|
|
||||||
|
return correction;
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_restart_init_enc(opj_mqc_t *mqc) {
|
||||||
|
/* <Re-init part> */
|
||||||
|
mqc_setcurctx(mqc, 0);
|
||||||
|
mqc->a = 0x8000;
|
||||||
|
mqc->c = 0;
|
||||||
|
mqc->ct = 12;
|
||||||
|
mqc->bp--;
|
||||||
|
if (*mqc->bp == 0xff) {
|
||||||
|
mqc->ct = 13;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_erterm_enc(opj_mqc_t *mqc) {
|
||||||
|
int k = 11 - mqc->ct + 1;
|
||||||
|
|
||||||
|
while (k > 0) {
|
||||||
|
mqc->c <<= mqc->ct;
|
||||||
|
mqc->ct = 0;
|
||||||
|
mqc_byteout(mqc);
|
||||||
|
k -= mqc->ct;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*mqc->bp != 0xff) {
|
||||||
|
mqc_byteout(mqc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_segmark_enc(opj_mqc_t *mqc) {
|
||||||
|
int i;
|
||||||
|
mqc_setcurctx(mqc, 18);
|
||||||
|
|
||||||
|
for (i = 1; i < 5; i++) {
|
||||||
|
mqc_encode(mqc, i % 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_init_dec(opj_mqc_t *mqc, unsigned char *bp, int len) {
|
||||||
|
mqc_setcurctx(mqc, 0);
|
||||||
|
mqc->start = bp;
|
||||||
|
mqc->end = bp + len;
|
||||||
|
mqc->bp = bp;
|
||||||
|
if (len==0) mqc->c = 0xff << 16;
|
||||||
|
else mqc->c = *mqc->bp << 16;
|
||||||
|
mqc_bytein(mqc);
|
||||||
|
mqc->c <<= 7;
|
||||||
|
mqc->ct -= 7;
|
||||||
|
mqc->a = 0x8000;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mqc_decode(opj_mqc_t *mqc) {
|
||||||
|
int d;
|
||||||
|
mqc->a -= (*mqc->curctx)->qeval;
|
||||||
|
if ((mqc->c >> 16) < (*mqc->curctx)->qeval) {
|
||||||
|
d = mqc_lpsexchange(mqc);
|
||||||
|
mqc_renormd(mqc);
|
||||||
|
} else {
|
||||||
|
mqc->c -= (*mqc->curctx)->qeval << 16;
|
||||||
|
if ((mqc->a & 0x8000) == 0) {
|
||||||
|
d = mqc_mpsexchange(mqc);
|
||||||
|
mqc_renormd(mqc);
|
||||||
|
} else {
|
||||||
|
d = (*mqc->curctx)->mps;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_resetstates(opj_mqc_t *mqc) {
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < MQC_NUMCTXS; i++) {
|
||||||
|
mqc->ctxs[i] = mqc_states;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void mqc_setstate(opj_mqc_t *mqc, int ctxno, int msb, int prob) {
|
||||||
|
mqc->ctxs[ctxno] = &mqc_states[msb + (prob << 1)];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
197
extern/libopenjpeg/mqc.h
vendored
Normal file
197
extern/libopenjpeg/mqc.h
vendored
Normal file
@@ -0,0 +1,197 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __MQC_H
|
||||||
|
#define __MQC_H
|
||||||
|
/**
|
||||||
|
@file mqc.h
|
||||||
|
@brief Implementation of an MQ-Coder (MQC)
|
||||||
|
|
||||||
|
The functions in MQC.C have for goal to realize the MQ-coder operations. The functions
|
||||||
|
in MQC.C are used by some function in T1.C.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup MQC MQC - Implementation of an MQ-Coder */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
This struct defines the state of a context.
|
||||||
|
*/
|
||||||
|
typedef struct opj_mqc_state {
|
||||||
|
/** the probability of the Least Probable Symbol (0.75->0x8000, 1.5->0xffff) */
|
||||||
|
unsigned int qeval;
|
||||||
|
/** the Most Probable Symbol (0 or 1) */
|
||||||
|
int mps;
|
||||||
|
/** next state if the next encoded symbol is the MPS */
|
||||||
|
struct opj_mqc_state *nmps;
|
||||||
|
/** next state if the next encoded symbol is the LPS */
|
||||||
|
struct opj_mqc_state *nlps;
|
||||||
|
} opj_mqc_state_t;
|
||||||
|
|
||||||
|
#define MQC_NUMCTXS 19
|
||||||
|
|
||||||
|
/**
|
||||||
|
MQ coder
|
||||||
|
*/
|
||||||
|
typedef struct opj_mqc {
|
||||||
|
unsigned int c;
|
||||||
|
unsigned int a;
|
||||||
|
unsigned int ct;
|
||||||
|
unsigned char *bp;
|
||||||
|
unsigned char *start;
|
||||||
|
unsigned char *end;
|
||||||
|
opj_mqc_state_t *ctxs[MQC_NUMCTXS];
|
||||||
|
opj_mqc_state_t **curctx;
|
||||||
|
} opj_mqc_t;
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Create a new MQC handle
|
||||||
|
@return Returns a new MQC handle if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_mqc_t* mqc_create(void);
|
||||||
|
/**
|
||||||
|
Destroy a previously created MQC handle
|
||||||
|
@param mqc MQC handle to destroy
|
||||||
|
*/
|
||||||
|
void mqc_destroy(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Return the number of bytes written/read since initialisation
|
||||||
|
@param mqc MQC handle
|
||||||
|
@return Returns the number of bytes already encoded
|
||||||
|
*/
|
||||||
|
int mqc_numbytes(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Reset the states of all the context of the coder/decoder
|
||||||
|
(each context is set to a state where 0 and 1 are more or less equiprobable)
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
void mqc_resetstates(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Set the state of a particular context
|
||||||
|
@param mqc MQC handle
|
||||||
|
@param ctxno Number that identifies the context
|
||||||
|
@param msb The MSB of the new state of the context
|
||||||
|
@param prob Number that identifies the probability of the symbols for the new state of the context
|
||||||
|
*/
|
||||||
|
void mqc_setstate(opj_mqc_t *mqc, int ctxno, int msb, int prob);
|
||||||
|
/**
|
||||||
|
Initialize the encoder
|
||||||
|
@param mqc MQC handle
|
||||||
|
@param bp Pointer to the start of the buffer where the bytes will be written
|
||||||
|
*/
|
||||||
|
void mqc_init_enc(opj_mqc_t *mqc, unsigned char *bp);
|
||||||
|
/**
|
||||||
|
Set the current context used for coding/decoding
|
||||||
|
@param mqc MQC handle
|
||||||
|
@param ctxno Number that identifies the context
|
||||||
|
*/
|
||||||
|
#define mqc_setcurctx(mqc, ctxno) (mqc)->curctx = &(mqc)->ctxs[(int)(ctxno)]
|
||||||
|
/**
|
||||||
|
Encode a symbol using the MQ-coder
|
||||||
|
@param mqc MQC handle
|
||||||
|
@param d The symbol to be encoded (0 or 1)
|
||||||
|
*/
|
||||||
|
void mqc_encode(opj_mqc_t *mqc, int d);
|
||||||
|
/**
|
||||||
|
Flush the encoder, so that all remaining data is written
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
void mqc_flush(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
BYPASS mode switch, initialization operation.
|
||||||
|
JPEG 2000 p 505.
|
||||||
|
<h2>Not fully implemented and tested !!</h2>
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
void mqc_bypass_init_enc(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
BYPASS mode switch, coding operation.
|
||||||
|
JPEG 2000 p 505.
|
||||||
|
<h2>Not fully implemented and tested !!</h2>
|
||||||
|
@param mqc MQC handle
|
||||||
|
@param d The symbol to be encoded (0 or 1)
|
||||||
|
*/
|
||||||
|
void mqc_bypass_enc(opj_mqc_t *mqc, int d);
|
||||||
|
/**
|
||||||
|
BYPASS mode switch, flush operation
|
||||||
|
<h2>Not fully implemented and tested !!</h2>
|
||||||
|
@param mqc MQC handle
|
||||||
|
@return Returns 1 (always)
|
||||||
|
*/
|
||||||
|
int mqc_bypass_flush_enc(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
RESET mode switch
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
void mqc_reset_enc(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
RESTART mode switch (TERMALL)
|
||||||
|
@param mqc MQC handle
|
||||||
|
@return Returns 1 (always)
|
||||||
|
*/
|
||||||
|
int mqc_restart_enc(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
RESTART mode switch (TERMALL) reinitialisation
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
void mqc_restart_init_enc(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
ERTERM mode switch (PTERM)
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
void mqc_erterm_enc(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
SEGMARK mode switch (SEGSYM)
|
||||||
|
@param mqc MQC handle
|
||||||
|
*/
|
||||||
|
void mqc_segmark_enc(opj_mqc_t *mqc);
|
||||||
|
/**
|
||||||
|
Initialize the decoder
|
||||||
|
@param mqc MQC handle
|
||||||
|
@param bp Pointer to the start of the buffer from which the bytes will be read
|
||||||
|
@param len Length of the input buffer
|
||||||
|
*/
|
||||||
|
void mqc_init_dec(opj_mqc_t *mqc, unsigned char *bp, int len);
|
||||||
|
/**
|
||||||
|
Decode a symbol
|
||||||
|
@param mqc MQC handle
|
||||||
|
@return Returns the decoded symbol (0 or 1)
|
||||||
|
*/
|
||||||
|
int mqc_decode(opj_mqc_t *mqc);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __MQC_H */
|
||||||
329
extern/libopenjpeg/openjpeg.c
vendored
Normal file
329
extern/libopenjpeg/openjpeg.c
vendored
Normal file
@@ -0,0 +1,329 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2005, Herv<72> Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef WIN32
|
||||||
|
#include <windows.h>
|
||||||
|
#endif /* WIN32 */
|
||||||
|
|
||||||
|
#include "opj_includes.h"
|
||||||
|
|
||||||
|
/* ---------------------------------------------------------------------- */
|
||||||
|
#ifdef WIN32
|
||||||
|
#ifndef OPJ_STATIC
|
||||||
|
BOOL APIENTRY
|
||||||
|
DllMain(HANDLE hModule, DWORD ul_reason_for_call, LPVOID lpReserved) {
|
||||||
|
switch (ul_reason_for_call) {
|
||||||
|
case DLL_PROCESS_ATTACH :
|
||||||
|
break;
|
||||||
|
case DLL_PROCESS_DETACH :
|
||||||
|
break;
|
||||||
|
case DLL_THREAD_ATTACH :
|
||||||
|
case DLL_THREAD_DETACH :
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
#endif /* OPJ_STATIC */
|
||||||
|
#endif /* WIN32 */
|
||||||
|
|
||||||
|
/* ---------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
|
||||||
|
const char* OPJ_CALLCONV opj_version(void) {
|
||||||
|
return OPENJPEG_VERSION;
|
||||||
|
}
|
||||||
|
|
||||||
|
opj_dinfo_t* OPJ_CALLCONV opj_create_decompress(OPJ_CODEC_FORMAT format) {
|
||||||
|
opj_dinfo_t *dinfo = (opj_dinfo_t*)opj_malloc(sizeof(opj_dinfo_t));
|
||||||
|
if(!dinfo) return NULL;
|
||||||
|
dinfo->is_decompressor = true;
|
||||||
|
switch(format) {
|
||||||
|
case CODEC_J2K:
|
||||||
|
case CODEC_JPT:
|
||||||
|
/* get a J2K decoder handle */
|
||||||
|
dinfo->j2k_handle = (void*)j2k_create_decompress((opj_common_ptr)dinfo);
|
||||||
|
if(!dinfo->j2k_handle) {
|
||||||
|
opj_free(dinfo);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case CODEC_JP2:
|
||||||
|
/* get a JP2 decoder handle */
|
||||||
|
dinfo->jp2_handle = (void*)jp2_create_decompress((opj_common_ptr)dinfo);
|
||||||
|
if(!dinfo->jp2_handle) {
|
||||||
|
opj_free(dinfo);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case CODEC_UNKNOWN:
|
||||||
|
default:
|
||||||
|
opj_free(dinfo);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
dinfo->codec_format = format;
|
||||||
|
|
||||||
|
return dinfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OPJ_CALLCONV opj_destroy_decompress(opj_dinfo_t *dinfo) {
|
||||||
|
if(dinfo) {
|
||||||
|
/* destroy the codec */
|
||||||
|
switch(dinfo->codec_format) {
|
||||||
|
case CODEC_J2K:
|
||||||
|
case CODEC_JPT:
|
||||||
|
j2k_destroy_decompress((opj_j2k_t*)dinfo->j2k_handle);
|
||||||
|
break;
|
||||||
|
case CODEC_JP2:
|
||||||
|
jp2_destroy_decompress((opj_jp2_t*)dinfo->jp2_handle);
|
||||||
|
break;
|
||||||
|
case CODEC_UNKNOWN:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* destroy the decompressor */
|
||||||
|
opj_free(dinfo);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void OPJ_CALLCONV opj_set_default_decoder_parameters(opj_dparameters_t *parameters) {
|
||||||
|
if(parameters) {
|
||||||
|
memset(parameters, 0, sizeof(opj_dparameters_t));
|
||||||
|
/* default decoding parameters */
|
||||||
|
parameters->cp_layer = 0;
|
||||||
|
parameters->cp_reduce = 0;
|
||||||
|
parameters->cp_limit_decoding = NO_LIMITATION;
|
||||||
|
|
||||||
|
parameters->decod_format = -1;
|
||||||
|
parameters->cod_format = -1;
|
||||||
|
/* UniPG>> */
|
||||||
|
#ifdef USE_JPWL
|
||||||
|
parameters->jpwl_correct = false;
|
||||||
|
parameters->jpwl_exp_comps = JPWL_EXPECTED_COMPONENTS;
|
||||||
|
parameters->jpwl_max_tiles = JPWL_MAXIMUM_TILES;
|
||||||
|
#endif /* USE_JPWL */
|
||||||
|
/* <<UniPG */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void OPJ_CALLCONV opj_setup_decoder(opj_dinfo_t *dinfo, opj_dparameters_t *parameters) {
|
||||||
|
if(dinfo && parameters) {
|
||||||
|
switch(dinfo->codec_format) {
|
||||||
|
case CODEC_J2K:
|
||||||
|
case CODEC_JPT:
|
||||||
|
j2k_setup_decoder((opj_j2k_t*)dinfo->j2k_handle, parameters);
|
||||||
|
break;
|
||||||
|
case CODEC_JP2:
|
||||||
|
jp2_setup_decoder((opj_jp2_t*)dinfo->jp2_handle, parameters);
|
||||||
|
break;
|
||||||
|
case CODEC_UNKNOWN:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
opj_image_t* OPJ_CALLCONV opj_decode(opj_dinfo_t *dinfo, opj_cio_t *cio) {
|
||||||
|
return opj_decode_with_info(dinfo, cio, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
opj_image_t* OPJ_CALLCONV opj_decode_with_info(opj_dinfo_t *dinfo, opj_cio_t *cio, opj_codestream_info_t *cstr_info) {
|
||||||
|
if(dinfo && cio) {
|
||||||
|
switch(dinfo->codec_format) {
|
||||||
|
case CODEC_J2K:
|
||||||
|
return j2k_decode((opj_j2k_t*)dinfo->j2k_handle, cio, cstr_info);
|
||||||
|
case CODEC_JPT:
|
||||||
|
return j2k_decode_jpt_stream((opj_j2k_t*)dinfo->j2k_handle, cio, cstr_info);
|
||||||
|
case CODEC_JP2:
|
||||||
|
return jp2_decode((opj_jp2_t*)dinfo->jp2_handle, cio, cstr_info);
|
||||||
|
case CODEC_UNKNOWN:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
opj_cinfo_t* OPJ_CALLCONV opj_create_compress(OPJ_CODEC_FORMAT format) {
|
||||||
|
opj_cinfo_t *cinfo = (opj_cinfo_t*)opj_malloc(sizeof(opj_cinfo_t));
|
||||||
|
if(!cinfo) return NULL;
|
||||||
|
cinfo->is_decompressor = false;
|
||||||
|
switch(format) {
|
||||||
|
case CODEC_J2K:
|
||||||
|
/* get a J2K coder handle */
|
||||||
|
cinfo->j2k_handle = (void*)j2k_create_compress((opj_common_ptr)cinfo);
|
||||||
|
if(!cinfo->j2k_handle) {
|
||||||
|
opj_free(cinfo);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case CODEC_JP2:
|
||||||
|
/* get a JP2 coder handle */
|
||||||
|
cinfo->jp2_handle = (void*)jp2_create_compress((opj_common_ptr)cinfo);
|
||||||
|
if(!cinfo->jp2_handle) {
|
||||||
|
opj_free(cinfo);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case CODEC_JPT:
|
||||||
|
case CODEC_UNKNOWN:
|
||||||
|
default:
|
||||||
|
opj_free(cinfo);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
cinfo->codec_format = format;
|
||||||
|
|
||||||
|
return cinfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OPJ_CALLCONV opj_destroy_compress(opj_cinfo_t *cinfo) {
|
||||||
|
if(cinfo) {
|
||||||
|
/* destroy the codec */
|
||||||
|
switch(cinfo->codec_format) {
|
||||||
|
case CODEC_J2K:
|
||||||
|
j2k_destroy_compress((opj_j2k_t*)cinfo->j2k_handle);
|
||||||
|
break;
|
||||||
|
case CODEC_JP2:
|
||||||
|
jp2_destroy_compress((opj_jp2_t*)cinfo->jp2_handle);
|
||||||
|
break;
|
||||||
|
case CODEC_JPT:
|
||||||
|
case CODEC_UNKNOWN:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* destroy the decompressor */
|
||||||
|
opj_free(cinfo);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void OPJ_CALLCONV opj_set_default_encoder_parameters(opj_cparameters_t *parameters) {
|
||||||
|
if(parameters) {
|
||||||
|
memset(parameters, 0, sizeof(opj_cparameters_t));
|
||||||
|
/* default coding parameters */
|
||||||
|
parameters->cp_cinema = OFF;
|
||||||
|
parameters->max_comp_size = 0;
|
||||||
|
parameters->numresolution = 6;
|
||||||
|
parameters->cp_rsiz = STD_RSIZ;
|
||||||
|
parameters->cblockw_init = 64;
|
||||||
|
parameters->cblockh_init = 64;
|
||||||
|
parameters->prog_order = LRCP;
|
||||||
|
parameters->roi_compno = -1; /* no ROI */
|
||||||
|
parameters->subsampling_dx = 1;
|
||||||
|
parameters->subsampling_dy = 1;
|
||||||
|
parameters->tp_on = 0;
|
||||||
|
parameters->decod_format = -1;
|
||||||
|
parameters->cod_format = -1;
|
||||||
|
/* UniPG>> */
|
||||||
|
#ifdef USE_JPWL
|
||||||
|
parameters->jpwl_epc_on = false;
|
||||||
|
parameters->jpwl_hprot_MH = -1; /* -1 means unassigned */
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < JPWL_MAX_NO_TILESPECS; i++) {
|
||||||
|
parameters->jpwl_hprot_TPH_tileno[i] = -1; /* unassigned */
|
||||||
|
parameters->jpwl_hprot_TPH[i] = 0; /* absent */
|
||||||
|
}
|
||||||
|
};
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < JPWL_MAX_NO_PACKSPECS; i++) {
|
||||||
|
parameters->jpwl_pprot_tileno[i] = -1; /* unassigned */
|
||||||
|
parameters->jpwl_pprot_packno[i] = -1; /* unassigned */
|
||||||
|
parameters->jpwl_pprot[i] = 0; /* absent */
|
||||||
|
}
|
||||||
|
};
|
||||||
|
parameters->jpwl_sens_size = 0; /* 0 means no ESD */
|
||||||
|
parameters->jpwl_sens_addr = 0; /* 0 means auto */
|
||||||
|
parameters->jpwl_sens_range = 0; /* 0 means packet */
|
||||||
|
parameters->jpwl_sens_MH = -1; /* -1 means unassigned */
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < JPWL_MAX_NO_TILESPECS; i++) {
|
||||||
|
parameters->jpwl_sens_TPH_tileno[i] = -1; /* unassigned */
|
||||||
|
parameters->jpwl_sens_TPH[i] = -1; /* absent */
|
||||||
|
}
|
||||||
|
};
|
||||||
|
#endif /* USE_JPWL */
|
||||||
|
/* <<UniPG */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void OPJ_CALLCONV opj_setup_encoder(opj_cinfo_t *cinfo, opj_cparameters_t *parameters, opj_image_t *image) {
|
||||||
|
if(cinfo && parameters && image) {
|
||||||
|
switch(cinfo->codec_format) {
|
||||||
|
case CODEC_J2K:
|
||||||
|
j2k_setup_encoder((opj_j2k_t*)cinfo->j2k_handle, parameters, image);
|
||||||
|
break;
|
||||||
|
case CODEC_JP2:
|
||||||
|
jp2_setup_encoder((opj_jp2_t*)cinfo->jp2_handle, parameters, image);
|
||||||
|
break;
|
||||||
|
case CODEC_JPT:
|
||||||
|
case CODEC_UNKNOWN:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool OPJ_CALLCONV opj_encode(opj_cinfo_t *cinfo, opj_cio_t *cio, opj_image_t *image, char *index) {
|
||||||
|
if (index != NULL)
|
||||||
|
opj_event_msg((opj_common_ptr)cinfo, EVT_WARNING, "Set index to NULL when calling the opj_encode function.\n"
|
||||||
|
"To extract the index, use the opj_encode_with_info() function.\n"
|
||||||
|
"No index will be generated during this encoding\n");
|
||||||
|
return opj_encode_with_info(cinfo, cio, image, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool OPJ_CALLCONV opj_encode_with_info(opj_cinfo_t *cinfo, opj_cio_t *cio, opj_image_t *image, opj_codestream_info_t *cstr_info) {
|
||||||
|
if(cinfo && cio && image) {
|
||||||
|
switch(cinfo->codec_format) {
|
||||||
|
case CODEC_J2K:
|
||||||
|
return j2k_encode((opj_j2k_t*)cinfo->j2k_handle, cio, image, cstr_info);
|
||||||
|
case CODEC_JP2:
|
||||||
|
return jp2_encode((opj_jp2_t*)cinfo->jp2_handle, cio, image, cstr_info);
|
||||||
|
case CODEC_JPT:
|
||||||
|
case CODEC_UNKNOWN:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OPJ_CALLCONV opj_destroy_cstr_info(opj_codestream_info_t *cstr_info) {
|
||||||
|
if (cstr_info) {
|
||||||
|
int tileno;
|
||||||
|
for (tileno = 0; tileno < cstr_info->tw * cstr_info->th; tileno++) {
|
||||||
|
opj_tile_info_t *tile_info = &cstr_info->tile[tileno];
|
||||||
|
opj_free(tile_info->thresh);
|
||||||
|
opj_free(tile_info->packet);
|
||||||
|
opj_free(tile_info->tp);
|
||||||
|
}
|
||||||
|
opj_free(cstr_info->tile);
|
||||||
|
opj_free(cstr_info->marker);
|
||||||
|
}
|
||||||
|
}
|
||||||
911
extern/libopenjpeg/openjpeg.h
vendored
Normal file
911
extern/libopenjpeg/openjpeg.h
vendored
Normal file
@@ -0,0 +1,911 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* Copyright (c) 2006-2007, Parvatha Elangovan
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef OPENJPEG_H
|
||||||
|
#define OPENJPEG_H
|
||||||
|
|
||||||
|
#define OPENJPEG_VERSION "1.3.0"
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
Compiler directives
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if defined(OPJ_STATIC) || !(defined(WIN32) || defined(__WIN32__))
|
||||||
|
#define OPJ_API
|
||||||
|
#define OPJ_CALLCONV
|
||||||
|
#else
|
||||||
|
#define OPJ_CALLCONV __stdcall
|
||||||
|
/*
|
||||||
|
The following ifdef block is the standard way of creating macros which make exporting
|
||||||
|
from a DLL simpler. All files within this DLL are compiled with the OPJ_EXPORTS
|
||||||
|
symbol defined on the command line. this symbol should not be defined on any project
|
||||||
|
that uses this DLL. This way any other project whose source files include this file see
|
||||||
|
OPJ_API functions as being imported from a DLL, wheras this DLL sees symbols
|
||||||
|
defined with this macro as being exported.
|
||||||
|
*/
|
||||||
|
#ifdef OPJ_EXPORTS
|
||||||
|
#define OPJ_API __declspec(dllexport)
|
||||||
|
#else
|
||||||
|
#define OPJ_API __declspec(dllimport)
|
||||||
|
#endif /* OPJ_EXPORTS */
|
||||||
|
#endif /* !OPJ_STATIC || !WIN32 */
|
||||||
|
|
||||||
|
#ifndef __cplusplus
|
||||||
|
#if defined(HAVE_STDBOOL_H)
|
||||||
|
/*
|
||||||
|
The C language implementation does correctly provide the standard header
|
||||||
|
file "stdbool.h".
|
||||||
|
*/
|
||||||
|
#include <stdbool.h>
|
||||||
|
#else
|
||||||
|
/*
|
||||||
|
The C language implementation does not provide the standard header file
|
||||||
|
"stdbool.h" as required by ISO/IEC 9899:1999. Try to compensate for this
|
||||||
|
braindamage below.
|
||||||
|
*/
|
||||||
|
#if !defined(bool)
|
||||||
|
#define bool int
|
||||||
|
#endif
|
||||||
|
#if !defined(true)
|
||||||
|
#define true 1
|
||||||
|
#endif
|
||||||
|
#if !defined(false)
|
||||||
|
#define false 0
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
#endif /* __cplusplus */
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
Useful constant definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define OPJ_PATH_LEN 4096 /**< Maximum allowed size for filenames */
|
||||||
|
|
||||||
|
#define J2K_MAXRLVLS 33 /**< Number of maximum resolution level authorized */
|
||||||
|
#define J2K_MAXBANDS (3*J2K_MAXRLVLS-2) /**< Number of maximum sub-band linked to number of resolution level */
|
||||||
|
|
||||||
|
/* UniPG>> */
|
||||||
|
#define JPWL_MAX_NO_TILESPECS 16 /**< Maximum number of tile parts expected by JPWL: increase at your will */
|
||||||
|
#define JPWL_MAX_NO_PACKSPECS 16 /**< Maximum number of packet parts expected by JPWL: increase at your will */
|
||||||
|
#define JPWL_MAX_NO_MARKERS 512 /**< Maximum number of JPWL markers: increase at your will */
|
||||||
|
#define JPWL_PRIVATEINDEX_NAME "jpwl_index_privatefilename" /**< index file name used when JPWL is on */
|
||||||
|
#define JPWL_EXPECTED_COMPONENTS 3 /**< Expect this number of components, so you'll find better the first EPB */
|
||||||
|
#define JPWL_MAXIMUM_TILES 8192 /**< Expect this maximum number of tiles, to avoid some crashes */
|
||||||
|
#define JPWL_MAXIMUM_HAMMING 2 /**< Expect this maximum number of bit errors in marker id's */
|
||||||
|
#define JPWL_MAXIMUM_EPB_ROOM 65450 /**< Expect this maximum number of bytes for composition of EPBs */
|
||||||
|
/* <<UniPG */
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
enum definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
Rsiz Capabilities
|
||||||
|
*/
|
||||||
|
typedef enum RSIZ_CAPABILITIES {
|
||||||
|
STD_RSIZ = 0, /** Standard JPEG2000 profile*/
|
||||||
|
CINEMA2K = 3, /** Profile name for a 2K image*/
|
||||||
|
CINEMA4K = 4 /** Profile name for a 4K image*/
|
||||||
|
} OPJ_RSIZ_CAPABILITIES;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Digital cinema operation mode
|
||||||
|
*/
|
||||||
|
typedef enum CINEMA_MODE {
|
||||||
|
OFF = 0, /** Not Digital Cinema*/
|
||||||
|
CINEMA2K_24 = 1, /** 2K Digital Cinema at 24 fps*/
|
||||||
|
CINEMA2K_48 = 2, /** 2K Digital Cinema at 48 fps*/
|
||||||
|
CINEMA4K_24 = 3 /** 4K Digital Cinema at 24 fps*/
|
||||||
|
}OPJ_CINEMA_MODE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Progression order
|
||||||
|
*/
|
||||||
|
typedef enum PROG_ORDER {
|
||||||
|
PROG_UNKNOWN = -1, /**< place-holder */
|
||||||
|
LRCP = 0, /**< layer-resolution-component-precinct order */
|
||||||
|
RLCP = 1, /**< resolution-layer-component-precinct order */
|
||||||
|
RPCL = 2, /**< resolution-precinct-component-layer order */
|
||||||
|
PCRL = 3, /**< precinct-component-resolution-layer order */
|
||||||
|
CPRL = 4 /**< component-precinct-resolution-layer order */
|
||||||
|
} OPJ_PROG_ORDER;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Supported image color spaces
|
||||||
|
*/
|
||||||
|
typedef enum COLOR_SPACE {
|
||||||
|
CLRSPC_UNKNOWN = -1, /**< place-holder */
|
||||||
|
CLRSPC_SRGB = 1, /**< sRGB */
|
||||||
|
CLRSPC_GRAY = 2, /**< grayscale */
|
||||||
|
CLRSPC_SYCC = 3 /**< YUV */
|
||||||
|
} OPJ_COLOR_SPACE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Supported codec
|
||||||
|
*/
|
||||||
|
typedef enum CODEC_FORMAT {
|
||||||
|
CODEC_UNKNOWN = -1, /**< place-holder */
|
||||||
|
CODEC_J2K = 0, /**< JPEG-2000 codestream : read/write */
|
||||||
|
CODEC_JPT = 1, /**< JPT-stream (JPEG 2000, JPIP) : read only */
|
||||||
|
CODEC_JP2 = 2 /**< JPEG-2000 file format : read/write */
|
||||||
|
} OPJ_CODEC_FORMAT;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Limit decoding to certain portions of the codestream.
|
||||||
|
*/
|
||||||
|
typedef enum LIMIT_DECODING {
|
||||||
|
NO_LIMITATION = 0, /**< No limitation for the decoding. The entire codestream will de decoded */
|
||||||
|
LIMIT_TO_MAIN_HEADER = 1, /**< The decoding is limited to the Main Header */
|
||||||
|
DECODE_ALL_BUT_PACKETS = 2 /**< Decode everything except the JPEG 2000 packets */
|
||||||
|
} OPJ_LIMIT_DECODING;
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
event manager typedef definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Callback function prototype for events
|
||||||
|
@param msg Event message
|
||||||
|
@param client_data
|
||||||
|
*/
|
||||||
|
typedef void (*opj_msg_callback) (const char *msg, void *client_data);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Message handler object
|
||||||
|
used for
|
||||||
|
<ul>
|
||||||
|
<li>Error messages
|
||||||
|
<li>Warning messages
|
||||||
|
<li>Debugging messages
|
||||||
|
</ul>
|
||||||
|
*/
|
||||||
|
typedef struct opj_event_mgr {
|
||||||
|
/** Error message callback if available, NULL otherwise */
|
||||||
|
opj_msg_callback error_handler;
|
||||||
|
/** Warning message callback if available, NULL otherwise */
|
||||||
|
opj_msg_callback warning_handler;
|
||||||
|
/** Debug message callback if available, NULL otherwise */
|
||||||
|
opj_msg_callback info_handler;
|
||||||
|
} opj_event_mgr_t;
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
codec typedef definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Progression order changes
|
||||||
|
*/
|
||||||
|
typedef struct opj_poc {
|
||||||
|
/** Resolution num start, Component num start, given by POC */
|
||||||
|
int resno0, compno0;
|
||||||
|
/** Layer num end,Resolution num end, Component num end, given by POC */
|
||||||
|
int layno1, resno1, compno1;
|
||||||
|
/** Layer num start,Precinct num start, Precinct num end */
|
||||||
|
int layno0, precno0, precno1;
|
||||||
|
/** Progression order enum*/
|
||||||
|
OPJ_PROG_ORDER prg1,prg;
|
||||||
|
/** Progression order string*/
|
||||||
|
char progorder[5];
|
||||||
|
/** Tile number */
|
||||||
|
int tile;
|
||||||
|
/** Start and end values for Tile width and height*/
|
||||||
|
int tx0,tx1,ty0,ty1;
|
||||||
|
/** Start value, initialised in pi_initialise_encode*/
|
||||||
|
int layS, resS, compS, prcS;
|
||||||
|
/** End value, initialised in pi_initialise_encode */
|
||||||
|
int layE, resE, compE, prcE;
|
||||||
|
/** Start and end values of Tile width and height, initialised in pi_initialise_encode*/
|
||||||
|
int txS,txE,tyS,tyE,dx,dy;
|
||||||
|
/** Temporary values for Tile parts, initialised in pi_create_encode */
|
||||||
|
int lay_t, res_t, comp_t, prc_t,tx0_t,ty0_t;
|
||||||
|
} opj_poc_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Compression parameters
|
||||||
|
*/
|
||||||
|
typedef struct opj_cparameters {
|
||||||
|
/** size of tile: tile_size_on = false (not in argument) or = true (in argument) */
|
||||||
|
bool tile_size_on;
|
||||||
|
/** XTOsiz */
|
||||||
|
int cp_tx0;
|
||||||
|
/** YTOsiz */
|
||||||
|
int cp_ty0;
|
||||||
|
/** XTsiz */
|
||||||
|
int cp_tdx;
|
||||||
|
/** YTsiz */
|
||||||
|
int cp_tdy;
|
||||||
|
/** allocation by rate/distortion */
|
||||||
|
int cp_disto_alloc;
|
||||||
|
/** allocation by fixed layer */
|
||||||
|
int cp_fixed_alloc;
|
||||||
|
/** add fixed_quality */
|
||||||
|
int cp_fixed_quality;
|
||||||
|
/** fixed layer */
|
||||||
|
int *cp_matrice;
|
||||||
|
/** comment for coding */
|
||||||
|
char *cp_comment;
|
||||||
|
/** csty : coding style */
|
||||||
|
int csty;
|
||||||
|
/** progression order (default LRCP) */
|
||||||
|
OPJ_PROG_ORDER prog_order;
|
||||||
|
/** progression order changes */
|
||||||
|
opj_poc_t POC[32];
|
||||||
|
/** number of progression order changes (POC), default to 0 */
|
||||||
|
int numpocs;
|
||||||
|
/** number of layers */
|
||||||
|
int tcp_numlayers;
|
||||||
|
/** rates of layers */
|
||||||
|
float tcp_rates[100];
|
||||||
|
/** different psnr for successive layers */
|
||||||
|
float tcp_distoratio[100];
|
||||||
|
/** number of resolutions */
|
||||||
|
int numresolution;
|
||||||
|
/** initial code block width, default to 64 */
|
||||||
|
int cblockw_init;
|
||||||
|
/** initial code block height, default to 64 */
|
||||||
|
int cblockh_init;
|
||||||
|
/** mode switch (cblk_style) */
|
||||||
|
int mode;
|
||||||
|
/** 1 : use the irreversible DWT 9-7, 0 : use lossless compression (default) */
|
||||||
|
int irreversible;
|
||||||
|
/** region of interest: affected component in [0..3], -1 means no ROI */
|
||||||
|
int roi_compno;
|
||||||
|
/** region of interest: upshift value */
|
||||||
|
int roi_shift;
|
||||||
|
/* number of precinct size specifications */
|
||||||
|
int res_spec;
|
||||||
|
/** initial precinct width */
|
||||||
|
int prcw_init[J2K_MAXRLVLS];
|
||||||
|
/** initial precinct height */
|
||||||
|
int prch_init[J2K_MAXRLVLS];
|
||||||
|
|
||||||
|
/**@name command line encoder parameters (not used inside the library) */
|
||||||
|
/*@{*/
|
||||||
|
/** input file name */
|
||||||
|
char infile[OPJ_PATH_LEN];
|
||||||
|
/** output file name */
|
||||||
|
char outfile[OPJ_PATH_LEN];
|
||||||
|
/** DEPRECATED. Index generation is now handeld with the opj_encode_with_info() function. Set to NULL */
|
||||||
|
int index_on;
|
||||||
|
/** DEPRECATED. Index generation is now handeld with the opj_encode_with_info() function. Set to NULL */
|
||||||
|
char index[OPJ_PATH_LEN];
|
||||||
|
/** subimage encoding: origin image offset in x direction */
|
||||||
|
int image_offset_x0;
|
||||||
|
/** subimage encoding: origin image offset in y direction */
|
||||||
|
int image_offset_y0;
|
||||||
|
/** subsampling value for dx */
|
||||||
|
int subsampling_dx;
|
||||||
|
/** subsampling value for dy */
|
||||||
|
int subsampling_dy;
|
||||||
|
/** input file format 0: PGX, 1: PxM, 2: BMP 3:TIF*/
|
||||||
|
int decod_format;
|
||||||
|
/** output file format 0: J2K, 1: JP2, 2: JPT */
|
||||||
|
int cod_format;
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/* UniPG>> */
|
||||||
|
/**@name JPWL encoding parameters */
|
||||||
|
/*@{*/
|
||||||
|
/** enables writing of EPC in MH, thus activating JPWL */
|
||||||
|
bool jpwl_epc_on;
|
||||||
|
/** error protection method for MH (0,1,16,32,37-128) */
|
||||||
|
int jpwl_hprot_MH;
|
||||||
|
/** tile number of header protection specification (>=0) */
|
||||||
|
int jpwl_hprot_TPH_tileno[JPWL_MAX_NO_TILESPECS];
|
||||||
|
/** error protection methods for TPHs (0,1,16,32,37-128) */
|
||||||
|
int jpwl_hprot_TPH[JPWL_MAX_NO_TILESPECS];
|
||||||
|
/** tile number of packet protection specification (>=0) */
|
||||||
|
int jpwl_pprot_tileno[JPWL_MAX_NO_PACKSPECS];
|
||||||
|
/** packet number of packet protection specification (>=0) */
|
||||||
|
int jpwl_pprot_packno[JPWL_MAX_NO_PACKSPECS];
|
||||||
|
/** error protection methods for packets (0,1,16,32,37-128) */
|
||||||
|
int jpwl_pprot[JPWL_MAX_NO_PACKSPECS];
|
||||||
|
/** enables writing of ESD, (0=no/1/2 bytes) */
|
||||||
|
int jpwl_sens_size;
|
||||||
|
/** sensitivity addressing size (0=auto/2/4 bytes) */
|
||||||
|
int jpwl_sens_addr;
|
||||||
|
/** sensitivity range (0-3) */
|
||||||
|
int jpwl_sens_range;
|
||||||
|
/** sensitivity method for MH (-1=no,0-7) */
|
||||||
|
int jpwl_sens_MH;
|
||||||
|
/** tile number of sensitivity specification (>=0) */
|
||||||
|
int jpwl_sens_TPH_tileno[JPWL_MAX_NO_TILESPECS];
|
||||||
|
/** sensitivity methods for TPHs (-1=no,0-7) */
|
||||||
|
int jpwl_sens_TPH[JPWL_MAX_NO_TILESPECS];
|
||||||
|
/*@}*/
|
||||||
|
/* <<UniPG */
|
||||||
|
|
||||||
|
/** Digital Cinema compliance 0-not compliant, 1-compliant*/
|
||||||
|
OPJ_CINEMA_MODE cp_cinema;
|
||||||
|
/** Maximum rate for each component. If == 0, component size limitation is not considered */
|
||||||
|
int max_comp_size;
|
||||||
|
/** Profile name*/
|
||||||
|
OPJ_RSIZ_CAPABILITIES cp_rsiz;
|
||||||
|
/** Tile part generation*/
|
||||||
|
char tp_on;
|
||||||
|
/** Flag for Tile part generation*/
|
||||||
|
char tp_flag;
|
||||||
|
/** MCT (multiple component transform) */
|
||||||
|
char tcp_mct;
|
||||||
|
} opj_cparameters_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Decompression parameters
|
||||||
|
*/
|
||||||
|
typedef struct opj_dparameters {
|
||||||
|
/**
|
||||||
|
Set the number of highest resolution levels to be discarded.
|
||||||
|
The image resolution is effectively divided by 2 to the power of the number of discarded levels.
|
||||||
|
The reduce factor is limited by the smallest total number of decomposition levels among tiles.
|
||||||
|
if != 0, then original dimension divided by 2^(reduce);
|
||||||
|
if == 0 or not used, image is decoded to the full resolution
|
||||||
|
*/
|
||||||
|
int cp_reduce;
|
||||||
|
/**
|
||||||
|
Set the maximum number of quality layers to decode.
|
||||||
|
If there are less quality layers than the specified number, all the quality layers are decoded.
|
||||||
|
if != 0, then only the first "layer" layers are decoded;
|
||||||
|
if == 0 or not used, all the quality layers are decoded
|
||||||
|
*/
|
||||||
|
int cp_layer;
|
||||||
|
|
||||||
|
/**@name command line encoder parameters (not used inside the library) */
|
||||||
|
/*@{*/
|
||||||
|
/** input file name */
|
||||||
|
char infile[OPJ_PATH_LEN];
|
||||||
|
/** output file name */
|
||||||
|
char outfile[OPJ_PATH_LEN];
|
||||||
|
/** input file format 0: J2K, 1: JP2, 2: JPT */
|
||||||
|
int decod_format;
|
||||||
|
/** output file format 0: PGX, 1: PxM, 2: BMP */
|
||||||
|
int cod_format;
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/* UniPG>> */
|
||||||
|
/**@name JPWL decoding parameters */
|
||||||
|
/*@{*/
|
||||||
|
/** activates the JPWL correction capabilities */
|
||||||
|
bool jpwl_correct;
|
||||||
|
/** expected number of components */
|
||||||
|
int jpwl_exp_comps;
|
||||||
|
/** maximum number of tiles */
|
||||||
|
int jpwl_max_tiles;
|
||||||
|
/*@}*/
|
||||||
|
/* <<UniPG */
|
||||||
|
|
||||||
|
/**
|
||||||
|
Specify whether the decoding should be done on the entire codestream, or be limited to the main header
|
||||||
|
Limiting the decoding to the main header makes it possible to extract the characteristics of the codestream
|
||||||
|
if == NO_LIMITATION, the entire codestream is decoded;
|
||||||
|
if == LIMIT_TO_MAIN_HEADER, only the main header is decoded;
|
||||||
|
*/
|
||||||
|
OPJ_LIMIT_DECODING cp_limit_decoding;
|
||||||
|
|
||||||
|
} opj_dparameters_t;
|
||||||
|
|
||||||
|
/** Common fields between JPEG-2000 compression and decompression master structs. */
|
||||||
|
|
||||||
|
#define opj_common_fields \
|
||||||
|
opj_event_mgr_t *event_mgr; /**< pointer to the event manager */\
|
||||||
|
void * client_data; /**< Available for use by application */\
|
||||||
|
bool is_decompressor; /**< So common code can tell which is which */\
|
||||||
|
OPJ_CODEC_FORMAT codec_format; /**< selected codec */\
|
||||||
|
void *j2k_handle; /**< pointer to the J2K codec */\
|
||||||
|
void *jp2_handle; /**< pointer to the JP2 codec */\
|
||||||
|
void *mj2_handle /**< pointer to the MJ2 codec */
|
||||||
|
|
||||||
|
/* Routines that are to be used by both halves of the library are declared
|
||||||
|
* to receive a pointer to this structure. There are no actual instances of
|
||||||
|
* opj_common_struct_t, only of opj_cinfo_t and opj_dinfo_t.
|
||||||
|
*/
|
||||||
|
typedef struct opj_common_struct {
|
||||||
|
opj_common_fields; /* Fields common to both master struct types */
|
||||||
|
/* Additional fields follow in an actual opj_cinfo_t or
|
||||||
|
* opj_dinfo_t. All three structs must agree on these
|
||||||
|
* initial fields! (This would be a lot cleaner in C++.)
|
||||||
|
*/
|
||||||
|
} opj_common_struct_t;
|
||||||
|
|
||||||
|
typedef opj_common_struct_t * opj_common_ptr;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Compression context info
|
||||||
|
*/
|
||||||
|
typedef struct opj_cinfo {
|
||||||
|
/** Fields shared with opj_dinfo_t */
|
||||||
|
opj_common_fields;
|
||||||
|
/* other specific fields go here */
|
||||||
|
} opj_cinfo_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Decompression context info
|
||||||
|
*/
|
||||||
|
typedef struct opj_dinfo {
|
||||||
|
/** Fields shared with opj_cinfo_t */
|
||||||
|
opj_common_fields;
|
||||||
|
/* other specific fields go here */
|
||||||
|
} opj_dinfo_t;
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
I/O stream typedef definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Stream open flags.
|
||||||
|
*/
|
||||||
|
/** The stream was opened for reading. */
|
||||||
|
#define OPJ_STREAM_READ 0x0001
|
||||||
|
/** The stream was opened for writing. */
|
||||||
|
#define OPJ_STREAM_WRITE 0x0002
|
||||||
|
|
||||||
|
/**
|
||||||
|
Byte input-output stream (CIO)
|
||||||
|
*/
|
||||||
|
typedef struct opj_cio {
|
||||||
|
/** codec context */
|
||||||
|
opj_common_ptr cinfo;
|
||||||
|
|
||||||
|
/** open mode (read/write) either OPJ_STREAM_READ or OPJ_STREAM_WRITE */
|
||||||
|
int openmode;
|
||||||
|
/** pointer to the start of the buffer */
|
||||||
|
unsigned char *buffer;
|
||||||
|
/** buffer size in bytes */
|
||||||
|
int length;
|
||||||
|
|
||||||
|
/** pointer to the start of the stream */
|
||||||
|
unsigned char *start;
|
||||||
|
/** pointer to the end of the stream */
|
||||||
|
unsigned char *end;
|
||||||
|
/** pointer to the current position */
|
||||||
|
unsigned char *bp;
|
||||||
|
} opj_cio_t;
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
image typedef definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Defines a single image component
|
||||||
|
*/
|
||||||
|
typedef struct opj_image_comp {
|
||||||
|
/** XRsiz: horizontal separation of a sample of ith component with respect to the reference grid */
|
||||||
|
int dx;
|
||||||
|
/** YRsiz: vertical separation of a sample of ith component with respect to the reference grid */
|
||||||
|
int dy;
|
||||||
|
/** data width */
|
||||||
|
int w;
|
||||||
|
/** data height */
|
||||||
|
int h;
|
||||||
|
/** x component offset compared to the whole image */
|
||||||
|
int x0;
|
||||||
|
/** y component offset compared to the whole image */
|
||||||
|
int y0;
|
||||||
|
/** precision */
|
||||||
|
int prec;
|
||||||
|
/** image depth in bits */
|
||||||
|
int bpp;
|
||||||
|
/** signed (1) / unsigned (0) */
|
||||||
|
int sgnd;
|
||||||
|
/** number of decoded resolution */
|
||||||
|
int resno_decoded;
|
||||||
|
/** number of division by 2 of the out image compared to the original size of image */
|
||||||
|
int factor;
|
||||||
|
/** image component data */
|
||||||
|
int *data;
|
||||||
|
} opj_image_comp_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Defines image data and characteristics
|
||||||
|
*/
|
||||||
|
typedef struct opj_image {
|
||||||
|
/** XOsiz: horizontal offset from the origin of the reference grid to the left side of the image area */
|
||||||
|
int x0;
|
||||||
|
/** YOsiz: vertical offset from the origin of the reference grid to the top side of the image area */
|
||||||
|
int y0;
|
||||||
|
/** Xsiz: width of the reference grid */
|
||||||
|
int x1;
|
||||||
|
/** Ysiz: height of the reference grid */
|
||||||
|
int y1;
|
||||||
|
/** number of components in the image */
|
||||||
|
int numcomps;
|
||||||
|
/** color space: sRGB, Greyscale or YUV */
|
||||||
|
OPJ_COLOR_SPACE color_space;
|
||||||
|
/** image components */
|
||||||
|
opj_image_comp_t *comps;
|
||||||
|
} opj_image_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Component parameters structure used by the opj_image_create function
|
||||||
|
*/
|
||||||
|
typedef struct opj_image_comptparm {
|
||||||
|
/** XRsiz: horizontal separation of a sample of ith component with respect to the reference grid */
|
||||||
|
int dx;
|
||||||
|
/** YRsiz: vertical separation of a sample of ith component with respect to the reference grid */
|
||||||
|
int dy;
|
||||||
|
/** data width */
|
||||||
|
int w;
|
||||||
|
/** data height */
|
||||||
|
int h;
|
||||||
|
/** x component offset compared to the whole image */
|
||||||
|
int x0;
|
||||||
|
/** y component offset compared to the whole image */
|
||||||
|
int y0;
|
||||||
|
/** precision */
|
||||||
|
int prec;
|
||||||
|
/** image depth in bits */
|
||||||
|
int bpp;
|
||||||
|
/** signed (1) / unsigned (0) */
|
||||||
|
int sgnd;
|
||||||
|
} opj_image_cmptparm_t;
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
Information on the JPEG 2000 codestream
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Index structure : Information concerning a packet inside tile
|
||||||
|
*/
|
||||||
|
typedef struct opj_packet_info {
|
||||||
|
/** packet start position (including SOP marker if it exists) */
|
||||||
|
int start_pos;
|
||||||
|
/** end of packet header position (including EPH marker if it exists)*/
|
||||||
|
int end_ph_pos;
|
||||||
|
/** packet end position */
|
||||||
|
int end_pos;
|
||||||
|
/** packet distorsion */
|
||||||
|
double disto;
|
||||||
|
} opj_packet_info_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Index structure : Information concerning tile-parts
|
||||||
|
*/
|
||||||
|
typedef struct opj_tp_info {
|
||||||
|
/** start position of tile part */
|
||||||
|
int tp_start_pos;
|
||||||
|
/** end position of tile part header */
|
||||||
|
int tp_end_header;
|
||||||
|
/** end position of tile part */
|
||||||
|
int tp_end_pos;
|
||||||
|
/** start packet of tile part */
|
||||||
|
int tp_start_pack;
|
||||||
|
/** number of packets of tile part */
|
||||||
|
int tp_numpacks;
|
||||||
|
} opj_tp_info_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Index structure : information regarding tiles
|
||||||
|
*/
|
||||||
|
typedef struct opj_tile_info {
|
||||||
|
/** value of thresh for each layer by tile cfr. Marcela */
|
||||||
|
double *thresh;
|
||||||
|
/** number of tile */
|
||||||
|
int tileno;
|
||||||
|
/** start position */
|
||||||
|
int start_pos;
|
||||||
|
/** end position of the header */
|
||||||
|
int end_header;
|
||||||
|
/** end position */
|
||||||
|
int end_pos;
|
||||||
|
/** precinct number for each resolution level (width) */
|
||||||
|
int pw[33];
|
||||||
|
/** precinct number for each resolution level (height) */
|
||||||
|
int ph[33];
|
||||||
|
/** precinct size (in power of 2), in X for each resolution level */
|
||||||
|
int pdx[33];
|
||||||
|
/** precinct size (in power of 2), in Y for each resolution level */
|
||||||
|
int pdy[33];
|
||||||
|
/** information concerning packets inside tile */
|
||||||
|
opj_packet_info_t *packet;
|
||||||
|
/** add fixed_quality */
|
||||||
|
int numpix;
|
||||||
|
/** add fixed_quality */
|
||||||
|
double distotile;
|
||||||
|
/** number of tile parts */
|
||||||
|
int num_tps;
|
||||||
|
/** information concerning tile parts */
|
||||||
|
opj_tp_info_t *tp;
|
||||||
|
} opj_tile_info_t;
|
||||||
|
|
||||||
|
/* UniPG>> */
|
||||||
|
/**
|
||||||
|
Marker structure
|
||||||
|
*/
|
||||||
|
typedef struct opj_marker_info_t {
|
||||||
|
/** marker type */
|
||||||
|
unsigned short int type;
|
||||||
|
/** position in codestream */
|
||||||
|
int pos;
|
||||||
|
/** length, marker val included */
|
||||||
|
int len;
|
||||||
|
} opj_marker_info_t;
|
||||||
|
/* <<UniPG */
|
||||||
|
|
||||||
|
/**
|
||||||
|
Index structure of the codestream
|
||||||
|
*/
|
||||||
|
typedef struct opj_codestream_info {
|
||||||
|
/** maximum distortion reduction on the whole image (add for Marcela) */
|
||||||
|
double D_max;
|
||||||
|
/** packet number */
|
||||||
|
int packno;
|
||||||
|
/** writing the packet in the index with t2_encode_packets */
|
||||||
|
int index_write;
|
||||||
|
/** image width */
|
||||||
|
int image_w;
|
||||||
|
/** image height */
|
||||||
|
int image_h;
|
||||||
|
/** progression order */
|
||||||
|
OPJ_PROG_ORDER prog;
|
||||||
|
/** tile size in x */
|
||||||
|
int tile_x;
|
||||||
|
/** tile size in y */
|
||||||
|
int tile_y;
|
||||||
|
/** */
|
||||||
|
int tile_Ox;
|
||||||
|
/** */
|
||||||
|
int tile_Oy;
|
||||||
|
/** number of tiles in X */
|
||||||
|
int tw;
|
||||||
|
/** number of tiles in Y */
|
||||||
|
int th;
|
||||||
|
/** component numbers */
|
||||||
|
int numcomps;
|
||||||
|
/** number of layer */
|
||||||
|
int numlayers;
|
||||||
|
/** number of decomposition for each component */
|
||||||
|
int *numdecompos;
|
||||||
|
/* UniPG>> */
|
||||||
|
/** number of markers */
|
||||||
|
int marknum;
|
||||||
|
/** list of markers */
|
||||||
|
opj_marker_info_t *marker;
|
||||||
|
/** actual size of markers array */
|
||||||
|
int maxmarknum;
|
||||||
|
/* <<UniPG */
|
||||||
|
/** main header position */
|
||||||
|
int main_head_start;
|
||||||
|
/** main header position */
|
||||||
|
int main_head_end;
|
||||||
|
/** codestream's size */
|
||||||
|
int codestream_size;
|
||||||
|
/** information regarding tiles inside image */
|
||||||
|
opj_tile_info_t *tile;
|
||||||
|
} opj_codestream_info_t;
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
openjpeg version
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
OPJ_API const char * OPJ_CALLCONV opj_version(void);
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
image functions definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Create an image
|
||||||
|
@param numcmpts number of components
|
||||||
|
@param cmptparms components parameters
|
||||||
|
@param clrspc image color space
|
||||||
|
@return returns a new image structure if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
OPJ_API opj_image_t* OPJ_CALLCONV opj_image_create(int numcmpts, opj_image_cmptparm_t *cmptparms, OPJ_COLOR_SPACE clrspc);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Deallocate any resources associated with an image
|
||||||
|
@param image image to be destroyed
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV opj_image_destroy(opj_image_t *image);
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
stream functions definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Open and allocate a memory stream for read / write.
|
||||||
|
On reading, the user must provide a buffer containing encoded data. The buffer will be
|
||||||
|
wrapped by the returned CIO handle.
|
||||||
|
On writing, buffer parameters must be set to 0: a buffer will be allocated by the library
|
||||||
|
to contain encoded data.
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@param buffer Reading: buffer address. Writing: NULL
|
||||||
|
@param length Reading: buffer length. Writing: 0
|
||||||
|
@return Returns a CIO handle if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
OPJ_API opj_cio_t* OPJ_CALLCONV opj_cio_open(opj_common_ptr cinfo, unsigned char *buffer, int length);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Close and free a CIO handle
|
||||||
|
@param cio CIO handle to free
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV opj_cio_close(opj_cio_t *cio);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Get position in byte stream
|
||||||
|
@param cio CIO handle
|
||||||
|
@return Returns the position in bytes
|
||||||
|
*/
|
||||||
|
OPJ_API int OPJ_CALLCONV cio_tell(opj_cio_t *cio);
|
||||||
|
/**
|
||||||
|
Set position in byte stream
|
||||||
|
@param cio CIO handle
|
||||||
|
@param pos Position, in number of bytes, from the beginning of the stream
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV cio_seek(opj_cio_t *cio, int pos);
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
event manager functions definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
OPJ_API opj_event_mgr_t* OPJ_CALLCONV opj_set_event_mgr(opj_common_ptr cinfo, opj_event_mgr_t *event_mgr, void *context);
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
codec functions definitions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
Creates a J2K/JPT/JP2 decompression structure
|
||||||
|
@param format Decoder to select
|
||||||
|
@return Returns a handle to a decompressor if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
OPJ_API opj_dinfo_t* OPJ_CALLCONV opj_create_decompress(OPJ_CODEC_FORMAT format);
|
||||||
|
/**
|
||||||
|
Destroy a decompressor handle
|
||||||
|
@param dinfo decompressor handle to destroy
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV opj_destroy_decompress(opj_dinfo_t *dinfo);
|
||||||
|
/**
|
||||||
|
Set decoding parameters to default values
|
||||||
|
@param parameters Decompression parameters
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV opj_set_default_decoder_parameters(opj_dparameters_t *parameters);
|
||||||
|
/**
|
||||||
|
Setup the decoder decoding parameters using user parameters.
|
||||||
|
Decoding parameters are returned in j2k->cp.
|
||||||
|
@param dinfo decompressor handle
|
||||||
|
@param parameters decompression parameters
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV opj_setup_decoder(opj_dinfo_t *dinfo, opj_dparameters_t *parameters);
|
||||||
|
/**
|
||||||
|
Decode an image from a JPEG-2000 codestream
|
||||||
|
@param dinfo decompressor handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@return Returns a decoded image if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
OPJ_API opj_image_t* OPJ_CALLCONV opj_decode(opj_dinfo_t *dinfo, opj_cio_t *cio);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Decode an image from a JPEG-2000 codestream and extract the codestream information
|
||||||
|
@param dinfo decompressor handle
|
||||||
|
@param cio Input buffer stream
|
||||||
|
@param cstr_info Codestream information structure if needed afterwards, NULL otherwise
|
||||||
|
@return Returns a decoded image if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
OPJ_API opj_image_t* OPJ_CALLCONV opj_decode_with_info(opj_dinfo_t *dinfo, opj_cio_t *cio, opj_codestream_info_t *cstr_info);
|
||||||
|
/**
|
||||||
|
Creates a J2K/JP2 compression structure
|
||||||
|
@param format Coder to select
|
||||||
|
@return Returns a handle to a compressor if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
OPJ_API opj_cinfo_t* OPJ_CALLCONV opj_create_compress(OPJ_CODEC_FORMAT format);
|
||||||
|
/**
|
||||||
|
Destroy a compressor handle
|
||||||
|
@param cinfo compressor handle to destroy
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV opj_destroy_compress(opj_cinfo_t *cinfo);
|
||||||
|
/**
|
||||||
|
Set encoding parameters to default values, that means :
|
||||||
|
<ul>
|
||||||
|
<li>Lossless
|
||||||
|
<li>1 tile
|
||||||
|
<li>Size of precinct : 2^15 x 2^15 (means 1 precinct)
|
||||||
|
<li>Size of code-block : 64 x 64
|
||||||
|
<li>Number of resolutions: 6
|
||||||
|
<li>No SOP marker in the codestream
|
||||||
|
<li>No EPH marker in the codestream
|
||||||
|
<li>No sub-sampling in x or y direction
|
||||||
|
<li>No mode switch activated
|
||||||
|
<li>Progression order: LRCP
|
||||||
|
<li>No index file
|
||||||
|
<li>No ROI upshifted
|
||||||
|
<li>No offset of the origin of the image
|
||||||
|
<li>No offset of the origin of the tiles
|
||||||
|
<li>Reversible DWT 5-3
|
||||||
|
</ul>
|
||||||
|
@param parameters Compression parameters
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV opj_set_default_encoder_parameters(opj_cparameters_t *parameters);
|
||||||
|
/**
|
||||||
|
Setup the encoder parameters using the current image and using user parameters.
|
||||||
|
@param cinfo Compressor handle
|
||||||
|
@param parameters Compression parameters
|
||||||
|
@param image Input filled image
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV opj_setup_encoder(opj_cinfo_t *cinfo, opj_cparameters_t *parameters, opj_image_t *image);
|
||||||
|
/**
|
||||||
|
Encode an image into a JPEG-2000 codestream
|
||||||
|
@param cinfo compressor handle
|
||||||
|
@param cio Output buffer stream
|
||||||
|
@param image Image to encode
|
||||||
|
@param index Depreacted -> Set to NULL. To extract index, used opj_encode_wci()
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
OPJ_API bool OPJ_CALLCONV opj_encode(opj_cinfo_t *cinfo, opj_cio_t *cio, opj_image_t *image, char *index);
|
||||||
|
/**
|
||||||
|
Encode an image into a JPEG-2000 codestream and extract the codestream information
|
||||||
|
@param cinfo compressor handle
|
||||||
|
@param cio Output buffer stream
|
||||||
|
@param image Image to encode
|
||||||
|
@param cstr_info Codestream information structure if needed afterwards, NULL otherwise
|
||||||
|
@return Returns true if successful, returns false otherwise
|
||||||
|
*/
|
||||||
|
OPJ_API bool OPJ_CALLCONV opj_encode_with_info(opj_cinfo_t *cinfo, opj_cio_t *cio, opj_image_t *image, opj_codestream_info_t *cstr_info);
|
||||||
|
/**
|
||||||
|
Destroy Codestream information after compression or decompression
|
||||||
|
@param cstr_info Codestream information structure
|
||||||
|
*/
|
||||||
|
OPJ_API void OPJ_CALLCONV opj_destroy_cstr_info(opj_codestream_info_t *cstr_info);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* OPENJPEG_H */
|
||||||
132
extern/libopenjpeg/opj_includes.h
vendored
Normal file
132
extern/libopenjpeg/opj_includes.h
vendored
Normal file
@@ -0,0 +1,132 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2005, Herv<72> Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef OPJ_INCLUDES_H
|
||||||
|
#define OPJ_INCLUDES_H
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
Standard includes used by the library
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
#include <memory.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <math.h>
|
||||||
|
#include <float.h>
|
||||||
|
#include <time.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
OpenJPEG interface
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
#include "openjpeg.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
OpenJPEG modules
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Ignore GCC attributes if this is not GCC */
|
||||||
|
#ifndef __GNUC__
|
||||||
|
#define __attribute__(x) /* __attribute__(x) */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
The inline keyword is supported by C99 but not by C90.
|
||||||
|
Most compilers implement their own version of this keyword ...
|
||||||
|
*/
|
||||||
|
#ifndef INLINE
|
||||||
|
#if defined(_MSC_VER)
|
||||||
|
#define INLINE __inline
|
||||||
|
#elif defined(__GNUC__)
|
||||||
|
#define INLINE __inline__
|
||||||
|
#elif defined(__MWERKS__)
|
||||||
|
#define INLINE inline
|
||||||
|
#else
|
||||||
|
/* add other compilers here ... */
|
||||||
|
#define INLINE
|
||||||
|
#endif /* defined(<Compiler>) */
|
||||||
|
#endif /* INLINE */
|
||||||
|
|
||||||
|
/* Are restricted pointers available? (C99) */
|
||||||
|
#if (__STDC_VERSION__ != 199901L)
|
||||||
|
/* Not a C99 compiler */
|
||||||
|
#ifdef __GNUC__
|
||||||
|
#define restrict __restrict__
|
||||||
|
#else
|
||||||
|
#define restrict /* restrict */
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* MSVC does not have lrintf */
|
||||||
|
#ifdef _MSC_VER
|
||||||
|
static INLINE long lrintf(float f){
|
||||||
|
int i;
|
||||||
|
|
||||||
|
_asm{
|
||||||
|
fld f
|
||||||
|
fistp i
|
||||||
|
};
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "j2k_lib.h"
|
||||||
|
#include "opj_malloc.h"
|
||||||
|
#include "event.h"
|
||||||
|
#include "cio.h"
|
||||||
|
|
||||||
|
#include "image.h"
|
||||||
|
#include "j2k.h"
|
||||||
|
#include "jp2.h"
|
||||||
|
#include "jpt.h"
|
||||||
|
|
||||||
|
#include "mqc.h"
|
||||||
|
#include "raw.h"
|
||||||
|
#include "bio.h"
|
||||||
|
#include "tgt.h"
|
||||||
|
#include "pi.h"
|
||||||
|
#include "tcd.h"
|
||||||
|
#include "t1.h"
|
||||||
|
#include "dwt.h"
|
||||||
|
#include "t2.h"
|
||||||
|
#include "mct.h"
|
||||||
|
#include "int.h"
|
||||||
|
#include "fix.h"
|
||||||
|
|
||||||
|
/* JPWL>> */
|
||||||
|
#ifdef USE_JPWL
|
||||||
|
#include "../jpwl/jpwl.h"
|
||||||
|
#endif /* USE_JPWL */
|
||||||
|
/* <<JPWL */
|
||||||
|
|
||||||
|
#endif /* OPJ_INCLUDES_H */
|
||||||
145
extern/libopenjpeg/opj_malloc.h
vendored
Normal file
145
extern/libopenjpeg/opj_malloc.h
vendored
Normal file
@@ -0,0 +1,145 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2005, Herv<72> Drolon, FreeImage Team
|
||||||
|
* Copyright (c) 2007, Callum Lerwick <seg@haxxed.com>
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __OPJ_MALLOC_H
|
||||||
|
#define __OPJ_MALLOC_H
|
||||||
|
/**
|
||||||
|
@file opj_malloc.h
|
||||||
|
@brief Internal functions
|
||||||
|
|
||||||
|
The functions in opj_malloc.h are internal utilities used for memory management.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup MISC MISC - Miscellaneous internal functions */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
Allocate an uninitialized memory block
|
||||||
|
@param size Bytes to allocate
|
||||||
|
@return Returns a void pointer to the allocated space, or NULL if there is insufficient memory available
|
||||||
|
*/
|
||||||
|
#define opj_malloc(size) malloc(size)
|
||||||
|
|
||||||
|
/**
|
||||||
|
Allocate a memory block with elements initialized to 0
|
||||||
|
@param num Blocks to allocate
|
||||||
|
@param size Bytes per block to allocate
|
||||||
|
@return Returns a void pointer to the allocated space, or NULL if there is insufficient memory available
|
||||||
|
*/
|
||||||
|
#define opj_calloc(num, size) calloc(num, size)
|
||||||
|
|
||||||
|
/**
|
||||||
|
Allocate memory aligned to a 16 byte boundry
|
||||||
|
@param size Bytes to allocate
|
||||||
|
@return Returns a void pointer to the allocated space, or NULL if there is insufficient memory available
|
||||||
|
*/
|
||||||
|
/* FIXME: These should be set with cmake tests, but we're currently not requiring use of cmake */
|
||||||
|
#ifdef WIN32
|
||||||
|
/* Someone should tell the mingw people that their malloc.h ought to provide _mm_malloc() */
|
||||||
|
#ifdef __GNUC__
|
||||||
|
#include <mm_malloc.h>
|
||||||
|
#define HAVE_MM_MALLOC
|
||||||
|
#else /* MSVC, Intel C++ */
|
||||||
|
#include <malloc.h>
|
||||||
|
#ifdef _mm_malloc
|
||||||
|
#define HAVE_MM_MALLOC
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
#else /* Not WIN32 */
|
||||||
|
#if defined(__sun)
|
||||||
|
#define HAVE_MEMALIGN
|
||||||
|
#elif defined(__GNUC__)
|
||||||
|
#define HAVE_MEMALIGN
|
||||||
|
#include <malloc.h>
|
||||||
|
/* Linux x86_64 and OSX always align allocations to 16 bytes */
|
||||||
|
#elif !defined(__amd64__) && !defined(__APPLE__)
|
||||||
|
/* FIXME: Yes, this is a big assumption */
|
||||||
|
#define HAVE_POSIX_MEMALIGN
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#define opj_aligned_malloc(size) malloc(size)
|
||||||
|
#define opj_aligned_free(m) free(m)
|
||||||
|
|
||||||
|
#ifdef HAVE_MM_MALLOC
|
||||||
|
#undef opj_aligned_malloc
|
||||||
|
#define opj_aligned_malloc(size) _mm_malloc(size, 16)
|
||||||
|
#undef opj_aligned_free
|
||||||
|
#define opj_aligned_free(m) _mm_free(m)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_MEMALIGN
|
||||||
|
extern void* memalign(size_t, size_t);
|
||||||
|
#undef opj_aligned_malloc
|
||||||
|
#define opj_aligned_malloc(size) memalign(16, (size))
|
||||||
|
#undef opj_aligned_free
|
||||||
|
#define opj_aligned_free(m) free(m)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_POSIX_MEMALIGN
|
||||||
|
#undef opj_aligned_malloc
|
||||||
|
extern int posix_memalign(void**, size_t, size_t);
|
||||||
|
|
||||||
|
static INLINE void* __attribute__ ((malloc)) opj_aligned_malloc(size_t size){
|
||||||
|
void* mem = NULL;
|
||||||
|
posix_memalign(&mem, 16, size);
|
||||||
|
return mem;
|
||||||
|
}
|
||||||
|
#undef opj_aligned_free
|
||||||
|
#define opj_aligned_free(m) free(m)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
Reallocate memory blocks.
|
||||||
|
@param memblock Pointer to previously allocated memory block
|
||||||
|
@param size New size in bytes
|
||||||
|
@return Returns a void pointer to the reallocated (and possibly moved) memory block
|
||||||
|
*/
|
||||||
|
#define opj_realloc(m, s) realloc(m, s)
|
||||||
|
|
||||||
|
/**
|
||||||
|
Deallocates or frees a memory block.
|
||||||
|
@param memblock Previously allocated memory block to be freed
|
||||||
|
*/
|
||||||
|
#define opj_free(m) free(m)
|
||||||
|
|
||||||
|
#ifdef __GNUC__
|
||||||
|
#pragma GCC poison malloc calloc realloc free
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __OPJ_MALLOC_H */
|
||||||
|
|
||||||
963
extern/libopenjpeg/pi.c
vendored
Normal file
963
extern/libopenjpeg/pi.c
vendored
Normal file
@@ -0,0 +1,963 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* Copyright (c) 2006-2007, Parvatha Elangovan
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/** @defgroup PI PI - Implementation of a packet iterator */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Local static functions */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Get next packet in layer-resolution-component-precinct order.
|
||||||
|
@param pi packet iterator to modify
|
||||||
|
@return returns false if pi pointed to the last packet or else returns true
|
||||||
|
*/
|
||||||
|
static bool pi_next_lrcp(opj_pi_iterator_t * pi);
|
||||||
|
/**
|
||||||
|
Get next packet in resolution-layer-component-precinct order.
|
||||||
|
@param pi packet iterator to modify
|
||||||
|
@return returns false if pi pointed to the last packet or else returns true
|
||||||
|
*/
|
||||||
|
static bool pi_next_rlcp(opj_pi_iterator_t * pi);
|
||||||
|
/**
|
||||||
|
Get next packet in resolution-precinct-component-layer order.
|
||||||
|
@param pi packet iterator to modify
|
||||||
|
@return returns false if pi pointed to the last packet or else returns true
|
||||||
|
*/
|
||||||
|
static bool pi_next_rpcl(opj_pi_iterator_t * pi);
|
||||||
|
/**
|
||||||
|
Get next packet in precinct-component-resolution-layer order.
|
||||||
|
@param pi packet iterator to modify
|
||||||
|
@return returns false if pi pointed to the last packet or else returns true
|
||||||
|
*/
|
||||||
|
static bool pi_next_pcrl(opj_pi_iterator_t * pi);
|
||||||
|
/**
|
||||||
|
Get next packet in component-precinct-resolution-layer order.
|
||||||
|
@param pi packet iterator to modify
|
||||||
|
@return returns false if pi pointed to the last packet or else returns true
|
||||||
|
*/
|
||||||
|
static bool pi_next_cprl(opj_pi_iterator_t * pi);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
local functions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
static bool pi_next_lrcp(opj_pi_iterator_t * pi) {
|
||||||
|
opj_pi_comp_t *comp = NULL;
|
||||||
|
opj_pi_resolution_t *res = NULL;
|
||||||
|
long index = 0;
|
||||||
|
|
||||||
|
if (!pi->first) {
|
||||||
|
comp = &pi->comps[pi->compno];
|
||||||
|
res = &comp->resolutions[pi->resno];
|
||||||
|
goto LABEL_SKIP;
|
||||||
|
} else {
|
||||||
|
pi->first = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (pi->layno = pi->poc.layno0; pi->layno < pi->poc.layno1; pi->layno++) {
|
||||||
|
for (pi->resno = pi->poc.resno0; pi->resno < pi->poc.resno1;
|
||||||
|
pi->resno++) {
|
||||||
|
for (pi->compno = pi->poc.compno0; pi->compno < pi->poc.compno1; pi->compno++) {
|
||||||
|
comp = &pi->comps[pi->compno];
|
||||||
|
if (pi->resno >= comp->numresolutions) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
res = &comp->resolutions[pi->resno];
|
||||||
|
if (!pi->tp_on){
|
||||||
|
pi->poc.precno1 = res->pw * res->ph;
|
||||||
|
}
|
||||||
|
for (pi->precno = pi->poc.precno0; pi->precno < pi->poc.precno1; pi->precno++) {
|
||||||
|
index = pi->layno * pi->step_l + pi->resno * pi->step_r + pi->compno * pi->step_c + pi->precno * pi->step_p;
|
||||||
|
if (!pi->include[index]) {
|
||||||
|
pi->include[index] = 1;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
LABEL_SKIP:;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool pi_next_rlcp(opj_pi_iterator_t * pi) {
|
||||||
|
opj_pi_comp_t *comp = NULL;
|
||||||
|
opj_pi_resolution_t *res = NULL;
|
||||||
|
long index = 0;
|
||||||
|
|
||||||
|
if (!pi->first) {
|
||||||
|
comp = &pi->comps[pi->compno];
|
||||||
|
res = &comp->resolutions[pi->resno];
|
||||||
|
goto LABEL_SKIP;
|
||||||
|
} else {
|
||||||
|
pi->first = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (pi->resno = pi->poc.resno0; pi->resno < pi->poc.resno1; pi->resno++) {
|
||||||
|
for (pi->layno = pi->poc.layno0; pi->layno < pi->poc.layno1; pi->layno++) {
|
||||||
|
for (pi->compno = pi->poc.compno0; pi->compno < pi->poc.compno1; pi->compno++) {
|
||||||
|
comp = &pi->comps[pi->compno];
|
||||||
|
if (pi->resno >= comp->numresolutions) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
res = &comp->resolutions[pi->resno];
|
||||||
|
if(!pi->tp_on){
|
||||||
|
pi->poc.precno1 = res->pw * res->ph;
|
||||||
|
}
|
||||||
|
for (pi->precno = pi->poc.precno0; pi->precno < pi->poc.precno1; pi->precno++) {
|
||||||
|
index = pi->layno * pi->step_l + pi->resno * pi->step_r + pi->compno * pi->step_c + pi->precno * pi->step_p;
|
||||||
|
if (!pi->include[index]) {
|
||||||
|
pi->include[index] = 1;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
LABEL_SKIP:;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool pi_next_rpcl(opj_pi_iterator_t * pi) {
|
||||||
|
opj_pi_comp_t *comp = NULL;
|
||||||
|
opj_pi_resolution_t *res = NULL;
|
||||||
|
long index = 0;
|
||||||
|
|
||||||
|
if (!pi->first) {
|
||||||
|
goto LABEL_SKIP;
|
||||||
|
} else {
|
||||||
|
int compno, resno;
|
||||||
|
pi->first = 0;
|
||||||
|
pi->dx = 0;
|
||||||
|
pi->dy = 0;
|
||||||
|
for (compno = 0; compno < pi->numcomps; compno++) {
|
||||||
|
comp = &pi->comps[compno];
|
||||||
|
for (resno = 0; resno < comp->numresolutions; resno++) {
|
||||||
|
int dx, dy;
|
||||||
|
res = &comp->resolutions[resno];
|
||||||
|
dx = comp->dx * (1 << (res->pdx + comp->numresolutions - 1 - resno));
|
||||||
|
dy = comp->dy * (1 << (res->pdy + comp->numresolutions - 1 - resno));
|
||||||
|
pi->dx = !pi->dx ? dx : int_min(pi->dx, dx);
|
||||||
|
pi->dy = !pi->dy ? dy : int_min(pi->dy, dy);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!pi->tp_on){
|
||||||
|
pi->poc.ty0 = pi->ty0;
|
||||||
|
pi->poc.tx0 = pi->tx0;
|
||||||
|
pi->poc.ty1 = pi->ty1;
|
||||||
|
pi->poc.tx1 = pi->tx1;
|
||||||
|
}
|
||||||
|
for (pi->resno = pi->poc.resno0; pi->resno < pi->poc.resno1; pi->resno++) {
|
||||||
|
for (pi->y = pi->poc.ty0; pi->y < pi->poc.ty1; pi->y += pi->dy - (pi->y % pi->dy)) {
|
||||||
|
for (pi->x = pi->poc.tx0; pi->x < pi->poc.tx1; pi->x += pi->dx - (pi->x % pi->dx)) {
|
||||||
|
for (pi->compno = pi->poc.compno0; pi->compno < pi->poc.compno1; pi->compno++) {
|
||||||
|
int levelno;
|
||||||
|
int trx0, try0;
|
||||||
|
int trx1, try1;
|
||||||
|
int rpx, rpy;
|
||||||
|
int prci, prcj;
|
||||||
|
comp = &pi->comps[pi->compno];
|
||||||
|
if (pi->resno >= comp->numresolutions) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
res = &comp->resolutions[pi->resno];
|
||||||
|
levelno = comp->numresolutions - 1 - pi->resno;
|
||||||
|
trx0 = int_ceildiv(pi->tx0, comp->dx << levelno);
|
||||||
|
try0 = int_ceildiv(pi->ty0, comp->dy << levelno);
|
||||||
|
trx1 = int_ceildiv(pi->tx1, comp->dx << levelno);
|
||||||
|
try1 = int_ceildiv(pi->ty1, comp->dy << levelno);
|
||||||
|
rpx = res->pdx + levelno;
|
||||||
|
rpy = res->pdy + levelno;
|
||||||
|
if (!((pi->y % (comp->dy << rpy) == 0) || ((pi->y == pi->ty0) && ((try0 << levelno) % (1 << rpx))))){
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!((pi->x % (comp->dx << rpx) == 0) || ((pi->x == pi->tx0) && ((trx0 << levelno) % (1 << rpx))))){
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((res->pw==0)||(res->pw==0)) continue;
|
||||||
|
|
||||||
|
if ((trx0==trx1)||(try0==try1)) continue;
|
||||||
|
|
||||||
|
prci = int_floordivpow2(int_ceildiv(pi->x, comp->dx << levelno), res->pdx)
|
||||||
|
- int_floordivpow2(trx0, res->pdx);
|
||||||
|
prcj = int_floordivpow2(int_ceildiv(pi->y, comp->dy << levelno), res->pdy)
|
||||||
|
- int_floordivpow2(try0, res->pdy);
|
||||||
|
pi->precno = prci + prcj * res->pw;
|
||||||
|
for (pi->layno = pi->poc.layno0; pi->layno < pi->poc.layno1; pi->layno++) {
|
||||||
|
index = pi->layno * pi->step_l + pi->resno * pi->step_r + pi->compno * pi->step_c + pi->precno * pi->step_p;
|
||||||
|
if (!pi->include[index]) {
|
||||||
|
pi->include[index] = 1;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
LABEL_SKIP:;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool pi_next_pcrl(opj_pi_iterator_t * pi) {
|
||||||
|
opj_pi_comp_t *comp = NULL;
|
||||||
|
opj_pi_resolution_t *res = NULL;
|
||||||
|
long index = 0;
|
||||||
|
|
||||||
|
if (!pi->first) {
|
||||||
|
comp = &pi->comps[pi->compno];
|
||||||
|
goto LABEL_SKIP;
|
||||||
|
} else {
|
||||||
|
int compno, resno;
|
||||||
|
pi->first = 0;
|
||||||
|
pi->dx = 0;
|
||||||
|
pi->dy = 0;
|
||||||
|
for (compno = 0; compno < pi->numcomps; compno++) {
|
||||||
|
comp = &pi->comps[compno];
|
||||||
|
for (resno = 0; resno < comp->numresolutions; resno++) {
|
||||||
|
int dx, dy;
|
||||||
|
res = &comp->resolutions[resno];
|
||||||
|
dx = comp->dx * (1 << (res->pdx + comp->numresolutions - 1 - resno));
|
||||||
|
dy = comp->dy * (1 << (res->pdy + comp->numresolutions - 1 - resno));
|
||||||
|
pi->dx = !pi->dx ? dx : int_min(pi->dx, dx);
|
||||||
|
pi->dy = !pi->dy ? dy : int_min(pi->dy, dy);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!pi->tp_on){
|
||||||
|
pi->poc.ty0 = pi->ty0;
|
||||||
|
pi->poc.tx0 = pi->tx0;
|
||||||
|
pi->poc.ty1 = pi->ty1;
|
||||||
|
pi->poc.tx1 = pi->tx1;
|
||||||
|
}
|
||||||
|
for (pi->y = pi->poc.ty0; pi->y < pi->poc.ty1; pi->y += pi->dy - (pi->y % pi->dy)) {
|
||||||
|
for (pi->x = pi->poc.tx0; pi->x < pi->poc.tx1; pi->x += pi->dx - (pi->x % pi->dx)) {
|
||||||
|
for (pi->compno = pi->poc.compno0; pi->compno < pi->poc.compno1; pi->compno++) {
|
||||||
|
comp = &pi->comps[pi->compno];
|
||||||
|
for (pi->resno = pi->poc.resno0; pi->resno < int_min(pi->poc.resno1, comp->numresolutions); pi->resno++) {
|
||||||
|
int levelno;
|
||||||
|
int trx0, try0;
|
||||||
|
int trx1, try1;
|
||||||
|
int rpx, rpy;
|
||||||
|
int prci, prcj;
|
||||||
|
res = &comp->resolutions[pi->resno];
|
||||||
|
levelno = comp->numresolutions - 1 - pi->resno;
|
||||||
|
trx0 = int_ceildiv(pi->tx0, comp->dx << levelno);
|
||||||
|
try0 = int_ceildiv(pi->ty0, comp->dy << levelno);
|
||||||
|
trx1 = int_ceildiv(pi->tx1, comp->dx << levelno);
|
||||||
|
try1 = int_ceildiv(pi->ty1, comp->dy << levelno);
|
||||||
|
rpx = res->pdx + levelno;
|
||||||
|
rpy = res->pdy + levelno;
|
||||||
|
if (!((pi->y % (comp->dy << rpy) == 0) || ((pi->y == pi->ty0) && ((try0 << levelno) % (1 << rpx))))){
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!((pi->x % (comp->dx << rpx) == 0) || ((pi->x == pi->tx0) && ((trx0 << levelno) % (1 << rpx))))){
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((res->pw==0)||(res->pw==0)) continue;
|
||||||
|
|
||||||
|
if ((trx0==trx1)||(try0==try1)) continue;
|
||||||
|
|
||||||
|
prci = int_floordivpow2(int_ceildiv(pi->x, comp->dx << levelno), res->pdx)
|
||||||
|
- int_floordivpow2(trx0, res->pdx);
|
||||||
|
prcj = int_floordivpow2(int_ceildiv(pi->y, comp->dy << levelno), res->pdy)
|
||||||
|
- int_floordivpow2(try0, res->pdy);
|
||||||
|
pi->precno = prci + prcj * res->pw;
|
||||||
|
for (pi->layno = pi->poc.layno0; pi->layno < pi->poc.layno1; pi->layno++) {
|
||||||
|
index = pi->layno * pi->step_l + pi->resno * pi->step_r + pi->compno * pi->step_c + pi->precno * pi->step_p;
|
||||||
|
if (!pi->include[index]) {
|
||||||
|
pi->include[index] = 1;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
LABEL_SKIP:;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool pi_next_cprl(opj_pi_iterator_t * pi) {
|
||||||
|
opj_pi_comp_t *comp = NULL;
|
||||||
|
opj_pi_resolution_t *res = NULL;
|
||||||
|
long index = 0;
|
||||||
|
|
||||||
|
if (!pi->first) {
|
||||||
|
comp = &pi->comps[pi->compno];
|
||||||
|
goto LABEL_SKIP;
|
||||||
|
} else {
|
||||||
|
pi->first = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (pi->compno = pi->poc.compno0; pi->compno < pi->poc.compno1; pi->compno++) {
|
||||||
|
int resno;
|
||||||
|
comp = &pi->comps[pi->compno];
|
||||||
|
pi->dx = 0;
|
||||||
|
pi->dy = 0;
|
||||||
|
for (resno = 0; resno < comp->numresolutions; resno++) {
|
||||||
|
int dx, dy;
|
||||||
|
res = &comp->resolutions[resno];
|
||||||
|
dx = comp->dx * (1 << (res->pdx + comp->numresolutions - 1 - resno));
|
||||||
|
dy = comp->dy * (1 << (res->pdy + comp->numresolutions - 1 - resno));
|
||||||
|
pi->dx = !pi->dx ? dx : int_min(pi->dx, dx);
|
||||||
|
pi->dy = !pi->dy ? dy : int_min(pi->dy, dy);
|
||||||
|
}
|
||||||
|
if (!pi->tp_on){
|
||||||
|
pi->poc.ty0 = pi->ty0;
|
||||||
|
pi->poc.tx0 = pi->tx0;
|
||||||
|
pi->poc.ty1 = pi->ty1;
|
||||||
|
pi->poc.tx1 = pi->tx1;
|
||||||
|
}
|
||||||
|
for (pi->y = pi->poc.ty0; pi->y < pi->poc.ty1; pi->y += pi->dy - (pi->y % pi->dy)) {
|
||||||
|
for (pi->x = pi->poc.tx0; pi->x < pi->poc.tx1; pi->x += pi->dx - (pi->x % pi->dx)) {
|
||||||
|
for (pi->resno = pi->poc.resno0; pi->resno < int_min(pi->poc.resno1, comp->numresolutions); pi->resno++) {
|
||||||
|
int levelno;
|
||||||
|
int trx0, try0;
|
||||||
|
int trx1, try1;
|
||||||
|
int rpx, rpy;
|
||||||
|
int prci, prcj;
|
||||||
|
res = &comp->resolutions[pi->resno];
|
||||||
|
levelno = comp->numresolutions - 1 - pi->resno;
|
||||||
|
trx0 = int_ceildiv(pi->tx0, comp->dx << levelno);
|
||||||
|
try0 = int_ceildiv(pi->ty0, comp->dy << levelno);
|
||||||
|
trx1 = int_ceildiv(pi->tx1, comp->dx << levelno);
|
||||||
|
try1 = int_ceildiv(pi->ty1, comp->dy << levelno);
|
||||||
|
rpx = res->pdx + levelno;
|
||||||
|
rpy = res->pdy + levelno;
|
||||||
|
if (!((pi->y % (comp->dy << rpy) == 0) || ((pi->y == pi->ty0) && ((try0 << levelno) % (1 << rpx))))){
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!((pi->x % (comp->dx << rpx) == 0) || ((pi->x == pi->tx0) && ((trx0 << levelno) % (1 << rpx))))){
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((res->pw==0)||(res->pw==0)) continue;
|
||||||
|
|
||||||
|
if ((trx0==trx1)||(try0==try1)) continue;
|
||||||
|
|
||||||
|
prci = int_floordivpow2(int_ceildiv(pi->x, comp->dx << levelno), res->pdx)
|
||||||
|
- int_floordivpow2(trx0, res->pdx);
|
||||||
|
prcj = int_floordivpow2(int_ceildiv(pi->y, comp->dy << levelno), res->pdy)
|
||||||
|
- int_floordivpow2(try0, res->pdy);
|
||||||
|
pi->precno = prci + prcj * res->pw;
|
||||||
|
for (pi->layno = pi->poc.layno0; pi->layno < pi->poc.layno1; pi->layno++) {
|
||||||
|
index = pi->layno * pi->step_l + pi->resno * pi->step_r + pi->compno * pi->step_c + pi->precno * pi->step_p;
|
||||||
|
if (!pi->include[index]) {
|
||||||
|
pi->include[index] = 1;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
LABEL_SKIP:;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
Packet iterator interface
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
opj_pi_iterator_t *pi_create_decode(opj_image_t *image, opj_cp_t *cp, int tileno) {
|
||||||
|
int p, q;
|
||||||
|
int compno, resno, pino;
|
||||||
|
opj_pi_iterator_t *pi = NULL;
|
||||||
|
opj_tcp_t *tcp = NULL;
|
||||||
|
opj_tccp_t *tccp = NULL;
|
||||||
|
|
||||||
|
tcp = &cp->tcps[tileno];
|
||||||
|
|
||||||
|
pi = (opj_pi_iterator_t*) opj_calloc((tcp->numpocs + 1), sizeof(opj_pi_iterator_t));
|
||||||
|
if(!pi) {
|
||||||
|
/* TODO: throw an error */
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (pino = 0; pino < tcp->numpocs + 1; pino++) { /* change */
|
||||||
|
int maxres = 0;
|
||||||
|
int maxprec = 0;
|
||||||
|
p = tileno % cp->tw;
|
||||||
|
q = tileno / cp->tw;
|
||||||
|
|
||||||
|
pi[pino].tx0 = int_max(cp->tx0 + p * cp->tdx, image->x0);
|
||||||
|
pi[pino].ty0 = int_max(cp->ty0 + q * cp->tdy, image->y0);
|
||||||
|
pi[pino].tx1 = int_min(cp->tx0 + (p + 1) * cp->tdx, image->x1);
|
||||||
|
pi[pino].ty1 = int_min(cp->ty0 + (q + 1) * cp->tdy, image->y1);
|
||||||
|
pi[pino].numcomps = image->numcomps;
|
||||||
|
|
||||||
|
pi[pino].comps = (opj_pi_comp_t*) opj_calloc(image->numcomps, sizeof(opj_pi_comp_t));
|
||||||
|
if(!pi[pino].comps) {
|
||||||
|
/* TODO: throw an error */
|
||||||
|
pi_destroy(pi, cp, tileno);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (compno = 0; compno < pi->numcomps; compno++) {
|
||||||
|
int tcx0, tcy0, tcx1, tcy1;
|
||||||
|
opj_pi_comp_t *comp = &pi[pino].comps[compno];
|
||||||
|
tccp = &tcp->tccps[compno];
|
||||||
|
comp->dx = image->comps[compno].dx;
|
||||||
|
comp->dy = image->comps[compno].dy;
|
||||||
|
comp->numresolutions = tccp->numresolutions;
|
||||||
|
|
||||||
|
comp->resolutions = (opj_pi_resolution_t*) opj_calloc(comp->numresolutions, sizeof(opj_pi_resolution_t));
|
||||||
|
if(!comp->resolutions) {
|
||||||
|
/* TODO: throw an error */
|
||||||
|
pi_destroy(pi, cp, tileno);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
tcx0 = int_ceildiv(pi->tx0, comp->dx);
|
||||||
|
tcy0 = int_ceildiv(pi->ty0, comp->dy);
|
||||||
|
tcx1 = int_ceildiv(pi->tx1, comp->dx);
|
||||||
|
tcy1 = int_ceildiv(pi->ty1, comp->dy);
|
||||||
|
if (comp->numresolutions > maxres) {
|
||||||
|
maxres = comp->numresolutions;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (resno = 0; resno < comp->numresolutions; resno++) {
|
||||||
|
int levelno;
|
||||||
|
int rx0, ry0, rx1, ry1;
|
||||||
|
int px0, py0, px1, py1;
|
||||||
|
opj_pi_resolution_t *res = &comp->resolutions[resno];
|
||||||
|
if (tccp->csty & J2K_CCP_CSTY_PRT) {
|
||||||
|
res->pdx = tccp->prcw[resno];
|
||||||
|
res->pdy = tccp->prch[resno];
|
||||||
|
} else {
|
||||||
|
res->pdx = 15;
|
||||||
|
res->pdy = 15;
|
||||||
|
}
|
||||||
|
levelno = comp->numresolutions - 1 - resno;
|
||||||
|
rx0 = int_ceildivpow2(tcx0, levelno);
|
||||||
|
ry0 = int_ceildivpow2(tcy0, levelno);
|
||||||
|
rx1 = int_ceildivpow2(tcx1, levelno);
|
||||||
|
ry1 = int_ceildivpow2(tcy1, levelno);
|
||||||
|
px0 = int_floordivpow2(rx0, res->pdx) << res->pdx;
|
||||||
|
py0 = int_floordivpow2(ry0, res->pdy) << res->pdy;
|
||||||
|
px1 = int_ceildivpow2(rx1, res->pdx) << res->pdx;
|
||||||
|
py1 = int_ceildivpow2(ry1, res->pdy) << res->pdy;
|
||||||
|
res->pw = (rx0==rx1)?0:((px1 - px0) >> res->pdx);
|
||||||
|
res->ph = (ry0==ry1)?0:((py1 - py0) >> res->pdy);
|
||||||
|
|
||||||
|
if (res->pw*res->ph > maxprec) {
|
||||||
|
maxprec = res->pw*res->ph;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tccp = &tcp->tccps[0];
|
||||||
|
pi[pino].step_p = 1;
|
||||||
|
pi[pino].step_c = maxprec * pi[pino].step_p;
|
||||||
|
pi[pino].step_r = image->numcomps * pi[pino].step_c;
|
||||||
|
pi[pino].step_l = maxres * pi[pino].step_r;
|
||||||
|
|
||||||
|
if (pino == 0) {
|
||||||
|
pi[pino].include = (short int*) opj_calloc(image->numcomps * maxres * tcp->numlayers * maxprec, sizeof(short int));
|
||||||
|
if(!pi[pino].include) {
|
||||||
|
/* TODO: throw an error */
|
||||||
|
pi_destroy(pi, cp, tileno);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
pi[pino].include = pi[pino - 1].include;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tcp->POC == 0) {
|
||||||
|
pi[pino].first = 1;
|
||||||
|
pi[pino].poc.resno0 = 0;
|
||||||
|
pi[pino].poc.compno0 = 0;
|
||||||
|
pi[pino].poc.layno1 = tcp->numlayers;
|
||||||
|
pi[pino].poc.resno1 = maxres;
|
||||||
|
pi[pino].poc.compno1 = image->numcomps;
|
||||||
|
pi[pino].poc.prg = tcp->prg;
|
||||||
|
} else {
|
||||||
|
pi[pino].first = 1;
|
||||||
|
pi[pino].poc.resno0 = tcp->pocs[pino].resno0;
|
||||||
|
pi[pino].poc.compno0 = tcp->pocs[pino].compno0;
|
||||||
|
pi[pino].poc.layno1 = tcp->pocs[pino].layno1;
|
||||||
|
pi[pino].poc.resno1 = tcp->pocs[pino].resno1;
|
||||||
|
pi[pino].poc.compno1 = tcp->pocs[pino].compno1;
|
||||||
|
pi[pino].poc.prg = tcp->pocs[pino].prg;
|
||||||
|
}
|
||||||
|
pi[pino].poc.layno0 = 0;
|
||||||
|
pi[pino].poc.precno0 = 0;
|
||||||
|
pi[pino].poc.precno1 = maxprec;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return pi;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
opj_pi_iterator_t *pi_initialise_encode(opj_image_t *image, opj_cp_t *cp, int tileno, J2K_T2_MODE t2_mode){
|
||||||
|
int p, q, pino;
|
||||||
|
int compno, resno;
|
||||||
|
int maxres = 0;
|
||||||
|
int maxprec = 0;
|
||||||
|
opj_pi_iterator_t *pi = NULL;
|
||||||
|
opj_tcp_t *tcp = NULL;
|
||||||
|
opj_tccp_t *tccp = NULL;
|
||||||
|
|
||||||
|
tcp = &cp->tcps[tileno];
|
||||||
|
|
||||||
|
pi = (opj_pi_iterator_t*) opj_calloc((tcp->numpocs + 1), sizeof(opj_pi_iterator_t));
|
||||||
|
if(!pi) { return NULL;}
|
||||||
|
pi->tp_on = cp->tp_on;
|
||||||
|
|
||||||
|
for(pino = 0;pino < tcp->numpocs+1 ; pino ++){
|
||||||
|
p = tileno % cp->tw;
|
||||||
|
q = tileno / cp->tw;
|
||||||
|
|
||||||
|
pi[pino].tx0 = int_max(cp->tx0 + p * cp->tdx, image->x0);
|
||||||
|
pi[pino].ty0 = int_max(cp->ty0 + q * cp->tdy, image->y0);
|
||||||
|
pi[pino].tx1 = int_min(cp->tx0 + (p + 1) * cp->tdx, image->x1);
|
||||||
|
pi[pino].ty1 = int_min(cp->ty0 + (q + 1) * cp->tdy, image->y1);
|
||||||
|
pi[pino].numcomps = image->numcomps;
|
||||||
|
|
||||||
|
pi[pino].comps = (opj_pi_comp_t*) opj_calloc(image->numcomps, sizeof(opj_pi_comp_t));
|
||||||
|
if(!pi[pino].comps) {
|
||||||
|
pi_destroy(pi, cp, tileno);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (compno = 0; compno < pi[pino].numcomps; compno++) {
|
||||||
|
int tcx0, tcy0, tcx1, tcy1;
|
||||||
|
opj_pi_comp_t *comp = &pi[pino].comps[compno];
|
||||||
|
tccp = &tcp->tccps[compno];
|
||||||
|
comp->dx = image->comps[compno].dx;
|
||||||
|
comp->dy = image->comps[compno].dy;
|
||||||
|
comp->numresolutions = tccp->numresolutions;
|
||||||
|
|
||||||
|
comp->resolutions = (opj_pi_resolution_t*) opj_malloc(comp->numresolutions * sizeof(opj_pi_resolution_t));
|
||||||
|
if(!comp->resolutions) {
|
||||||
|
pi_destroy(pi, cp, tileno);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
tcx0 = int_ceildiv(pi[pino].tx0, comp->dx);
|
||||||
|
tcy0 = int_ceildiv(pi[pino].ty0, comp->dy);
|
||||||
|
tcx1 = int_ceildiv(pi[pino].tx1, comp->dx);
|
||||||
|
tcy1 = int_ceildiv(pi[pino].ty1, comp->dy);
|
||||||
|
if (comp->numresolutions > maxres) {
|
||||||
|
maxres = comp->numresolutions;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (resno = 0; resno < comp->numresolutions; resno++) {
|
||||||
|
int levelno;
|
||||||
|
int rx0, ry0, rx1, ry1;
|
||||||
|
int px0, py0, px1, py1;
|
||||||
|
opj_pi_resolution_t *res = &comp->resolutions[resno];
|
||||||
|
if (tccp->csty & J2K_CCP_CSTY_PRT) {
|
||||||
|
res->pdx = tccp->prcw[resno];
|
||||||
|
res->pdy = tccp->prch[resno];
|
||||||
|
} else {
|
||||||
|
res->pdx = 15;
|
||||||
|
res->pdy = 15;
|
||||||
|
}
|
||||||
|
levelno = comp->numresolutions - 1 - resno;
|
||||||
|
rx0 = int_ceildivpow2(tcx0, levelno);
|
||||||
|
ry0 = int_ceildivpow2(tcy0, levelno);
|
||||||
|
rx1 = int_ceildivpow2(tcx1, levelno);
|
||||||
|
ry1 = int_ceildivpow2(tcy1, levelno);
|
||||||
|
px0 = int_floordivpow2(rx0, res->pdx) << res->pdx;
|
||||||
|
py0 = int_floordivpow2(ry0, res->pdy) << res->pdy;
|
||||||
|
px1 = int_ceildivpow2(rx1, res->pdx) << res->pdx;
|
||||||
|
py1 = int_ceildivpow2(ry1, res->pdy) << res->pdy;
|
||||||
|
res->pw = (rx0==rx1)?0:((px1 - px0) >> res->pdx);
|
||||||
|
res->ph = (ry0==ry1)?0:((py1 - py0) >> res->pdy);
|
||||||
|
|
||||||
|
if (res->pw*res->ph > maxprec) {
|
||||||
|
maxprec = res->pw * res->ph;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tccp = &tcp->tccps[0];
|
||||||
|
pi[pino].step_p = 1;
|
||||||
|
pi[pino].step_c = maxprec * pi[pino].step_p;
|
||||||
|
pi[pino].step_r = image->numcomps * pi[pino].step_c;
|
||||||
|
pi[pino].step_l = maxres * pi[pino].step_r;
|
||||||
|
|
||||||
|
for (compno = 0; compno < pi->numcomps; compno++) {
|
||||||
|
opj_pi_comp_t *comp = &pi->comps[compno];
|
||||||
|
for (resno = 0; resno < comp->numresolutions; resno++) {
|
||||||
|
int dx, dy;
|
||||||
|
opj_pi_resolution_t *res = &comp->resolutions[resno];
|
||||||
|
dx = comp->dx * (1 << (res->pdx + comp->numresolutions - 1 - resno));
|
||||||
|
dy = comp->dy * (1 << (res->pdy + comp->numresolutions - 1 - resno));
|
||||||
|
pi[pino].dx = !pi->dx ? dx : int_min(pi->dx, dx);
|
||||||
|
pi[pino].dy = !pi->dy ? dy : int_min(pi->dy, dy);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pino == 0) {
|
||||||
|
pi[pino].include = (short int*) opj_calloc(tcp->numlayers * pi[pino].step_l, sizeof(short int));
|
||||||
|
if(!pi[pino].include) {
|
||||||
|
pi_destroy(pi, cp, tileno);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
pi[pino].include = pi[pino - 1].include;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Generation of boundaries for each prog flag*/
|
||||||
|
if(tcp->POC && ( cp->cinema || ((!cp->cinema) && (t2_mode == FINAL_PASS)))){
|
||||||
|
tcp->pocs[pino].compS= tcp->pocs[pino].compno0;
|
||||||
|
tcp->pocs[pino].compE= tcp->pocs[pino].compno1;
|
||||||
|
tcp->pocs[pino].resS = tcp->pocs[pino].resno0;
|
||||||
|
tcp->pocs[pino].resE = tcp->pocs[pino].resno1;
|
||||||
|
tcp->pocs[pino].layE = tcp->pocs[pino].layno1;
|
||||||
|
tcp->pocs[pino].prg = tcp->pocs[pino].prg1;
|
||||||
|
if (pino > 0)
|
||||||
|
tcp->pocs[pino].layS = (tcp->pocs[pino].layE > tcp->pocs[pino - 1].layE) ? tcp->pocs[pino - 1].layE : 0;
|
||||||
|
}else {
|
||||||
|
tcp->pocs[pino].compS= 0;
|
||||||
|
tcp->pocs[pino].compE= image->numcomps;
|
||||||
|
tcp->pocs[pino].resS = 0;
|
||||||
|
tcp->pocs[pino].resE = maxres;
|
||||||
|
tcp->pocs[pino].layS = 0;
|
||||||
|
tcp->pocs[pino].layE = tcp->numlayers;
|
||||||
|
tcp->pocs[pino].prg = tcp->prg;
|
||||||
|
}
|
||||||
|
tcp->pocs[pino].prcS = 0;
|
||||||
|
tcp->pocs[pino].prcE = maxprec;;
|
||||||
|
tcp->pocs[pino].txS = pi[pino].tx0;
|
||||||
|
tcp->pocs[pino].txE = pi[pino].tx1;
|
||||||
|
tcp->pocs[pino].tyS = pi[pino].ty0;
|
||||||
|
tcp->pocs[pino].tyE = pi[pino].ty1;
|
||||||
|
tcp->pocs[pino].dx = pi[pino].dx;
|
||||||
|
tcp->pocs[pino].dy = pi[pino].dy;
|
||||||
|
}
|
||||||
|
return pi;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
void pi_destroy(opj_pi_iterator_t *pi, opj_cp_t *cp, int tileno) {
|
||||||
|
int compno, pino;
|
||||||
|
opj_tcp_t *tcp = &cp->tcps[tileno];
|
||||||
|
if(pi) {
|
||||||
|
for (pino = 0; pino < tcp->numpocs + 1; pino++) {
|
||||||
|
if(pi[pino].comps) {
|
||||||
|
for (compno = 0; compno < pi->numcomps; compno++) {
|
||||||
|
opj_pi_comp_t *comp = &pi[pino].comps[compno];
|
||||||
|
if(comp->resolutions) {
|
||||||
|
opj_free(comp->resolutions);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
opj_free(pi[pino].comps);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(pi->include) {
|
||||||
|
opj_free(pi->include);
|
||||||
|
}
|
||||||
|
opj_free(pi);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool pi_next(opj_pi_iterator_t * pi) {
|
||||||
|
switch (pi->poc.prg) {
|
||||||
|
case LRCP:
|
||||||
|
return pi_next_lrcp(pi);
|
||||||
|
case RLCP:
|
||||||
|
return pi_next_rlcp(pi);
|
||||||
|
case RPCL:
|
||||||
|
return pi_next_rpcl(pi);
|
||||||
|
case PCRL:
|
||||||
|
return pi_next_pcrl(pi);
|
||||||
|
case CPRL:
|
||||||
|
return pi_next_cprl(pi);
|
||||||
|
case PROG_UNKNOWN:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool pi_create_encode( opj_pi_iterator_t *pi, opj_cp_t *cp,int tileno, int pino,int tpnum, int tppos, J2K_T2_MODE t2_mode,int cur_totnum_tp){
|
||||||
|
char prog[4];
|
||||||
|
int i;
|
||||||
|
int incr_top=1,resetX=0;
|
||||||
|
opj_tcp_t *tcps =&cp->tcps[tileno];
|
||||||
|
opj_poc_t *tcp= &tcps->pocs[pino];
|
||||||
|
|
||||||
|
pi[pino].first = 1;
|
||||||
|
pi[pino].poc.prg = tcp->prg;
|
||||||
|
|
||||||
|
switch(tcp->prg){
|
||||||
|
case CPRL: strncpy(prog, "CPRL",4);
|
||||||
|
break;
|
||||||
|
case LRCP: strncpy(prog, "LRCP",4);
|
||||||
|
break;
|
||||||
|
case PCRL: strncpy(prog, "PCRL",4);
|
||||||
|
break;
|
||||||
|
case RLCP: strncpy(prog, "RLCP",4);
|
||||||
|
break;
|
||||||
|
case RPCL: strncpy(prog, "RPCL",4);
|
||||||
|
break;
|
||||||
|
case PROG_UNKNOWN:
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!(cp->tp_on && ((!cp->cinema && (t2_mode == FINAL_PASS)) || cp->cinema))){
|
||||||
|
pi[pino].poc.resno0 = tcp->resS;
|
||||||
|
pi[pino].poc.resno1 = tcp->resE;
|
||||||
|
pi[pino].poc.compno0 = tcp->compS;
|
||||||
|
pi[pino].poc.compno1 = tcp->compE;
|
||||||
|
pi[pino].poc.layno0 = tcp->layS;
|
||||||
|
pi[pino].poc.layno1 = tcp->layE;
|
||||||
|
pi[pino].poc.precno0 = tcp->prcS;
|
||||||
|
pi[pino].poc.precno1 = tcp->prcE;
|
||||||
|
pi[pino].poc.tx0 = tcp->txS;
|
||||||
|
pi[pino].poc.ty0 = tcp->tyS;
|
||||||
|
pi[pino].poc.tx1 = tcp->txE;
|
||||||
|
pi[pino].poc.ty1 = tcp->tyE;
|
||||||
|
}else {
|
||||||
|
if( tpnum < cur_totnum_tp){
|
||||||
|
for(i=3;i>=0;i--){
|
||||||
|
switch(prog[i]){
|
||||||
|
case 'C':
|
||||||
|
if (i > tppos){
|
||||||
|
pi[pino].poc.compno0 = tcp->compS;
|
||||||
|
pi[pino].poc.compno1 = tcp->compE;
|
||||||
|
}else{
|
||||||
|
if (tpnum == 0){
|
||||||
|
tcp->comp_t = tcp->compS;
|
||||||
|
pi[pino].poc.compno0 = tcp->comp_t;
|
||||||
|
pi[pino].poc.compno1 = tcp->comp_t+1;
|
||||||
|
tcp->comp_t+=1;
|
||||||
|
}else{
|
||||||
|
if (incr_top == 1){
|
||||||
|
if(tcp->comp_t ==tcp->compE){
|
||||||
|
tcp->comp_t = tcp->compS;
|
||||||
|
pi[pino].poc.compno0 = tcp->comp_t;
|
||||||
|
pi[pino].poc.compno1 = tcp->comp_t+1;
|
||||||
|
tcp->comp_t+=1;
|
||||||
|
incr_top=1;
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.compno0 = tcp->comp_t;
|
||||||
|
pi[pino].poc.compno1 = tcp->comp_t+1;
|
||||||
|
tcp->comp_t+=1;
|
||||||
|
incr_top=0;
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.compno0 = tcp->comp_t-1;
|
||||||
|
pi[pino].poc.compno1 = tcp->comp_t;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'R':
|
||||||
|
if (i > tppos){
|
||||||
|
pi[pino].poc.resno0 = tcp->resS;
|
||||||
|
pi[pino].poc.resno1 = tcp->resE;
|
||||||
|
}else{
|
||||||
|
if (tpnum == 0){
|
||||||
|
tcp->res_t = tcp->resS;
|
||||||
|
pi[pino].poc.resno0 = tcp->res_t;
|
||||||
|
pi[pino].poc.resno1 = tcp->res_t+1;
|
||||||
|
tcp->res_t+=1;
|
||||||
|
}else{
|
||||||
|
if (incr_top == 1){
|
||||||
|
if(tcp->res_t==tcp->resE){
|
||||||
|
tcp->res_t = tcp->resS;
|
||||||
|
pi[pino].poc.resno0 = tcp->res_t;
|
||||||
|
pi[pino].poc.resno1 = tcp->res_t+1;
|
||||||
|
tcp->res_t+=1;
|
||||||
|
incr_top=1;
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.resno0 = tcp->res_t;
|
||||||
|
pi[pino].poc.resno1 = tcp->res_t+1;
|
||||||
|
tcp->res_t+=1;
|
||||||
|
incr_top=0;
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.resno0 = tcp->res_t - 1;
|
||||||
|
pi[pino].poc.resno1 = tcp->res_t;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'L':
|
||||||
|
if (i > tppos){
|
||||||
|
pi[pino].poc.layno0 = tcp->layS;
|
||||||
|
pi[pino].poc.layno1 = tcp->layE;
|
||||||
|
}else{
|
||||||
|
if (tpnum == 0){
|
||||||
|
tcp->lay_t = tcp->layS;
|
||||||
|
pi[pino].poc.layno0 = tcp->lay_t;
|
||||||
|
pi[pino].poc.layno1 = tcp->lay_t+1;
|
||||||
|
tcp->lay_t+=1;
|
||||||
|
}else{
|
||||||
|
if (incr_top == 1){
|
||||||
|
if(tcp->lay_t == tcp->layE){
|
||||||
|
tcp->lay_t = tcp->layS;
|
||||||
|
pi[pino].poc.layno0 = tcp->lay_t;
|
||||||
|
pi[pino].poc.layno1 = tcp->lay_t+1;
|
||||||
|
tcp->lay_t+=1;
|
||||||
|
incr_top=1;
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.layno0 = tcp->lay_t;
|
||||||
|
pi[pino].poc.layno1 = tcp->lay_t+1;
|
||||||
|
tcp->lay_t+=1;
|
||||||
|
incr_top=0;
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.layno0 = tcp->lay_t - 1;
|
||||||
|
pi[pino].poc.layno1 = tcp->lay_t;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'P':
|
||||||
|
switch(tcp->prg){
|
||||||
|
case LRCP:
|
||||||
|
case RLCP:
|
||||||
|
if (i > tppos){
|
||||||
|
pi[pino].poc.precno0 = tcp->prcS;
|
||||||
|
pi[pino].poc.precno1 = tcp->prcE;
|
||||||
|
}else{
|
||||||
|
if (tpnum == 0){
|
||||||
|
tcp->prc_t = tcp->prcS;
|
||||||
|
pi[pino].poc.precno0 = tcp->prc_t;
|
||||||
|
pi[pino].poc.precno1 = tcp->prc_t+1;
|
||||||
|
tcp->prc_t+=1;
|
||||||
|
}else{
|
||||||
|
if (incr_top == 1){
|
||||||
|
if(tcp->prc_t == tcp->prcE){
|
||||||
|
tcp->prc_t = tcp->prcS;
|
||||||
|
pi[pino].poc.precno0 = tcp->prc_t;
|
||||||
|
pi[pino].poc.precno1 = tcp->prc_t+1;
|
||||||
|
tcp->prc_t+=1;
|
||||||
|
incr_top=1;
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.precno0 = tcp->prc_t;
|
||||||
|
pi[pino].poc.precno1 = tcp->prc_t+1;
|
||||||
|
tcp->prc_t+=1;
|
||||||
|
incr_top=0;
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.precno0 = tcp->prc_t - 1;
|
||||||
|
pi[pino].poc.precno1 = tcp->prc_t;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (i > tppos){
|
||||||
|
pi[pino].poc.tx0 = tcp->txS;
|
||||||
|
pi[pino].poc.ty0 = tcp->tyS;
|
||||||
|
pi[pino].poc.tx1 = tcp->txE;
|
||||||
|
pi[pino].poc.ty1 = tcp->tyE;
|
||||||
|
}else{
|
||||||
|
if (tpnum == 0){
|
||||||
|
tcp->tx0_t = tcp->txS;
|
||||||
|
tcp->ty0_t = tcp->tyS;
|
||||||
|
pi[pino].poc.tx0 = tcp->tx0_t;
|
||||||
|
pi[pino].poc.tx1 = tcp->tx0_t + tcp->dx - (tcp->tx0_t % tcp->dx);
|
||||||
|
pi[pino].poc.ty0 = tcp->ty0_t;
|
||||||
|
pi[pino].poc.ty1 = tcp->ty0_t + tcp->dy - (tcp->ty0_t % tcp->dy);
|
||||||
|
tcp->tx0_t = pi[pino].poc.tx1;
|
||||||
|
tcp->ty0_t = pi[pino].poc.ty1;
|
||||||
|
}else{
|
||||||
|
if (incr_top == 1){
|
||||||
|
if(tcp->tx0_t >= tcp->txE){
|
||||||
|
if(tcp->ty0_t >= tcp->tyE){
|
||||||
|
tcp->ty0_t = tcp->tyS;
|
||||||
|
pi[pino].poc.ty0 = tcp->ty0_t;
|
||||||
|
pi[pino].poc.ty1 = tcp->ty0_t + tcp->dy - (tcp->ty0_t % tcp->dy);
|
||||||
|
tcp->ty0_t = pi[pino].poc.ty1;
|
||||||
|
incr_top=1;resetX=1;
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.ty0 = tcp->ty0_t;
|
||||||
|
pi[pino].poc.ty1 = tcp->ty0_t + tcp->dy - (tcp->ty0_t % tcp->dy);
|
||||||
|
tcp->ty0_t = pi[pino].poc.ty1;
|
||||||
|
incr_top=0;resetX=1;
|
||||||
|
}
|
||||||
|
if(resetX==1){
|
||||||
|
tcp->tx0_t = tcp->txS;
|
||||||
|
pi[pino].poc.tx0 = tcp->tx0_t;
|
||||||
|
pi[pino].poc.tx1 = tcp->tx0_t + tcp->dx- (tcp->tx0_t % tcp->dx);
|
||||||
|
tcp->tx0_t = pi[pino].poc.tx1;
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.tx0 = tcp->tx0_t;
|
||||||
|
pi[pino].poc.tx1 = tcp->tx0_t + tcp->dx- (tcp->tx0_t % tcp->dx);
|
||||||
|
tcp->tx0_t = pi[pino].poc.tx1;
|
||||||
|
pi[pino].poc.ty0 = tcp->ty0_t - tcp->dy - (tcp->ty0_t % tcp->dy);
|
||||||
|
pi[pino].poc.ty1 = tcp->ty0_t ;
|
||||||
|
incr_top=0;
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
pi[pino].poc.tx0 = tcp->tx0_t - tcp->dx - (tcp->tx0_t % tcp->dx);
|
||||||
|
pi[pino].poc.tx1 = tcp->tx0_t ;
|
||||||
|
pi[pino].poc.ty0 = tcp->ty0_t - tcp->dy - (tcp->ty0_t % tcp->dy);
|
||||||
|
pi[pino].poc.ty1 = tcp->ty0_t ;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
154
extern/libopenjpeg/pi.h
vendored
Normal file
154
extern/libopenjpeg/pi.h
vendored
Normal file
@@ -0,0 +1,154 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __PI_H
|
||||||
|
#define __PI_H
|
||||||
|
/**
|
||||||
|
@file pi.h
|
||||||
|
@brief Implementation of a packet iterator (PI)
|
||||||
|
|
||||||
|
The functions in PI.C have for goal to realize a packet iterator that permits to get the next
|
||||||
|
packet following the progression order and change of it. The functions in PI.C are used
|
||||||
|
by some function in T2.C.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup PI PI - Implementation of a packet iterator */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_pi_resolution {
|
||||||
|
int pdx, pdy;
|
||||||
|
int pw, ph;
|
||||||
|
} opj_pi_resolution_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_pi_comp {
|
||||||
|
int dx, dy;
|
||||||
|
/** number of resolution levels */
|
||||||
|
int numresolutions;
|
||||||
|
opj_pi_resolution_t *resolutions;
|
||||||
|
} opj_pi_comp_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Packet iterator
|
||||||
|
*/
|
||||||
|
typedef struct opj_pi_iterator {
|
||||||
|
/** Enabling Tile part generation*/
|
||||||
|
char tp_on;
|
||||||
|
/** precise if the packet has been already used (usefull for progression order change) */
|
||||||
|
short int *include;
|
||||||
|
/** layer step used to localize the packet in the include vector */
|
||||||
|
int step_l;
|
||||||
|
/** resolution step used to localize the packet in the include vector */
|
||||||
|
int step_r;
|
||||||
|
/** component step used to localize the packet in the include vector */
|
||||||
|
int step_c;
|
||||||
|
/** precinct step used to localize the packet in the include vector */
|
||||||
|
int step_p;
|
||||||
|
/** component that identify the packet */
|
||||||
|
int compno;
|
||||||
|
/** resolution that identify the packet */
|
||||||
|
int resno;
|
||||||
|
/** precinct that identify the packet */
|
||||||
|
int precno;
|
||||||
|
/** layer that identify the packet */
|
||||||
|
int layno;
|
||||||
|
/** 0 if the first packet */
|
||||||
|
int first;
|
||||||
|
/** progression order change information */
|
||||||
|
opj_poc_t poc;
|
||||||
|
/** number of components in the image */
|
||||||
|
int numcomps;
|
||||||
|
/** Components*/
|
||||||
|
opj_pi_comp_t *comps;
|
||||||
|
int tx0, ty0, tx1, ty1;
|
||||||
|
int x, y, dx, dy;
|
||||||
|
} opj_pi_iterator_t;
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Create a packet iterator for Encoder
|
||||||
|
@param image Raw image for which the packets will be listed
|
||||||
|
@param cp Coding parameters
|
||||||
|
@param tileno Number that identifies the tile for which to list the packets
|
||||||
|
@param t2_mode If == 0 In Threshold calculation ,If == 1 Final pass
|
||||||
|
@return Returns a packet iterator that points to the first packet of the tile
|
||||||
|
@see pi_destroy
|
||||||
|
*/
|
||||||
|
opj_pi_iterator_t *pi_initialise_encode(opj_image_t *image, opj_cp_t *cp, int tileno,J2K_T2_MODE t2_mode);
|
||||||
|
/**
|
||||||
|
Modify the packet iterator for enabling tile part generation
|
||||||
|
@param pi Handle to the packet iterator generated in pi_initialise_encode
|
||||||
|
@param cp Coding parameters
|
||||||
|
@param tileno Number that identifies the tile for which to list the packets
|
||||||
|
@param tpnum Tile part number of the current tile
|
||||||
|
@param tppos The position of the tile part flag in the progression order
|
||||||
|
@param cur_totnum_tp The total number of tile parts in the current tile
|
||||||
|
@return Returns true if an error is detected
|
||||||
|
*/
|
||||||
|
bool pi_create_encode(opj_pi_iterator_t *pi, opj_cp_t *cp,int tileno, int pino,int tpnum, int tppos, J2K_T2_MODE t2_mode,int cur_totnum_tp);
|
||||||
|
/**
|
||||||
|
Create a packet iterator for Decoder
|
||||||
|
@param image Raw image for which the packets will be listed
|
||||||
|
@param cp Coding parameters
|
||||||
|
@param tileno Number that identifies the tile for which to list the packets
|
||||||
|
@return Returns a packet iterator that points to the first packet of the tile
|
||||||
|
@see pi_destroy
|
||||||
|
*/
|
||||||
|
opj_pi_iterator_t *pi_create_decode(opj_image_t * image, opj_cp_t * cp, int tileno);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Destroy a packet iterator
|
||||||
|
@param pi Previously created packet iterator
|
||||||
|
@param cp Coding parameters
|
||||||
|
@param tileno Number that identifies the tile for which the packets were listed
|
||||||
|
@see pi_create
|
||||||
|
*/
|
||||||
|
void pi_destroy(opj_pi_iterator_t *pi, opj_cp_t *cp, int tileno);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Modify the packet iterator to point to the next packet
|
||||||
|
@param pi Packet iterator to modify
|
||||||
|
@return Returns false if pi pointed to the last packet or else returns true
|
||||||
|
*/
|
||||||
|
bool pi_next(opj_pi_iterator_t * pi);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __PI_H */
|
||||||
87
extern/libopenjpeg/raw.c
vendored
Normal file
87
extern/libopenjpeg/raw.c
vendored
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
local functions
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
RAW encoding interface
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
opj_raw_t* raw_create(void) {
|
||||||
|
opj_raw_t *raw = (opj_raw_t*)opj_malloc(sizeof(opj_raw_t));
|
||||||
|
return raw;
|
||||||
|
}
|
||||||
|
|
||||||
|
void raw_destroy(opj_raw_t *raw) {
|
||||||
|
if(raw) {
|
||||||
|
opj_free(raw);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int raw_numbytes(opj_raw_t *raw) {
|
||||||
|
return raw->bp - raw->start;
|
||||||
|
}
|
||||||
|
|
||||||
|
void raw_init_dec(opj_raw_t *raw, unsigned char *bp, int len) {
|
||||||
|
raw->start = bp;
|
||||||
|
raw->lenmax = len;
|
||||||
|
raw->len = 0;
|
||||||
|
raw->c = 0;
|
||||||
|
raw->ct = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int raw_decode(opj_raw_t *raw) {
|
||||||
|
int d;
|
||||||
|
if (raw->ct == 0) {
|
||||||
|
raw->ct = 8;
|
||||||
|
if (raw->len == raw->lenmax) {
|
||||||
|
raw->c = 0xff;
|
||||||
|
} else {
|
||||||
|
if (raw->c == 0xff) {
|
||||||
|
raw->ct = 7;
|
||||||
|
}
|
||||||
|
raw->c = *(raw->start + raw->len);
|
||||||
|
raw->len++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
raw->ct--;
|
||||||
|
d = (raw->c >> raw->ct) & 0x01;
|
||||||
|
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
|
||||||
100
extern/libopenjpeg/raw.h
vendored
Normal file
100
extern/libopenjpeg/raw.h
vendored
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __RAW_H
|
||||||
|
#define __RAW_H
|
||||||
|
/**
|
||||||
|
@file raw.h
|
||||||
|
@brief Implementation of operations for raw encoding (RAW)
|
||||||
|
|
||||||
|
The functions in RAW.C have for goal to realize the operation of raw encoding linked
|
||||||
|
with the corresponding mode switch.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup RAW RAW - Implementation of operations for raw encoding */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
RAW encoding operations
|
||||||
|
*/
|
||||||
|
typedef struct opj_raw {
|
||||||
|
/** temporary buffer where bits are coded or decoded */
|
||||||
|
unsigned char c;
|
||||||
|
/** number of bits already read or free to write */
|
||||||
|
unsigned int ct;
|
||||||
|
/** maximum length to decode */
|
||||||
|
unsigned int lenmax;
|
||||||
|
/** length decoded */
|
||||||
|
unsigned int len;
|
||||||
|
/** pointer to the current position in the buffer */
|
||||||
|
unsigned char *bp;
|
||||||
|
/** pointer to the start of the buffer */
|
||||||
|
unsigned char *start;
|
||||||
|
/** pointer to the end of the buffer */
|
||||||
|
unsigned char *end;
|
||||||
|
} opj_raw_t;
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Create a new RAW handle
|
||||||
|
@return Returns a new RAW handle if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_raw_t* raw_create(void);
|
||||||
|
/**
|
||||||
|
Destroy a previously created RAW handle
|
||||||
|
@param raw RAW handle to destroy
|
||||||
|
*/
|
||||||
|
void raw_destroy(opj_raw_t *raw);
|
||||||
|
/**
|
||||||
|
Return the number of bytes written/read since initialisation
|
||||||
|
@param raw RAW handle to destroy
|
||||||
|
@return Returns the number of bytes already encoded
|
||||||
|
*/
|
||||||
|
int raw_numbytes(opj_raw_t *raw);
|
||||||
|
/**
|
||||||
|
Initialize the decoder
|
||||||
|
@param raw RAW handle
|
||||||
|
@param bp Pointer to the start of the buffer from which the bytes will be read
|
||||||
|
@param len Length of the input buffer
|
||||||
|
*/
|
||||||
|
void raw_init_dec(opj_raw_t *raw, unsigned char *bp, int len);
|
||||||
|
/**
|
||||||
|
Decode a symbol using raw-decoder. Cfr p.506 TAUBMAN
|
||||||
|
@param raw RAW handle
|
||||||
|
@return Returns the decoded symbol (0 or 1)
|
||||||
|
*/
|
||||||
|
int raw_decode(opj_raw_t *raw);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __RAW_H */
|
||||||
1208
extern/libopenjpeg/t1.c
vendored
Normal file
1208
extern/libopenjpeg/t1.c
vendored
Normal file
File diff suppressed because it is too large
Load Diff
147
extern/libopenjpeg/t1.h
vendored
Normal file
147
extern/libopenjpeg/t1.h
vendored
Normal file
@@ -0,0 +1,147 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __T1_H
|
||||||
|
#define __T1_H
|
||||||
|
/**
|
||||||
|
@file t1.h
|
||||||
|
@brief Implementation of the tier-1 coding (coding of code-block coefficients) (T1)
|
||||||
|
|
||||||
|
The functions in T1.C have for goal to realize the tier-1 coding operation. The functions
|
||||||
|
in T1.C are used by some function in TCD.C.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup T1 T1 - Implementation of the tier-1 coding */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
#define T1_NMSEDEC_BITS 7
|
||||||
|
|
||||||
|
#define T1_SIG_NE 0x0001 /**< Context orientation : North-East direction */
|
||||||
|
#define T1_SIG_SE 0x0002 /**< Context orientation : South-East direction */
|
||||||
|
#define T1_SIG_SW 0x0004 /**< Context orientation : South-West direction */
|
||||||
|
#define T1_SIG_NW 0x0008 /**< Context orientation : North-West direction */
|
||||||
|
#define T1_SIG_N 0x0010 /**< Context orientation : North direction */
|
||||||
|
#define T1_SIG_E 0x0020 /**< Context orientation : East direction */
|
||||||
|
#define T1_SIG_S 0x0040 /**< Context orientation : South direction */
|
||||||
|
#define T1_SIG_W 0x0080 /**< Context orientation : West direction */
|
||||||
|
#define T1_SIG_OTH (T1_SIG_N|T1_SIG_NE|T1_SIG_E|T1_SIG_SE|T1_SIG_S|T1_SIG_SW|T1_SIG_W|T1_SIG_NW)
|
||||||
|
#define T1_SIG_PRIM (T1_SIG_N|T1_SIG_E|T1_SIG_S|T1_SIG_W)
|
||||||
|
|
||||||
|
#define T1_SGN_N 0x0100
|
||||||
|
#define T1_SGN_E 0x0200
|
||||||
|
#define T1_SGN_S 0x0400
|
||||||
|
#define T1_SGN_W 0x0800
|
||||||
|
#define T1_SGN (T1_SGN_N|T1_SGN_E|T1_SGN_S|T1_SGN_W)
|
||||||
|
|
||||||
|
#define T1_SIG 0x1000
|
||||||
|
#define T1_REFINE 0x2000
|
||||||
|
#define T1_VISIT 0x4000
|
||||||
|
|
||||||
|
#define T1_NUMCTXS_ZC 9
|
||||||
|
#define T1_NUMCTXS_SC 5
|
||||||
|
#define T1_NUMCTXS_MAG 3
|
||||||
|
#define T1_NUMCTXS_AGG 1
|
||||||
|
#define T1_NUMCTXS_UNI 1
|
||||||
|
|
||||||
|
#define T1_CTXNO_ZC 0
|
||||||
|
#define T1_CTXNO_SC (T1_CTXNO_ZC+T1_NUMCTXS_ZC)
|
||||||
|
#define T1_CTXNO_MAG (T1_CTXNO_SC+T1_NUMCTXS_SC)
|
||||||
|
#define T1_CTXNO_AGG (T1_CTXNO_MAG+T1_NUMCTXS_MAG)
|
||||||
|
#define T1_CTXNO_UNI (T1_CTXNO_AGG+T1_NUMCTXS_AGG)
|
||||||
|
#define T1_NUMCTXS (T1_CTXNO_UNI+T1_NUMCTXS_UNI)
|
||||||
|
|
||||||
|
#define T1_NMSEDEC_FRACBITS (T1_NMSEDEC_BITS-1)
|
||||||
|
|
||||||
|
#define T1_TYPE_MQ 0 /**< Normal coding using entropy coder */
|
||||||
|
#define T1_TYPE_RAW 1 /**< No encoding the information is store under raw format in codestream (mode switch RAW)*/
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
typedef short flag_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tier-1 coding (coding of code-block coefficients)
|
||||||
|
*/
|
||||||
|
typedef struct opj_t1 {
|
||||||
|
/** codec context */
|
||||||
|
opj_common_ptr cinfo;
|
||||||
|
|
||||||
|
/** MQC component */
|
||||||
|
opj_mqc_t *mqc;
|
||||||
|
/** RAW component */
|
||||||
|
opj_raw_t *raw;
|
||||||
|
|
||||||
|
int *data;
|
||||||
|
flag_t *flags;
|
||||||
|
int w;
|
||||||
|
int h;
|
||||||
|
int datasize;
|
||||||
|
int flagssize;
|
||||||
|
int flags_stride;
|
||||||
|
} opj_t1_t;
|
||||||
|
|
||||||
|
#define MACRO_t1_flags(x,y) t1->flags[((x)*(t1->flags_stride))+(y)]
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Create a new T1 handle
|
||||||
|
and initialize the look-up tables of the Tier-1 coder/decoder
|
||||||
|
@return Returns a new T1 handle if successful, returns NULL otherwise
|
||||||
|
@see t1_init_luts
|
||||||
|
*/
|
||||||
|
opj_t1_t* t1_create(opj_common_ptr cinfo);
|
||||||
|
/**
|
||||||
|
Destroy a previously created T1 handle
|
||||||
|
@param t1 T1 handle to destroy
|
||||||
|
*/
|
||||||
|
void t1_destroy(opj_t1_t *t1);
|
||||||
|
/**
|
||||||
|
Encode the code-blocks of a tile
|
||||||
|
@param t1 T1 handle
|
||||||
|
@param tile The tile to encode
|
||||||
|
@param tcp Tile coding parameters
|
||||||
|
*/
|
||||||
|
void t1_encode_cblks(opj_t1_t *t1, opj_tcd_tile_t *tile, opj_tcp_t *tcp);
|
||||||
|
/**
|
||||||
|
Decode the code-blocks of a tile
|
||||||
|
@param t1 T1 handle
|
||||||
|
@param tile The tile to decode
|
||||||
|
@param tcp Tile coding parameters
|
||||||
|
*/
|
||||||
|
void t1_decode_cblks(opj_t1_t* t1, opj_tcd_tilecomp_t* tilec, opj_tccp_t* tccp);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __T1_H */
|
||||||
275
extern/libopenjpeg/t1_generate_luts.c
vendored
Normal file
275
extern/libopenjpeg/t1_generate_luts.c
vendored
Normal file
@@ -0,0 +1,275 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* Copyright (c) 2007, Callum Lerwick <seg@haxxed.com>
|
||||||
|
* 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"
|
||||||
|
#include <math.h>
|
||||||
|
|
||||||
|
static int t1_init_ctxno_zc(int f, int orient) {
|
||||||
|
int h, v, d, n, t, hv;
|
||||||
|
n = 0;
|
||||||
|
h = ((f & T1_SIG_W) != 0) + ((f & T1_SIG_E) != 0);
|
||||||
|
v = ((f & T1_SIG_N) != 0) + ((f & T1_SIG_S) != 0);
|
||||||
|
d = ((f & T1_SIG_NW) != 0) + ((f & T1_SIG_NE) != 0) + ((f & T1_SIG_SE) != 0) + ((f & T1_SIG_SW) != 0);
|
||||||
|
|
||||||
|
switch (orient) {
|
||||||
|
case 2:
|
||||||
|
t = h;
|
||||||
|
h = v;
|
||||||
|
v = t;
|
||||||
|
case 0:
|
||||||
|
case 1:
|
||||||
|
if (!h) {
|
||||||
|
if (!v) {
|
||||||
|
if (!d)
|
||||||
|
n = 0;
|
||||||
|
else if (d == 1)
|
||||||
|
n = 1;
|
||||||
|
else
|
||||||
|
n = 2;
|
||||||
|
} else if (v == 1) {
|
||||||
|
n = 3;
|
||||||
|
} else {
|
||||||
|
n = 4;
|
||||||
|
}
|
||||||
|
} else if (h == 1) {
|
||||||
|
if (!v) {
|
||||||
|
if (!d)
|
||||||
|
n = 5;
|
||||||
|
else
|
||||||
|
n = 6;
|
||||||
|
} else {
|
||||||
|
n = 7;
|
||||||
|
}
|
||||||
|
} else
|
||||||
|
n = 8;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
hv = h + v;
|
||||||
|
if (!d) {
|
||||||
|
if (!hv) {
|
||||||
|
n = 0;
|
||||||
|
} else if (hv == 1) {
|
||||||
|
n = 1;
|
||||||
|
} else {
|
||||||
|
n = 2;
|
||||||
|
}
|
||||||
|
} else if (d == 1) {
|
||||||
|
if (!hv) {
|
||||||
|
n = 3;
|
||||||
|
} else if (hv == 1) {
|
||||||
|
n = 4;
|
||||||
|
} else {
|
||||||
|
n = 5;
|
||||||
|
}
|
||||||
|
} else if (d == 2) {
|
||||||
|
if (!hv) {
|
||||||
|
n = 6;
|
||||||
|
} else {
|
||||||
|
n = 7;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n = 8;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (T1_CTXNO_ZC + n);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int t1_init_ctxno_sc(int f) {
|
||||||
|
int hc, vc, n;
|
||||||
|
n = 0;
|
||||||
|
|
||||||
|
hc = int_min(((f & (T1_SIG_E | T1_SGN_E)) ==
|
||||||
|
T1_SIG_E) + ((f & (T1_SIG_W | T1_SGN_W)) == T1_SIG_W),
|
||||||
|
1) - int_min(((f & (T1_SIG_E | T1_SGN_E)) ==
|
||||||
|
(T1_SIG_E | T1_SGN_E)) +
|
||||||
|
((f & (T1_SIG_W | T1_SGN_W)) ==
|
||||||
|
(T1_SIG_W | T1_SGN_W)), 1);
|
||||||
|
|
||||||
|
vc = int_min(((f & (T1_SIG_N | T1_SGN_N)) ==
|
||||||
|
T1_SIG_N) + ((f & (T1_SIG_S | T1_SGN_S)) == T1_SIG_S),
|
||||||
|
1) - int_min(((f & (T1_SIG_N | T1_SGN_N)) ==
|
||||||
|
(T1_SIG_N | T1_SGN_N)) +
|
||||||
|
((f & (T1_SIG_S | T1_SGN_S)) ==
|
||||||
|
(T1_SIG_S | T1_SGN_S)), 1);
|
||||||
|
|
||||||
|
if (hc < 0) {
|
||||||
|
hc = -hc;
|
||||||
|
vc = -vc;
|
||||||
|
}
|
||||||
|
if (!hc) {
|
||||||
|
if (vc == -1)
|
||||||
|
n = 1;
|
||||||
|
else if (!vc)
|
||||||
|
n = 0;
|
||||||
|
else
|
||||||
|
n = 1;
|
||||||
|
} else if (hc == 1) {
|
||||||
|
if (vc == -1)
|
||||||
|
n = 2;
|
||||||
|
else if (!vc)
|
||||||
|
n = 3;
|
||||||
|
else
|
||||||
|
n = 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (T1_CTXNO_SC + n);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int t1_init_spb(int f) {
|
||||||
|
int hc, vc, n;
|
||||||
|
|
||||||
|
hc = int_min(((f & (T1_SIG_E | T1_SGN_E)) ==
|
||||||
|
T1_SIG_E) + ((f & (T1_SIG_W | T1_SGN_W)) == T1_SIG_W),
|
||||||
|
1) - int_min(((f & (T1_SIG_E | T1_SGN_E)) ==
|
||||||
|
(T1_SIG_E | T1_SGN_E)) +
|
||||||
|
((f & (T1_SIG_W | T1_SGN_W)) ==
|
||||||
|
(T1_SIG_W | T1_SGN_W)), 1);
|
||||||
|
|
||||||
|
vc = int_min(((f & (T1_SIG_N | T1_SGN_N)) ==
|
||||||
|
T1_SIG_N) + ((f & (T1_SIG_S | T1_SGN_S)) == T1_SIG_S),
|
||||||
|
1) - int_min(((f & (T1_SIG_N | T1_SGN_N)) ==
|
||||||
|
(T1_SIG_N | T1_SGN_N)) +
|
||||||
|
((f & (T1_SIG_S | T1_SGN_S)) ==
|
||||||
|
(T1_SIG_S | T1_SGN_S)), 1);
|
||||||
|
|
||||||
|
if (!hc && !vc)
|
||||||
|
n = 0;
|
||||||
|
else
|
||||||
|
n = (!(hc > 0 || (!hc && vc > 0)));
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
void dump_array16(int array[],int size){
|
||||||
|
int i;
|
||||||
|
--size;
|
||||||
|
for (i = 0; i < size; ++i) {
|
||||||
|
printf("0x%04x, ", array[i]);
|
||||||
|
if(!((i+1)&0x7))
|
||||||
|
printf("\n ");
|
||||||
|
}
|
||||||
|
printf("0x%04x\n};\n\n", array[size]);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(){
|
||||||
|
int i, j;
|
||||||
|
double u, v, t;
|
||||||
|
|
||||||
|
int lut_ctxno_zc[1024];
|
||||||
|
int lut_nmsedec_sig[1 << T1_NMSEDEC_BITS];
|
||||||
|
int lut_nmsedec_sig0[1 << T1_NMSEDEC_BITS];
|
||||||
|
int lut_nmsedec_ref[1 << T1_NMSEDEC_BITS];
|
||||||
|
int lut_nmsedec_ref0[1 << T1_NMSEDEC_BITS];
|
||||||
|
|
||||||
|
printf("/* This file was automatically generated by t1_generate_luts.c */\n\n");
|
||||||
|
|
||||||
|
// lut_ctxno_zc
|
||||||
|
for (j = 0; j < 4; ++j) {
|
||||||
|
for (i = 0; i < 256; ++i) {
|
||||||
|
int orient = j;
|
||||||
|
if (orient == 2) {
|
||||||
|
orient = 1;
|
||||||
|
} else if (orient == 1) {
|
||||||
|
orient = 2;
|
||||||
|
}
|
||||||
|
lut_ctxno_zc[(orient << 8) | i] = t1_init_ctxno_zc(i, j);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("static char lut_ctxno_zc[1024] = {\n ");
|
||||||
|
for (i = 0; i < 1023; ++i) {
|
||||||
|
printf("%i, ", lut_ctxno_zc[i]);
|
||||||
|
if(!((i+1)&0x1f))
|
||||||
|
printf("\n ");
|
||||||
|
}
|
||||||
|
printf("%i\n};\n\n", lut_ctxno_zc[1023]);
|
||||||
|
|
||||||
|
// lut_ctxno_sc
|
||||||
|
printf("static char lut_ctxno_sc[256] = {\n ");
|
||||||
|
for (i = 0; i < 255; ++i) {
|
||||||
|
printf("0x%x, ", t1_init_ctxno_sc(i << 4));
|
||||||
|
if(!((i+1)&0xf))
|
||||||
|
printf("\n ");
|
||||||
|
}
|
||||||
|
printf("0x%x\n};\n\n", t1_init_ctxno_sc(255 << 4));
|
||||||
|
|
||||||
|
// lut_spb
|
||||||
|
printf("static char lut_spb[256] = {\n ");
|
||||||
|
for (i = 0; i < 255; ++i) {
|
||||||
|
printf("%i, ", t1_init_spb(i << 4));
|
||||||
|
if(!((i+1)&0x1f))
|
||||||
|
printf("\n ");
|
||||||
|
}
|
||||||
|
printf("%i\n};\n\n", t1_init_spb(255 << 4));
|
||||||
|
|
||||||
|
/* FIXME FIXME FIXME */
|
||||||
|
/* fprintf(stdout,"nmsedec luts:\n"); */
|
||||||
|
for (i = 0; i < (1 << T1_NMSEDEC_BITS); ++i) {
|
||||||
|
t = i / pow(2, T1_NMSEDEC_FRACBITS);
|
||||||
|
u = t;
|
||||||
|
v = t - 1.5;
|
||||||
|
lut_nmsedec_sig[i] =
|
||||||
|
int_max(0,
|
||||||
|
(int) (floor((u * u - v * v) * pow(2, T1_NMSEDEC_FRACBITS) + 0.5) / pow(2, T1_NMSEDEC_FRACBITS) * 8192.0));
|
||||||
|
lut_nmsedec_sig0[i] =
|
||||||
|
int_max(0,
|
||||||
|
(int) (floor((u * u) * pow(2, T1_NMSEDEC_FRACBITS) + 0.5) / pow(2, T1_NMSEDEC_FRACBITS) * 8192.0));
|
||||||
|
u = t - 1.0;
|
||||||
|
if (i & (1 << (T1_NMSEDEC_BITS - 1))) {
|
||||||
|
v = t - 1.5;
|
||||||
|
} else {
|
||||||
|
v = t - 0.5;
|
||||||
|
}
|
||||||
|
lut_nmsedec_ref[i] =
|
||||||
|
int_max(0,
|
||||||
|
(int) (floor((u * u - v * v) * pow(2, T1_NMSEDEC_FRACBITS) + 0.5) / pow(2, T1_NMSEDEC_FRACBITS) * 8192.0));
|
||||||
|
lut_nmsedec_ref0[i] =
|
||||||
|
int_max(0,
|
||||||
|
(int) (floor((u * u) * pow(2, T1_NMSEDEC_FRACBITS) + 0.5) / pow(2, T1_NMSEDEC_FRACBITS) * 8192.0));
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("static short lut_nmsedec_sig[1 << T1_NMSEDEC_BITS] = {\n ");
|
||||||
|
dump_array16(&lut_nmsedec_sig, 1 << T1_NMSEDEC_BITS);
|
||||||
|
|
||||||
|
printf("static short lut_nmsedec_sig0[1 << T1_NMSEDEC_BITS] = {\n ");
|
||||||
|
dump_array16(&lut_nmsedec_sig0, 1 << T1_NMSEDEC_BITS);
|
||||||
|
|
||||||
|
printf("static short lut_nmsedec_ref[1 << T1_NMSEDEC_BITS] = {\n ");
|
||||||
|
dump_array16(&lut_nmsedec_ref, 1 << T1_NMSEDEC_BITS);
|
||||||
|
|
||||||
|
printf("static short lut_nmsedec_ref0[1 << T1_NMSEDEC_BITS] = {\n ");
|
||||||
|
dump_array16(&lut_nmsedec_ref0, 1 << T1_NMSEDEC_BITS);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
143
extern/libopenjpeg/t1_luts.h
vendored
Normal file
143
extern/libopenjpeg/t1_luts.h
vendored
Normal file
@@ -0,0 +1,143 @@
|
|||||||
|
/* This file was automatically generated by t1_generate_luts.c */
|
||||||
|
|
||||||
|
static char lut_ctxno_zc[1024] = {
|
||||||
|
0, 1, 1, 2, 1, 2, 2, 2, 1, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
|
||||||
|
5, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4,
|
||||||
|
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
5, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||||||
|
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||||||
|
0, 1, 1, 2, 1, 2, 2, 2, 1, 2, 2, 2, 2, 2, 2, 2, 5, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6,
|
||||||
|
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
5, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||||||
|
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||||||
|
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||||||
|
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||||||
|
0, 1, 1, 2, 1, 2, 2, 2, 1, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
|
||||||
|
5, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4,
|
||||||
|
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
5, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||||||
|
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||||||
|
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||||||
|
0, 3, 3, 6, 3, 6, 6, 8, 3, 6, 6, 8, 6, 8, 8, 8, 1, 4, 4, 7, 4, 7, 7, 8, 4, 7, 7, 8, 7, 8, 8, 8,
|
||||||
|
1, 4, 4, 7, 4, 7, 7, 8, 4, 7, 7, 8, 7, 8, 8, 8, 2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8,
|
||||||
|
1, 4, 4, 7, 4, 7, 7, 8, 4, 7, 7, 8, 7, 8, 8, 8, 2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8,
|
||||||
|
2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8, 2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8,
|
||||||
|
1, 4, 4, 7, 4, 7, 7, 8, 4, 7, 7, 8, 7, 8, 8, 8, 2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8,
|
||||||
|
2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8, 2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8,
|
||||||
|
2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8, 2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8,
|
||||||
|
2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8, 2, 5, 5, 7, 5, 7, 7, 8, 5, 7, 7, 8, 7, 8, 8, 8
|
||||||
|
};
|
||||||
|
|
||||||
|
static char lut_ctxno_sc[256] = {
|
||||||
|
0x9, 0xa, 0xc, 0xd, 0xa, 0xa, 0xd, 0xd, 0xc, 0xd, 0xc, 0xd, 0xd, 0xd, 0xd, 0xd,
|
||||||
|
0x9, 0xa, 0xc, 0xb, 0xa, 0x9, 0xd, 0xc, 0xc, 0xb, 0xc, 0xb, 0xd, 0xc, 0xd, 0xc,
|
||||||
|
0x9, 0xa, 0xc, 0xb, 0xa, 0xa, 0xb, 0xb, 0xc, 0xd, 0x9, 0xa, 0xd, 0xd, 0xa, 0xa,
|
||||||
|
0x9, 0xa, 0xc, 0xd, 0xa, 0x9, 0xb, 0xc, 0xc, 0xb, 0x9, 0xa, 0xd, 0xc, 0xa, 0x9,
|
||||||
|
0x9, 0xa, 0xc, 0xd, 0xa, 0x9, 0xb, 0xc, 0xc, 0xd, 0xc, 0xd, 0xb, 0xc, 0xb, 0xc,
|
||||||
|
0x9, 0xa, 0xc, 0xb, 0xa, 0xa, 0xb, 0xb, 0xc, 0xb, 0xc, 0xb, 0xb, 0xb, 0xb, 0xb,
|
||||||
|
0x9, 0xa, 0xc, 0xb, 0xa, 0x9, 0xd, 0xc, 0xc, 0xd, 0x9, 0xa, 0xb, 0xc, 0xa, 0x9,
|
||||||
|
0x9, 0xa, 0xc, 0xd, 0xa, 0xa, 0xd, 0xd, 0xc, 0xb, 0x9, 0xa, 0xb, 0xb, 0xa, 0xa,
|
||||||
|
0x9, 0xa, 0xc, 0xd, 0xa, 0xa, 0xd, 0xd, 0xc, 0xb, 0x9, 0xa, 0xb, 0xb, 0xa, 0xa,
|
||||||
|
0x9, 0xa, 0xc, 0xb, 0xa, 0x9, 0xd, 0xc, 0xc, 0xd, 0x9, 0xa, 0xb, 0xc, 0xa, 0x9,
|
||||||
|
0x9, 0xa, 0xc, 0xb, 0xa, 0xa, 0xb, 0xb, 0xc, 0xb, 0xc, 0xb, 0xb, 0xb, 0xb, 0xb,
|
||||||
|
0x9, 0xa, 0xc, 0xd, 0xa, 0x9, 0xb, 0xc, 0xc, 0xd, 0xc, 0xd, 0xb, 0xc, 0xb, 0xc,
|
||||||
|
0x9, 0xa, 0xc, 0xd, 0xa, 0x9, 0xb, 0xc, 0xc, 0xb, 0x9, 0xa, 0xd, 0xc, 0xa, 0x9,
|
||||||
|
0x9, 0xa, 0xc, 0xb, 0xa, 0xa, 0xb, 0xb, 0xc, 0xd, 0x9, 0xa, 0xd, 0xd, 0xa, 0xa,
|
||||||
|
0x9, 0xa, 0xc, 0xb, 0xa, 0x9, 0xd, 0xc, 0xc, 0xb, 0xc, 0xb, 0xd, 0xc, 0xd, 0xc,
|
||||||
|
0x9, 0xa, 0xc, 0xd, 0xa, 0xa, 0xd, 0xd, 0xc, 0xd, 0xc, 0xd, 0xd, 0xd, 0xd, 0xd
|
||||||
|
};
|
||||||
|
|
||||||
|
static char lut_spb[256] = {
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0, 0, 1, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 1, 1, 1, 0, 1, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 1, 0, 0, 1, 1,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 1, 0, 1, 1, 1, 0, 0,
|
||||||
|
0, 0, 1, 1, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
0, 0, 0, 0, 1, 0, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 0, 0, 1, 1, 0, 0, 1, 1, 0, 1, 1, 1, 1, 1,
|
||||||
|
0, 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
|
||||||
|
};
|
||||||
|
|
||||||
|
static short lut_nmsedec_sig[1 << T1_NMSEDEC_BITS] = {
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0180, 0x0300, 0x0480, 0x0600, 0x0780, 0x0900, 0x0a80,
|
||||||
|
0x0c00, 0x0d80, 0x0f00, 0x1080, 0x1200, 0x1380, 0x1500, 0x1680,
|
||||||
|
0x1800, 0x1980, 0x1b00, 0x1c80, 0x1e00, 0x1f80, 0x2100, 0x2280,
|
||||||
|
0x2400, 0x2580, 0x2700, 0x2880, 0x2a00, 0x2b80, 0x2d00, 0x2e80,
|
||||||
|
0x3000, 0x3180, 0x3300, 0x3480, 0x3600, 0x3780, 0x3900, 0x3a80,
|
||||||
|
0x3c00, 0x3d80, 0x3f00, 0x4080, 0x4200, 0x4380, 0x4500, 0x4680,
|
||||||
|
0x4800, 0x4980, 0x4b00, 0x4c80, 0x4e00, 0x4f80, 0x5100, 0x5280,
|
||||||
|
0x5400, 0x5580, 0x5700, 0x5880, 0x5a00, 0x5b80, 0x5d00, 0x5e80,
|
||||||
|
0x6000, 0x6180, 0x6300, 0x6480, 0x6600, 0x6780, 0x6900, 0x6a80,
|
||||||
|
0x6c00, 0x6d80, 0x6f00, 0x7080, 0x7200, 0x7380, 0x7500, 0x7680
|
||||||
|
};
|
||||||
|
|
||||||
|
static short lut_nmsedec_sig0[1 << T1_NMSEDEC_BITS] = {
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0080, 0x0080,
|
||||||
|
0x0080, 0x0080, 0x0100, 0x0100, 0x0100, 0x0180, 0x0180, 0x0200,
|
||||||
|
0x0200, 0x0280, 0x0280, 0x0300, 0x0300, 0x0380, 0x0400, 0x0400,
|
||||||
|
0x0480, 0x0500, 0x0580, 0x0580, 0x0600, 0x0680, 0x0700, 0x0780,
|
||||||
|
0x0800, 0x0880, 0x0900, 0x0980, 0x0a00, 0x0a80, 0x0b80, 0x0c00,
|
||||||
|
0x0c80, 0x0d00, 0x0e00, 0x0e80, 0x0f00, 0x1000, 0x1080, 0x1180,
|
||||||
|
0x1200, 0x1300, 0x1380, 0x1480, 0x1500, 0x1600, 0x1700, 0x1780,
|
||||||
|
0x1880, 0x1980, 0x1a80, 0x1b00, 0x1c00, 0x1d00, 0x1e00, 0x1f00,
|
||||||
|
0x2000, 0x2100, 0x2200, 0x2300, 0x2400, 0x2500, 0x2680, 0x2780,
|
||||||
|
0x2880, 0x2980, 0x2b00, 0x2c00, 0x2d00, 0x2e80, 0x2f80, 0x3100,
|
||||||
|
0x3200, 0x3380, 0x3480, 0x3600, 0x3700, 0x3880, 0x3a00, 0x3b00,
|
||||||
|
0x3c80, 0x3e00, 0x3f80, 0x4080, 0x4200, 0x4380, 0x4500, 0x4680,
|
||||||
|
0x4800, 0x4980, 0x4b00, 0x4c80, 0x4e00, 0x4f80, 0x5180, 0x5300,
|
||||||
|
0x5480, 0x5600, 0x5800, 0x5980, 0x5b00, 0x5d00, 0x5e80, 0x6080,
|
||||||
|
0x6200, 0x6400, 0x6580, 0x6780, 0x6900, 0x6b00, 0x6d00, 0x6e80,
|
||||||
|
0x7080, 0x7280, 0x7480, 0x7600, 0x7800, 0x7a00, 0x7c00, 0x7e00
|
||||||
|
};
|
||||||
|
|
||||||
|
static short lut_nmsedec_ref[1 << T1_NMSEDEC_BITS] = {
|
||||||
|
0x1800, 0x1780, 0x1700, 0x1680, 0x1600, 0x1580, 0x1500, 0x1480,
|
||||||
|
0x1400, 0x1380, 0x1300, 0x1280, 0x1200, 0x1180, 0x1100, 0x1080,
|
||||||
|
0x1000, 0x0f80, 0x0f00, 0x0e80, 0x0e00, 0x0d80, 0x0d00, 0x0c80,
|
||||||
|
0x0c00, 0x0b80, 0x0b00, 0x0a80, 0x0a00, 0x0980, 0x0900, 0x0880,
|
||||||
|
0x0800, 0x0780, 0x0700, 0x0680, 0x0600, 0x0580, 0x0500, 0x0480,
|
||||||
|
0x0400, 0x0380, 0x0300, 0x0280, 0x0200, 0x0180, 0x0100, 0x0080,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0080, 0x0100, 0x0180, 0x0200, 0x0280, 0x0300, 0x0380,
|
||||||
|
0x0400, 0x0480, 0x0500, 0x0580, 0x0600, 0x0680, 0x0700, 0x0780,
|
||||||
|
0x0800, 0x0880, 0x0900, 0x0980, 0x0a00, 0x0a80, 0x0b00, 0x0b80,
|
||||||
|
0x0c00, 0x0c80, 0x0d00, 0x0d80, 0x0e00, 0x0e80, 0x0f00, 0x0f80,
|
||||||
|
0x1000, 0x1080, 0x1100, 0x1180, 0x1200, 0x1280, 0x1300, 0x1380,
|
||||||
|
0x1400, 0x1480, 0x1500, 0x1580, 0x1600, 0x1680, 0x1700, 0x1780
|
||||||
|
};
|
||||||
|
|
||||||
|
static short lut_nmsedec_ref0[1 << T1_NMSEDEC_BITS] = {
|
||||||
|
0x2000, 0x1f00, 0x1e00, 0x1d00, 0x1c00, 0x1b00, 0x1a80, 0x1980,
|
||||||
|
0x1880, 0x1780, 0x1700, 0x1600, 0x1500, 0x1480, 0x1380, 0x1300,
|
||||||
|
0x1200, 0x1180, 0x1080, 0x1000, 0x0f00, 0x0e80, 0x0e00, 0x0d00,
|
||||||
|
0x0c80, 0x0c00, 0x0b80, 0x0a80, 0x0a00, 0x0980, 0x0900, 0x0880,
|
||||||
|
0x0800, 0x0780, 0x0700, 0x0680, 0x0600, 0x0580, 0x0580, 0x0500,
|
||||||
|
0x0480, 0x0400, 0x0400, 0x0380, 0x0300, 0x0300, 0x0280, 0x0280,
|
||||||
|
0x0200, 0x0200, 0x0180, 0x0180, 0x0100, 0x0100, 0x0100, 0x0080,
|
||||||
|
0x0080, 0x0080, 0x0080, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
|
||||||
|
0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0080, 0x0080,
|
||||||
|
0x0080, 0x0080, 0x0100, 0x0100, 0x0100, 0x0180, 0x0180, 0x0200,
|
||||||
|
0x0200, 0x0280, 0x0280, 0x0300, 0x0300, 0x0380, 0x0400, 0x0400,
|
||||||
|
0x0480, 0x0500, 0x0580, 0x0580, 0x0600, 0x0680, 0x0700, 0x0780,
|
||||||
|
0x0800, 0x0880, 0x0900, 0x0980, 0x0a00, 0x0a80, 0x0b80, 0x0c00,
|
||||||
|
0x0c80, 0x0d00, 0x0e00, 0x0e80, 0x0f00, 0x1000, 0x1080, 0x1180,
|
||||||
|
0x1200, 0x1300, 0x1380, 0x1480, 0x1500, 0x1600, 0x1700, 0x1780,
|
||||||
|
0x1880, 0x1980, 0x1a80, 0x1b00, 0x1c00, 0x1d00, 0x1e00, 0x1f00
|
||||||
|
};
|
||||||
|
|
||||||
787
extern/libopenjpeg/t2.c
vendored
Normal file
787
extern/libopenjpeg/t2.c
vendored
Normal file
@@ -0,0 +1,787 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/** @defgroup T2 T2 - Implementation of a tier-2 coding */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/** @name Local static functions */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
static void t2_putcommacode(opj_bio_t *bio, int n);
|
||||||
|
static int t2_getcommacode(opj_bio_t *bio);
|
||||||
|
/**
|
||||||
|
Variable length code for signalling delta Zil (truncation point)
|
||||||
|
@param bio Bit Input/Output component
|
||||||
|
@param n delta Zil
|
||||||
|
*/
|
||||||
|
static void t2_putnumpasses(opj_bio_t *bio, int n);
|
||||||
|
static int t2_getnumpasses(opj_bio_t *bio);
|
||||||
|
/**
|
||||||
|
Encode a packet of a tile to a destination buffer
|
||||||
|
@param tile Tile for which to write the packets
|
||||||
|
@param tcp Tile coding parameters
|
||||||
|
@param pi Packet identity
|
||||||
|
@param dest Destination buffer
|
||||||
|
@param len Length of the destination buffer
|
||||||
|
@param cstr_info Codestream information structure
|
||||||
|
@param tileno Number of the tile encoded
|
||||||
|
@return
|
||||||
|
*/
|
||||||
|
static int t2_encode_packet(opj_tcd_tile_t *tile, opj_tcp_t *tcp, opj_pi_iterator_t *pi, unsigned char *dest, int len, opj_codestream_info_t *cstr_info, int tileno);
|
||||||
|
/**
|
||||||
|
@param seg
|
||||||
|
@param cblksty
|
||||||
|
@param first
|
||||||
|
*/
|
||||||
|
static void t2_init_seg(opj_tcd_cblk_dec_t* cblk, int index, int cblksty, int first);
|
||||||
|
/**
|
||||||
|
Decode a packet of a tile from a source buffer
|
||||||
|
@param t2 T2 handle
|
||||||
|
@param src Source buffer
|
||||||
|
@param len Length of the source buffer
|
||||||
|
@param tile Tile for which to write the packets
|
||||||
|
@param tcp Tile coding parameters
|
||||||
|
@param pi Packet identity
|
||||||
|
@return
|
||||||
|
*/
|
||||||
|
static int t2_decode_packet(opj_t2_t* t2, unsigned char *src, int len, opj_tcd_tile_t *tile,
|
||||||
|
opj_tcp_t *tcp, opj_pi_iterator_t *pi, opj_packet_info_t *pack_info);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/* #define RESTART 0x04 */
|
||||||
|
|
||||||
|
static void t2_putcommacode(opj_bio_t *bio, int n) {
|
||||||
|
while (--n >= 0) {
|
||||||
|
bio_write(bio, 1, 1);
|
||||||
|
}
|
||||||
|
bio_write(bio, 0, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int t2_getcommacode(opj_bio_t *bio) {
|
||||||
|
int n;
|
||||||
|
for (n = 0; bio_read(bio, 1); n++) {
|
||||||
|
;
|
||||||
|
}
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void t2_putnumpasses(opj_bio_t *bio, int n) {
|
||||||
|
if (n == 1) {
|
||||||
|
bio_write(bio, 0, 1);
|
||||||
|
} else if (n == 2) {
|
||||||
|
bio_write(bio, 2, 2);
|
||||||
|
} else if (n <= 5) {
|
||||||
|
bio_write(bio, 0xc | (n - 3), 4);
|
||||||
|
} else if (n <= 36) {
|
||||||
|
bio_write(bio, 0x1e0 | (n - 6), 9);
|
||||||
|
} else if (n <= 164) {
|
||||||
|
bio_write(bio, 0xff80 | (n - 37), 16);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int t2_getnumpasses(opj_bio_t *bio) {
|
||||||
|
int n;
|
||||||
|
if (!bio_read(bio, 1))
|
||||||
|
return 1;
|
||||||
|
if (!bio_read(bio, 1))
|
||||||
|
return 2;
|
||||||
|
if ((n = bio_read(bio, 2)) != 3)
|
||||||
|
return (3 + n);
|
||||||
|
if ((n = bio_read(bio, 5)) != 31)
|
||||||
|
return (6 + n);
|
||||||
|
return (37 + bio_read(bio, 7));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int t2_encode_packet(opj_tcd_tile_t * tile, opj_tcp_t * tcp, opj_pi_iterator_t *pi, unsigned char *dest, int length, opj_codestream_info_t *cstr_info, int tileno) {
|
||||||
|
int bandno, cblkno;
|
||||||
|
unsigned char *c = dest;
|
||||||
|
|
||||||
|
int compno = pi->compno; /* component value */
|
||||||
|
int resno = pi->resno; /* resolution level value */
|
||||||
|
int precno = pi->precno; /* precinct value */
|
||||||
|
int layno = pi->layno; /* quality layer value */
|
||||||
|
|
||||||
|
opj_tcd_tilecomp_t *tilec = &tile->comps[compno];
|
||||||
|
opj_tcd_resolution_t *res = &tilec->resolutions[resno];
|
||||||
|
|
||||||
|
opj_bio_t *bio = NULL; /* BIO component */
|
||||||
|
|
||||||
|
/* <SOP 0xff91> */
|
||||||
|
if (tcp->csty & J2K_CP_CSTY_SOP) {
|
||||||
|
c[0] = 255;
|
||||||
|
c[1] = 145;
|
||||||
|
c[2] = 0;
|
||||||
|
c[3] = 4;
|
||||||
|
c[4] = (tile->packno % 65536) / 256;
|
||||||
|
c[5] = (tile->packno % 65536) % 256;
|
||||||
|
c += 6;
|
||||||
|
}
|
||||||
|
/* </SOP> */
|
||||||
|
|
||||||
|
if (!layno) {
|
||||||
|
for (bandno = 0; bandno < res->numbands; bandno++) {
|
||||||
|
opj_tcd_band_t *band = &res->bands[bandno];
|
||||||
|
opj_tcd_precinct_t *prc = &band->precincts[precno];
|
||||||
|
tgt_reset(prc->incltree);
|
||||||
|
tgt_reset(prc->imsbtree);
|
||||||
|
for (cblkno = 0; cblkno < prc->cw * prc->ch; cblkno++) {
|
||||||
|
opj_tcd_cblk_enc_t* cblk = &prc->cblks.enc[cblkno];
|
||||||
|
cblk->numpasses = 0;
|
||||||
|
tgt_setvalue(prc->imsbtree, cblkno, band->numbps - cblk->numbps);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bio = bio_create();
|
||||||
|
bio_init_enc(bio, c, length);
|
||||||
|
bio_write(bio, 1, 1); /* Empty header bit */
|
||||||
|
|
||||||
|
/* Writing Packet header */
|
||||||
|
for (bandno = 0; bandno < res->numbands; bandno++) {
|
||||||
|
opj_tcd_band_t *band = &res->bands[bandno];
|
||||||
|
opj_tcd_precinct_t *prc = &band->precincts[precno];
|
||||||
|
for (cblkno = 0; cblkno < prc->cw * prc->ch; cblkno++) {
|
||||||
|
opj_tcd_cblk_enc_t* cblk = &prc->cblks.enc[cblkno];
|
||||||
|
opj_tcd_layer_t *layer = &cblk->layers[layno];
|
||||||
|
if (!cblk->numpasses && layer->numpasses) {
|
||||||
|
tgt_setvalue(prc->incltree, cblkno, layno);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (cblkno = 0; cblkno < prc->cw * prc->ch; cblkno++) {
|
||||||
|
opj_tcd_cblk_enc_t* cblk = &prc->cblks.enc[cblkno];
|
||||||
|
opj_tcd_layer_t *layer = &cblk->layers[layno];
|
||||||
|
int increment = 0;
|
||||||
|
int nump = 0;
|
||||||
|
int len = 0, passno;
|
||||||
|
/* cblk inclusion bits */
|
||||||
|
if (!cblk->numpasses) {
|
||||||
|
tgt_encode(bio, prc->incltree, cblkno, layno + 1);
|
||||||
|
} else {
|
||||||
|
bio_write(bio, layer->numpasses != 0, 1);
|
||||||
|
}
|
||||||
|
/* if cblk not included, go to the next cblk */
|
||||||
|
if (!layer->numpasses) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
/* if first instance of cblk --> zero bit-planes information */
|
||||||
|
if (!cblk->numpasses) {
|
||||||
|
cblk->numlenbits = 3;
|
||||||
|
tgt_encode(bio, prc->imsbtree, cblkno, 999);
|
||||||
|
}
|
||||||
|
/* number of coding passes included */
|
||||||
|
t2_putnumpasses(bio, layer->numpasses);
|
||||||
|
|
||||||
|
/* computation of the increase of the length indicator and insertion in the header */
|
||||||
|
for (passno = cblk->numpasses; passno < cblk->numpasses + layer->numpasses; passno++) {
|
||||||
|
opj_tcd_pass_t *pass = &cblk->passes[passno];
|
||||||
|
nump++;
|
||||||
|
len += pass->len;
|
||||||
|
if (pass->term || passno == (cblk->numpasses + layer->numpasses) - 1) {
|
||||||
|
increment = int_max(increment, int_floorlog2(len) + 1 - (cblk->numlenbits + int_floorlog2(nump)));
|
||||||
|
len = 0;
|
||||||
|
nump = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
t2_putcommacode(bio, increment);
|
||||||
|
|
||||||
|
/* computation of the new Length indicator */
|
||||||
|
cblk->numlenbits += increment;
|
||||||
|
|
||||||
|
/* insertion of the codeword segment length */
|
||||||
|
for (passno = cblk->numpasses; passno < cblk->numpasses + layer->numpasses; passno++) {
|
||||||
|
opj_tcd_pass_t *pass = &cblk->passes[passno];
|
||||||
|
nump++;
|
||||||
|
len += pass->len;
|
||||||
|
if (pass->term || passno == (cblk->numpasses + layer->numpasses) - 1) {
|
||||||
|
bio_write(bio, len, cblk->numlenbits + int_floorlog2(nump));
|
||||||
|
len = 0;
|
||||||
|
nump = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bio_flush(bio)) {
|
||||||
|
bio_destroy(bio);
|
||||||
|
return -999; /* modified to eliminate longjmp !! */
|
||||||
|
}
|
||||||
|
|
||||||
|
c += bio_numbytes(bio);
|
||||||
|
bio_destroy(bio);
|
||||||
|
|
||||||
|
/* <EPH 0xff92> */
|
||||||
|
if (tcp->csty & J2K_CP_CSTY_EPH) {
|
||||||
|
c[0] = 255;
|
||||||
|
c[1] = 146;
|
||||||
|
c += 2;
|
||||||
|
}
|
||||||
|
/* </EPH> */
|
||||||
|
|
||||||
|
/* << INDEX */
|
||||||
|
// End of packet header position. Currently only represents the distance to start of packet
|
||||||
|
// Will be updated later by incrementing with packet start value
|
||||||
|
if(cstr_info && cstr_info->index_write) {
|
||||||
|
opj_packet_info_t *info_PK = &cstr_info->tile[tileno].packet[cstr_info->packno];
|
||||||
|
info_PK->end_ph_pos = (int)(c - dest);
|
||||||
|
}
|
||||||
|
/* INDEX >> */
|
||||||
|
|
||||||
|
/* Writing the packet body */
|
||||||
|
|
||||||
|
for (bandno = 0; bandno < res->numbands; bandno++) {
|
||||||
|
opj_tcd_band_t *band = &res->bands[bandno];
|
||||||
|
opj_tcd_precinct_t *prc = &band->precincts[precno];
|
||||||
|
for (cblkno = 0; cblkno < prc->cw * prc->ch; cblkno++) {
|
||||||
|
opj_tcd_cblk_enc_t* cblk = &prc->cblks.enc[cblkno];
|
||||||
|
opj_tcd_layer_t *layer = &cblk->layers[layno];
|
||||||
|
if (!layer->numpasses) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (c + layer->len > dest + length) {
|
||||||
|
return -999;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(c, layer->data, layer->len);
|
||||||
|
cblk->numpasses += layer->numpasses;
|
||||||
|
c += layer->len;
|
||||||
|
/* << INDEX */
|
||||||
|
if(cstr_info && cstr_info->index_write) {
|
||||||
|
opj_packet_info_t *info_PK = &cstr_info->tile[tileno].packet[cstr_info->packno];
|
||||||
|
info_PK->disto += layer->disto;
|
||||||
|
if (cstr_info->D_max < info_PK->disto) {
|
||||||
|
cstr_info->D_max = info_PK->disto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* INDEX >> */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (c - dest);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void t2_init_seg(opj_tcd_cblk_dec_t* cblk, int index, int cblksty, int first) {
|
||||||
|
opj_tcd_seg_t* seg;
|
||||||
|
cblk->segs = (opj_tcd_seg_t*) opj_realloc(cblk->segs, (index + 1) * sizeof(opj_tcd_seg_t));
|
||||||
|
seg = &cblk->segs[index];
|
||||||
|
seg->data = NULL;
|
||||||
|
seg->dataindex = 0;
|
||||||
|
seg->numpasses = 0;
|
||||||
|
seg->len = 0;
|
||||||
|
if (cblksty & J2K_CCP_CBLKSTY_TERMALL) {
|
||||||
|
seg->maxpasses = 1;
|
||||||
|
}
|
||||||
|
else if (cblksty & J2K_CCP_CBLKSTY_LAZY) {
|
||||||
|
if (first) {
|
||||||
|
seg->maxpasses = 10;
|
||||||
|
} else {
|
||||||
|
seg->maxpasses = (((seg - 1)->maxpasses == 1) || ((seg - 1)->maxpasses == 10)) ? 2 : 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
seg->maxpasses = 109;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int t2_decode_packet(opj_t2_t* t2, unsigned char *src, int len, opj_tcd_tile_t *tile,
|
||||||
|
opj_tcp_t *tcp, opj_pi_iterator_t *pi, opj_packet_info_t *pack_info) {
|
||||||
|
int bandno, cblkno;
|
||||||
|
unsigned char *c = src;
|
||||||
|
|
||||||
|
opj_cp_t *cp = t2->cp;
|
||||||
|
|
||||||
|
int compno = pi->compno; /* component value */
|
||||||
|
int resno = pi->resno; /* resolution level value */
|
||||||
|
int precno = pi->precno; /* precinct value */
|
||||||
|
int layno = pi->layno; /* quality layer value */
|
||||||
|
|
||||||
|
opj_tcd_resolution_t* res = &tile->comps[compno].resolutions[resno];
|
||||||
|
|
||||||
|
unsigned char *hd = NULL;
|
||||||
|
int present;
|
||||||
|
|
||||||
|
opj_bio_t *bio = NULL; /* BIO component */
|
||||||
|
|
||||||
|
if (layno == 0) {
|
||||||
|
for (bandno = 0; bandno < res->numbands; bandno++) {
|
||||||
|
opj_tcd_band_t *band = &res->bands[bandno];
|
||||||
|
opj_tcd_precinct_t *prc = &band->precincts[precno];
|
||||||
|
|
||||||
|
if ((band->x1-band->x0 == 0)||(band->y1-band->y0 == 0)) continue;
|
||||||
|
|
||||||
|
tgt_reset(prc->incltree);
|
||||||
|
tgt_reset(prc->imsbtree);
|
||||||
|
for (cblkno = 0; cblkno < prc->cw * prc->ch; cblkno++) {
|
||||||
|
opj_tcd_cblk_dec_t* cblk = &prc->cblks.dec[cblkno];
|
||||||
|
cblk->numsegs = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* SOP markers */
|
||||||
|
|
||||||
|
if (tcp->csty & J2K_CP_CSTY_SOP) {
|
||||||
|
if ((*c) != 0xff || (*(c + 1) != 0x91)) {
|
||||||
|
opj_event_msg(t2->cinfo, EVT_WARNING, "Expected SOP marker\n");
|
||||||
|
} else {
|
||||||
|
c += 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** TODO : check the Nsop value */
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
When the marker PPT/PPM is used the packet header are store in PPT/PPM marker
|
||||||
|
This part deal with this caracteristic
|
||||||
|
step 1: Read packet header in the saved structure
|
||||||
|
step 2: Return to codestream for decoding
|
||||||
|
*/
|
||||||
|
|
||||||
|
bio = bio_create();
|
||||||
|
|
||||||
|
if (cp->ppm == 1) { /* PPM */
|
||||||
|
hd = cp->ppm_data;
|
||||||
|
bio_init_dec(bio, hd, cp->ppm_len);
|
||||||
|
} else if (tcp->ppt == 1) { /* PPT */
|
||||||
|
hd = tcp->ppt_data;
|
||||||
|
bio_init_dec(bio, hd, tcp->ppt_len);
|
||||||
|
} else { /* Normal Case */
|
||||||
|
hd = c;
|
||||||
|
bio_init_dec(bio, hd, src+len-hd);
|
||||||
|
}
|
||||||
|
|
||||||
|
present = bio_read(bio, 1);
|
||||||
|
|
||||||
|
if (!present) {
|
||||||
|
bio_inalign(bio);
|
||||||
|
hd += bio_numbytes(bio);
|
||||||
|
bio_destroy(bio);
|
||||||
|
|
||||||
|
/* EPH markers */
|
||||||
|
|
||||||
|
if (tcp->csty & J2K_CP_CSTY_EPH) {
|
||||||
|
if ((*hd) != 0xff || (*(hd + 1) != 0x92)) {
|
||||||
|
printf("Error : expected EPH marker\n");
|
||||||
|
} else {
|
||||||
|
hd += 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* << INDEX */
|
||||||
|
// End of packet header position. Currently only represents the distance to start of packet
|
||||||
|
// Will be updated later by incrementing with packet start value
|
||||||
|
if(pack_info) {
|
||||||
|
pack_info->end_ph_pos = (int)(c - src);
|
||||||
|
}
|
||||||
|
/* INDEX >> */
|
||||||
|
|
||||||
|
if (cp->ppm == 1) { /* PPM case */
|
||||||
|
cp->ppm_len += cp->ppm_data-hd;
|
||||||
|
cp->ppm_data = hd;
|
||||||
|
return (c - src);
|
||||||
|
}
|
||||||
|
if (tcp->ppt == 1) { /* PPT case */
|
||||||
|
tcp->ppt_len+=tcp->ppt_data-hd;
|
||||||
|
tcp->ppt_data = hd;
|
||||||
|
return (c - src);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (hd - src);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (bandno = 0; bandno < res->numbands; bandno++) {
|
||||||
|
opj_tcd_band_t *band = &res->bands[bandno];
|
||||||
|
opj_tcd_precinct_t *prc = &band->precincts[precno];
|
||||||
|
|
||||||
|
if ((band->x1-band->x0 == 0)||(band->y1-band->y0 == 0)) continue;
|
||||||
|
|
||||||
|
for (cblkno = 0; cblkno < prc->cw * prc->ch; cblkno++) {
|
||||||
|
int included, increment, n, segno;
|
||||||
|
opj_tcd_cblk_dec_t* cblk = &prc->cblks.dec[cblkno];
|
||||||
|
/* if cblk not yet included before --> inclusion tagtree */
|
||||||
|
if (!cblk->numsegs) {
|
||||||
|
included = tgt_decode(bio, prc->incltree, cblkno, layno + 1);
|
||||||
|
/* else one bit */
|
||||||
|
} else {
|
||||||
|
included = bio_read(bio, 1);
|
||||||
|
}
|
||||||
|
/* if cblk not included */
|
||||||
|
if (!included) {
|
||||||
|
cblk->numnewpasses = 0;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
/* if cblk not yet included --> zero-bitplane tagtree */
|
||||||
|
if (!cblk->numsegs) {
|
||||||
|
int i, numimsbs;
|
||||||
|
for (i = 0; !tgt_decode(bio, prc->imsbtree, cblkno, i); i++) {
|
||||||
|
;
|
||||||
|
}
|
||||||
|
numimsbs = i - 1;
|
||||||
|
cblk->numbps = band->numbps - numimsbs;
|
||||||
|
cblk->numlenbits = 3;
|
||||||
|
}
|
||||||
|
/* number of coding passes */
|
||||||
|
cblk->numnewpasses = t2_getnumpasses(bio);
|
||||||
|
increment = t2_getcommacode(bio);
|
||||||
|
/* length indicator increment */
|
||||||
|
cblk->numlenbits += increment;
|
||||||
|
segno = 0;
|
||||||
|
if (!cblk->numsegs) {
|
||||||
|
t2_init_seg(cblk, segno, tcp->tccps[compno].cblksty, 1);
|
||||||
|
} else {
|
||||||
|
segno = cblk->numsegs - 1;
|
||||||
|
if (cblk->segs[segno].numpasses == cblk->segs[segno].maxpasses) {
|
||||||
|
++segno;
|
||||||
|
t2_init_seg(cblk, segno, tcp->tccps[compno].cblksty, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
n = cblk->numnewpasses;
|
||||||
|
|
||||||
|
do {
|
||||||
|
cblk->segs[segno].numnewpasses = int_min(cblk->segs[segno].maxpasses - cblk->segs[segno].numpasses, n);
|
||||||
|
cblk->segs[segno].newlen = bio_read(bio, cblk->numlenbits + int_floorlog2(cblk->segs[segno].numnewpasses));
|
||||||
|
n -= cblk->segs[segno].numnewpasses;
|
||||||
|
if (n > 0) {
|
||||||
|
++segno;
|
||||||
|
t2_init_seg(cblk, segno, tcp->tccps[compno].cblksty, 0);
|
||||||
|
}
|
||||||
|
} while (n > 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bio_inalign(bio)) {
|
||||||
|
bio_destroy(bio);
|
||||||
|
return -999;
|
||||||
|
}
|
||||||
|
|
||||||
|
hd += bio_numbytes(bio);
|
||||||
|
bio_destroy(bio);
|
||||||
|
|
||||||
|
/* EPH markers */
|
||||||
|
if (tcp->csty & J2K_CP_CSTY_EPH) {
|
||||||
|
if ((*hd) != 0xff || (*(hd + 1) != 0x92)) {
|
||||||
|
opj_event_msg(t2->cinfo, EVT_ERROR, "Expected EPH marker\n");
|
||||||
|
} else {
|
||||||
|
hd += 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* << INDEX */
|
||||||
|
// End of packet header position. Currently only represents the distance to start of packet
|
||||||
|
// Will be updated later by incrementing with packet start value
|
||||||
|
if(pack_info) {
|
||||||
|
pack_info->end_ph_pos = (int)(hd - src);
|
||||||
|
}
|
||||||
|
/* INDEX >> */
|
||||||
|
|
||||||
|
if (cp->ppm==1) {
|
||||||
|
cp->ppm_len+=cp->ppm_data-hd;
|
||||||
|
cp->ppm_data = hd;
|
||||||
|
} else if (tcp->ppt == 1) {
|
||||||
|
tcp->ppt_len+=tcp->ppt_data-hd;
|
||||||
|
tcp->ppt_data = hd;
|
||||||
|
} else {
|
||||||
|
c=hd;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (bandno = 0; bandno < res->numbands; bandno++) {
|
||||||
|
opj_tcd_band_t *band = &res->bands[bandno];
|
||||||
|
opj_tcd_precinct_t *prc = &band->precincts[precno];
|
||||||
|
|
||||||
|
if ((band->x1-band->x0 == 0)||(band->y1-band->y0 == 0)) continue;
|
||||||
|
|
||||||
|
for (cblkno = 0; cblkno < prc->cw * prc->ch; cblkno++) {
|
||||||
|
opj_tcd_cblk_dec_t* cblk = &prc->cblks.dec[cblkno];
|
||||||
|
opj_tcd_seg_t *seg = NULL;
|
||||||
|
if (!cblk->numnewpasses)
|
||||||
|
continue;
|
||||||
|
if (!cblk->numsegs) {
|
||||||
|
seg = &cblk->segs[0];
|
||||||
|
cblk->numsegs++;
|
||||||
|
cblk->len = 0;
|
||||||
|
} else {
|
||||||
|
seg = &cblk->segs[cblk->numsegs - 1];
|
||||||
|
if (seg->numpasses == seg->maxpasses) {
|
||||||
|
seg++;
|
||||||
|
cblk->numsegs++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
do {
|
||||||
|
if (c + seg->newlen > src + len) {
|
||||||
|
return -999;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef USE_JPWL
|
||||||
|
/* we need here a j2k handle to verify if making a check to
|
||||||
|
the validity of cblocks parameters is selected from user (-W) */
|
||||||
|
|
||||||
|
/* let's check that we are not exceeding */
|
||||||
|
if ((cblk->len + seg->newlen) > 8192) {
|
||||||
|
opj_event_msg(t2->cinfo, EVT_WARNING,
|
||||||
|
"JPWL: segment too long (%d) for codeblock %d (p=%d, b=%d, r=%d, c=%d)\n",
|
||||||
|
seg->newlen, cblkno, precno, bandno, resno, compno);
|
||||||
|
if (!JPWL_ASSUME) {
|
||||||
|
opj_event_msg(t2->cinfo, EVT_ERROR, "JPWL: giving up\n");
|
||||||
|
return -999;
|
||||||
|
}
|
||||||
|
seg->newlen = 8192 - cblk->len;
|
||||||
|
opj_event_msg(t2->cinfo, EVT_WARNING, " - truncating segment to %d\n", seg->newlen);
|
||||||
|
break;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* USE_JPWL */
|
||||||
|
|
||||||
|
cblk->data = (unsigned char*) opj_realloc(cblk->data, (cblk->len + seg->newlen) * sizeof(unsigned char*));
|
||||||
|
memcpy(cblk->data + cblk->len, c, seg->newlen);
|
||||||
|
if (seg->numpasses == 0) {
|
||||||
|
seg->data = &cblk->data;
|
||||||
|
seg->dataindex = cblk->len;
|
||||||
|
}
|
||||||
|
c += seg->newlen;
|
||||||
|
cblk->len += seg->newlen;
|
||||||
|
seg->len += seg->newlen;
|
||||||
|
seg->numpasses += seg->numnewpasses;
|
||||||
|
cblk->numnewpasses -= seg->numnewpasses;
|
||||||
|
if (cblk->numnewpasses > 0) {
|
||||||
|
seg++;
|
||||||
|
cblk->numsegs++;
|
||||||
|
}
|
||||||
|
} while (cblk->numnewpasses > 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (c - src);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
int t2_encode_packets(opj_t2_t* t2,int tileno, opj_tcd_tile_t *tile, int maxlayers, unsigned char *dest, int len, opj_codestream_info_t *cstr_info,int tpnum, int tppos,int pino, J2K_T2_MODE t2_mode, int cur_totnum_tp){
|
||||||
|
unsigned char *c = dest;
|
||||||
|
int e = 0;
|
||||||
|
int compno;
|
||||||
|
opj_pi_iterator_t *pi = NULL;
|
||||||
|
int poc;
|
||||||
|
opj_image_t *image = t2->image;
|
||||||
|
opj_cp_t *cp = t2->cp;
|
||||||
|
opj_tcp_t *tcp = &cp->tcps[tileno];
|
||||||
|
int pocno = cp->cinema == CINEMA4K_24? 2: 1;
|
||||||
|
int maxcomp = cp->max_comp_size > 0 ? image->numcomps : 1;
|
||||||
|
|
||||||
|
pi = pi_initialise_encode(image, cp, tileno, t2_mode);
|
||||||
|
if(!pi) {
|
||||||
|
/* TODO: throw an error */
|
||||||
|
return -999;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(t2_mode == THRESH_CALC ){ /* Calculating threshold */
|
||||||
|
for(compno = 0; compno < maxcomp; compno++ ){
|
||||||
|
for(poc = 0; poc < pocno ; poc++){
|
||||||
|
int comp_len = 0;
|
||||||
|
int tpnum = compno;
|
||||||
|
if (pi_create_encode(pi, cp,tileno,poc,tpnum,tppos,t2_mode,cur_totnum_tp)) {
|
||||||
|
opj_event_msg(t2->cinfo, EVT_ERROR, "Error initializing Packet Iterator\n");
|
||||||
|
return -999;
|
||||||
|
}
|
||||||
|
while (pi_next(&pi[poc])) {
|
||||||
|
if (pi[poc].layno < maxlayers) {
|
||||||
|
e = t2_encode_packet(tile, &cp->tcps[tileno], &pi[poc], c, dest + len - c, cstr_info, tileno);
|
||||||
|
comp_len = comp_len + e;
|
||||||
|
if (e == -999) {
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
c += e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (e == -999) break;
|
||||||
|
if (cp->max_comp_size){
|
||||||
|
if (comp_len > cp->max_comp_size){
|
||||||
|
e = -999;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (e == -999) break;
|
||||||
|
}
|
||||||
|
}else{ /* t2_mode == FINAL_PASS */
|
||||||
|
pi_create_encode(pi, cp,tileno,pino,tpnum,tppos,t2_mode,cur_totnum_tp);
|
||||||
|
while (pi_next(&pi[pino])) {
|
||||||
|
if (pi[pino].layno < maxlayers) {
|
||||||
|
e = t2_encode_packet(tile, &cp->tcps[tileno], &pi[pino], c, dest + len - c, cstr_info, tileno);
|
||||||
|
if (e == -999) {
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
c += e;
|
||||||
|
}
|
||||||
|
/* INDEX >> */
|
||||||
|
if(cstr_info) {
|
||||||
|
if(cstr_info->index_write) {
|
||||||
|
opj_tile_info_t *info_TL = &cstr_info->tile[tileno];
|
||||||
|
opj_packet_info_t *info_PK = &info_TL->packet[cstr_info->packno];
|
||||||
|
if (!cstr_info->packno) {
|
||||||
|
info_PK->start_pos = info_TL->end_header + 1;
|
||||||
|
} else {
|
||||||
|
info_PK->start_pos = ((cp->tp_on | tcp->POC)&& info_PK->start_pos) ? info_PK->start_pos : info_TL->packet[cstr_info->packno - 1].end_pos + 1;
|
||||||
|
}
|
||||||
|
info_PK->end_pos = info_PK->start_pos + e - 1;
|
||||||
|
info_PK->end_ph_pos += info_PK->start_pos - 1; // End of packet header which now only represents the distance
|
||||||
|
// to start of packet is incremented by value of start of packet
|
||||||
|
}
|
||||||
|
|
||||||
|
cstr_info->packno++;
|
||||||
|
}
|
||||||
|
/* << INDEX */
|
||||||
|
tile->packno++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pi_destroy(pi, cp, tileno);
|
||||||
|
|
||||||
|
if (e == -999) {
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (c - dest);
|
||||||
|
}
|
||||||
|
|
||||||
|
int t2_decode_packets(opj_t2_t *t2, unsigned char *src, int len, int tileno, opj_tcd_tile_t *tile, opj_codestream_info_t *cstr_info) {
|
||||||
|
unsigned char *c = src;
|
||||||
|
opj_pi_iterator_t *pi;
|
||||||
|
int pino, e = 0;
|
||||||
|
int n = 0, curtp = 0;
|
||||||
|
int tp_start_packno;
|
||||||
|
|
||||||
|
opj_image_t *image = t2->image;
|
||||||
|
opj_cp_t *cp = t2->cp;
|
||||||
|
|
||||||
|
/* create a packet iterator */
|
||||||
|
pi = pi_create_decode(image, cp, tileno);
|
||||||
|
if(!pi) {
|
||||||
|
/* TODO: throw an error */
|
||||||
|
return -999;
|
||||||
|
}
|
||||||
|
|
||||||
|
tp_start_packno = 0;
|
||||||
|
|
||||||
|
for (pino = 0; pino <= cp->tcps[tileno].numpocs; pino++) {
|
||||||
|
while (pi_next(&pi[pino])) {
|
||||||
|
if ((cp->layer==0) || (cp->layer>=((pi[pino].layno)+1))) {
|
||||||
|
opj_packet_info_t *pack_info;
|
||||||
|
if (cstr_info)
|
||||||
|
pack_info = &cstr_info->tile[tileno].packet[cstr_info->packno];
|
||||||
|
else
|
||||||
|
pack_info = NULL;
|
||||||
|
e = t2_decode_packet(t2, c, src + len - c, tile, &cp->tcps[tileno], &pi[pino], pack_info);
|
||||||
|
} else {
|
||||||
|
e = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* progression in resolution */
|
||||||
|
image->comps[pi[pino].compno].resno_decoded =
|
||||||
|
(e > 0) ?
|
||||||
|
int_max(pi[pino].resno, image->comps[pi[pino].compno].resno_decoded)
|
||||||
|
: image->comps[pi[pino].compno].resno_decoded;
|
||||||
|
n++;
|
||||||
|
|
||||||
|
/* INDEX >> */
|
||||||
|
if(cstr_info) {
|
||||||
|
opj_tile_info_t *info_TL = &cstr_info->tile[tileno];
|
||||||
|
opj_packet_info_t *info_PK = &info_TL->packet[cstr_info->packno];
|
||||||
|
if (!cstr_info->packno) {
|
||||||
|
info_PK->start_pos = info_TL->end_header + 1;
|
||||||
|
} else if (info_TL->packet[cstr_info->packno-1].end_pos >= (int)cstr_info->tile[tileno].tp[curtp].tp_end_pos){ // New tile part
|
||||||
|
info_TL->tp[curtp].tp_numpacks = cstr_info->packno - tp_start_packno; // Number of packets in previous tile-part
|
||||||
|
tp_start_packno = cstr_info->packno;
|
||||||
|
curtp++;
|
||||||
|
info_PK->start_pos = cstr_info->tile[tileno].tp[curtp].tp_end_header+1;
|
||||||
|
} else {
|
||||||
|
info_PK->start_pos = (cp->tp_on && info_PK->start_pos) ? info_PK->start_pos : info_TL->packet[cstr_info->packno - 1].end_pos + 1;
|
||||||
|
}
|
||||||
|
info_PK->end_pos = info_PK->start_pos + e - 1;
|
||||||
|
info_PK->end_ph_pos += info_PK->start_pos - 1; // End of packet header which now only represents the distance
|
||||||
|
// to start of packet is incremented by value of start of packet
|
||||||
|
cstr_info->packno++;
|
||||||
|
}
|
||||||
|
/* << INDEX */
|
||||||
|
|
||||||
|
if (e == -999) { /* ADD */
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
c += e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* INDEX >> */
|
||||||
|
if(cstr_info) {
|
||||||
|
cstr_info->tile[tileno].tp[curtp].tp_numpacks = cstr_info->packno - tp_start_packno; // Number of packets in last tile-part
|
||||||
|
}
|
||||||
|
/* << INDEX */
|
||||||
|
|
||||||
|
/* don't forget to release pi */
|
||||||
|
pi_destroy(pi, cp, tileno);
|
||||||
|
|
||||||
|
if (e == -999) {
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (c - src);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
opj_t2_t* t2_create(opj_common_ptr cinfo, opj_image_t *image, opj_cp_t *cp) {
|
||||||
|
/* create the tcd structure */
|
||||||
|
opj_t2_t *t2 = (opj_t2_t*)opj_malloc(sizeof(opj_t2_t));
|
||||||
|
if(!t2) return NULL;
|
||||||
|
t2->cinfo = cinfo;
|
||||||
|
t2->image = image;
|
||||||
|
t2->cp = cp;
|
||||||
|
|
||||||
|
return t2;
|
||||||
|
}
|
||||||
|
|
||||||
|
void t2_destroy(opj_t2_t *t2) {
|
||||||
|
if(t2) {
|
||||||
|
opj_free(t2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
103
extern/libopenjpeg/t2.h
vendored
Normal file
103
extern/libopenjpeg/t2.h
vendored
Normal file
@@ -0,0 +1,103 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __T2_H
|
||||||
|
#define __T2_H
|
||||||
|
/**
|
||||||
|
@file t2.h
|
||||||
|
@brief Implementation of a tier-2 coding (packetization of code-block data) (T2)
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup T2 T2 - Implementation of a tier-2 coding */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tier-2 coding
|
||||||
|
*/
|
||||||
|
typedef struct opj_t2 {
|
||||||
|
/** codec context */
|
||||||
|
opj_common_ptr cinfo;
|
||||||
|
|
||||||
|
/** Encoding: pointer to the src image. Decoding: pointer to the dst image. */
|
||||||
|
opj_image_t *image;
|
||||||
|
/** pointer to the image coding parameters */
|
||||||
|
opj_cp_t *cp;
|
||||||
|
} opj_t2_t;
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
Encode the packets of a tile to a destination buffer
|
||||||
|
@param t2 T2 handle
|
||||||
|
@param tileno number of the tile encoded
|
||||||
|
@param tile the tile for which to write the packets
|
||||||
|
@param maxlayers maximum number of layers
|
||||||
|
@param dest the destination buffer
|
||||||
|
@param len the length of the destination buffer
|
||||||
|
@param cstr_info Codestream information structure
|
||||||
|
@param tpnum Tile part number of the current tile
|
||||||
|
@param tppos The position of the tile part flag in the progression order
|
||||||
|
@param t2_mode If == 0 In Threshold calculation ,If == 1 Final pass
|
||||||
|
@param cur_totnum_tp The total number of tile parts in the current tile
|
||||||
|
*/
|
||||||
|
int t2_encode_packets(opj_t2_t* t2,int tileno, opj_tcd_tile_t *tile, int maxlayers, unsigned char *dest, int len, opj_codestream_info_t *cstr_info,int tpnum, int tppos,int pino,J2K_T2_MODE t2_mode,int cur_totnum_tp);
|
||||||
|
/**
|
||||||
|
Decode the packets of a tile from a source buffer
|
||||||
|
@param t2 T2 handle
|
||||||
|
@param src the source buffer
|
||||||
|
@param len length of the source buffer
|
||||||
|
@param tileno number that identifies the tile for which to decode the packets
|
||||||
|
@param tile tile for which to decode the packets
|
||||||
|
*/
|
||||||
|
int t2_decode_packets(opj_t2_t *t2, unsigned char *src, int len, int tileno, opj_tcd_tile_t *tile, opj_codestream_info_t *cstr_info);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Create a T2 handle
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@param image Source or destination image
|
||||||
|
@param cp Image coding parameters
|
||||||
|
@return Returns a new T2 handle if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_t2_t* t2_create(opj_common_ptr cinfo, opj_image_t *image, opj_cp_t *cp);
|
||||||
|
/**
|
||||||
|
Destroy a T2 handle
|
||||||
|
@param t2 T2 handle to destroy
|
||||||
|
*/
|
||||||
|
void t2_destroy(opj_t2_t *t2);
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __T2_H */
|
||||||
1506
extern/libopenjpeg/tcd.c
vendored
Normal file
1506
extern/libopenjpeg/tcd.c
vendored
Normal file
File diff suppressed because it is too large
Load Diff
285
extern/libopenjpeg/tcd.h
vendored
Normal file
285
extern/libopenjpeg/tcd.h
vendored
Normal file
@@ -0,0 +1,285 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
#ifndef __TCD_H
|
||||||
|
#define __TCD_H
|
||||||
|
/**
|
||||||
|
@file tcd.h
|
||||||
|
@brief Implementation of a tile coder/decoder (TCD)
|
||||||
|
|
||||||
|
The functions in TCD.C have for goal to encode or decode each tile independently from
|
||||||
|
each other. The functions in TCD.C are used by some function in J2K.C.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup TCD TCD - Implementation of a tile coder/decoder */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_seg {
|
||||||
|
unsigned char** data;
|
||||||
|
int dataindex;
|
||||||
|
int numpasses;
|
||||||
|
int len;
|
||||||
|
int maxpasses;
|
||||||
|
int numnewpasses;
|
||||||
|
int newlen;
|
||||||
|
} opj_tcd_seg_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_pass {
|
||||||
|
int rate;
|
||||||
|
double distortiondec;
|
||||||
|
int term, len;
|
||||||
|
} opj_tcd_pass_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_layer {
|
||||||
|
int numpasses; /* Number of passes in the layer */
|
||||||
|
int len; /* len of information */
|
||||||
|
double disto; /* add for index (Cfr. Marcela) */
|
||||||
|
unsigned char *data; /* data */
|
||||||
|
} opj_tcd_layer_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_cblk_enc {
|
||||||
|
unsigned char* data; /* Data */
|
||||||
|
opj_tcd_layer_t* layers; /* layer information */
|
||||||
|
opj_tcd_pass_t* passes; /* information about the passes */
|
||||||
|
int x0, y0, x1, y1; /* dimension of the code-blocks : left upper corner (x0, y0) right low corner (x1,y1) */
|
||||||
|
int numbps;
|
||||||
|
int numlenbits;
|
||||||
|
int numpasses; /* number of pass already done for the code-blocks */
|
||||||
|
int numpassesinlayers; /* number of passes in the layer */
|
||||||
|
int totalpasses; /* total number of passes */
|
||||||
|
} opj_tcd_cblk_enc_t;
|
||||||
|
|
||||||
|
typedef struct opj_tcd_cblk_dec {
|
||||||
|
unsigned char* data; /* Data */
|
||||||
|
opj_tcd_seg_t* segs; /* segments informations */
|
||||||
|
int x0, y0, x1, y1; /* dimension of the code-blocks : left upper corner (x0, y0) right low corner (x1,y1) */
|
||||||
|
int numbps;
|
||||||
|
int numlenbits;
|
||||||
|
int len; /* length */
|
||||||
|
int numnewpasses; /* number of pass added to the code-blocks */
|
||||||
|
int numsegs; /* number of segments */
|
||||||
|
} opj_tcd_cblk_dec_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_precinct {
|
||||||
|
int x0, y0, x1, y1; /* dimension of the precinct : left upper corner (x0, y0) right low corner (x1,y1) */
|
||||||
|
int cw, ch; /* number of precinct in width and heigth */
|
||||||
|
union{ /* code-blocks informations */
|
||||||
|
opj_tcd_cblk_enc_t* enc;
|
||||||
|
opj_tcd_cblk_dec_t* dec;
|
||||||
|
} cblks;
|
||||||
|
opj_tgt_tree_t *incltree; /* inclusion tree */
|
||||||
|
opj_tgt_tree_t *imsbtree; /* IMSB tree */
|
||||||
|
} opj_tcd_precinct_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_band {
|
||||||
|
int x0, y0, x1, y1; /* dimension of the subband : left upper corner (x0, y0) right low corner (x1,y1) */
|
||||||
|
int bandno;
|
||||||
|
opj_tcd_precinct_t *precincts; /* precinct information */
|
||||||
|
int numbps;
|
||||||
|
float stepsize;
|
||||||
|
} opj_tcd_band_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_resolution {
|
||||||
|
int x0, y0, x1, y1; /* dimension of the resolution level : left upper corner (x0, y0) right low corner (x1,y1) */
|
||||||
|
int pw, ph;
|
||||||
|
int numbands; /* number sub-band for the resolution level */
|
||||||
|
opj_tcd_band_t bands[3]; /* subband information */
|
||||||
|
} opj_tcd_resolution_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_tilecomp {
|
||||||
|
int x0, y0, x1, y1; /* dimension of component : left upper corner (x0, y0) right low corner (x1,y1) */
|
||||||
|
int numresolutions; /* number of resolutions level */
|
||||||
|
opj_tcd_resolution_t *resolutions; /* resolutions information */
|
||||||
|
int *data; /* data of the component */
|
||||||
|
int numpix; /* add fixed_quality */
|
||||||
|
} opj_tcd_tilecomp_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_tile {
|
||||||
|
int x0, y0, x1, y1; /* dimension of the tile : left upper corner (x0, y0) right low corner (x1,y1) */
|
||||||
|
int numcomps; /* number of components in tile */
|
||||||
|
opj_tcd_tilecomp_t *comps; /* Components information */
|
||||||
|
int numpix; /* add fixed_quality */
|
||||||
|
double distotile; /* add fixed_quality */
|
||||||
|
double distolayer[100]; /* add fixed_quality */
|
||||||
|
/** packet number */
|
||||||
|
int packno;
|
||||||
|
} opj_tcd_tile_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
FIXME: documentation
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd_image {
|
||||||
|
int tw, th; /* number of tiles in width and heigth */
|
||||||
|
opj_tcd_tile_t *tiles; /* Tiles information */
|
||||||
|
} opj_tcd_image_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tile coder/decoder
|
||||||
|
*/
|
||||||
|
typedef struct opj_tcd {
|
||||||
|
/** Position of the tilepart flag in Progression order*/
|
||||||
|
int tp_pos;
|
||||||
|
/** Tile part number*/
|
||||||
|
int tp_num;
|
||||||
|
/** Current tile part number*/
|
||||||
|
int cur_tp_num;
|
||||||
|
/** Total number of tileparts of the current tile*/
|
||||||
|
int cur_totnum_tp;
|
||||||
|
/** Current Packet iterator number */
|
||||||
|
int cur_pino;
|
||||||
|
/** codec context */
|
||||||
|
opj_common_ptr cinfo;
|
||||||
|
|
||||||
|
/** info on each image tile */
|
||||||
|
opj_tcd_image_t *tcd_image;
|
||||||
|
/** image */
|
||||||
|
opj_image_t *image;
|
||||||
|
/** coding parameters */
|
||||||
|
opj_cp_t *cp;
|
||||||
|
/** pointer to the current encoded/decoded tile */
|
||||||
|
opj_tcd_tile_t *tcd_tile;
|
||||||
|
/** coding/decoding parameters common to all tiles */
|
||||||
|
opj_tcp_t *tcp;
|
||||||
|
/** current encoded/decoded tile */
|
||||||
|
int tcd_tileno;
|
||||||
|
/** Time taken to encode a tile*/
|
||||||
|
double encoding_time;
|
||||||
|
} opj_tcd_t;
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
Dump the content of a tcd structure
|
||||||
|
*/
|
||||||
|
void tcd_dump(FILE *fd, opj_tcd_t *tcd, opj_tcd_image_t *img);
|
||||||
|
/**
|
||||||
|
Create a new TCD handle
|
||||||
|
@param cinfo Codec context info
|
||||||
|
@return Returns a new TCD handle if successful returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_tcd_t* tcd_create(opj_common_ptr cinfo);
|
||||||
|
/**
|
||||||
|
Destroy a previously created TCD handle
|
||||||
|
@param tcd TCD handle to destroy
|
||||||
|
*/
|
||||||
|
void tcd_destroy(opj_tcd_t *tcd);
|
||||||
|
/**
|
||||||
|
Initialize the tile coder (allocate the memory)
|
||||||
|
@param tcd TCD handle
|
||||||
|
@param image Raw image
|
||||||
|
@param cp Coding parameters
|
||||||
|
@param curtileno Number that identifies the tile that will be encoded
|
||||||
|
*/
|
||||||
|
void tcd_malloc_encode(opj_tcd_t *tcd, opj_image_t * image, opj_cp_t * cp, int curtileno);
|
||||||
|
/**
|
||||||
|
Free the memory allocated for encoding
|
||||||
|
@param tcd TCD handle
|
||||||
|
*/
|
||||||
|
void tcd_free_encode(opj_tcd_t *tcd);
|
||||||
|
/**
|
||||||
|
Initialize the tile coder (reuses the memory allocated by tcd_malloc_encode)
|
||||||
|
@param tcd TCD handle
|
||||||
|
@param image Raw image
|
||||||
|
@param cp Coding parameters
|
||||||
|
@param curtileno Number that identifies the tile that will be encoded
|
||||||
|
*/
|
||||||
|
void tcd_init_encode(opj_tcd_t *tcd, opj_image_t * image, opj_cp_t * cp, int curtileno);
|
||||||
|
/**
|
||||||
|
Initialize the tile decoder
|
||||||
|
@param tcd TCD handle
|
||||||
|
@param image Raw image
|
||||||
|
@param cp Coding parameters
|
||||||
|
*/
|
||||||
|
void tcd_malloc_decode(opj_tcd_t *tcd, opj_image_t * image, opj_cp_t * cp);
|
||||||
|
void tcd_malloc_decode_tile(opj_tcd_t *tcd, opj_image_t * image, opj_cp_t * cp, int tileno, opj_codestream_info_t *cstr_info);
|
||||||
|
void tcd_makelayer_fixed(opj_tcd_t *tcd, int layno, int final);
|
||||||
|
void tcd_rateallocate_fixed(opj_tcd_t *tcd);
|
||||||
|
void tcd_makelayer(opj_tcd_t *tcd, int layno, double thresh, int final);
|
||||||
|
bool tcd_rateallocate(opj_tcd_t *tcd, unsigned char *dest, int len, opj_codestream_info_t *cstr_info);
|
||||||
|
/**
|
||||||
|
Encode a tile from the raw image into a buffer
|
||||||
|
@param tcd TCD handle
|
||||||
|
@param tileno Number that identifies one of the tiles to be encoded
|
||||||
|
@param dest Destination buffer
|
||||||
|
@param len Length of destination buffer
|
||||||
|
@param cstr_info Codestream information structure
|
||||||
|
@return
|
||||||
|
*/
|
||||||
|
int tcd_encode_tile(opj_tcd_t *tcd, int tileno, unsigned char *dest, int len, opj_codestream_info_t *cstr_info);
|
||||||
|
/**
|
||||||
|
Decode a tile from a buffer into a raw image
|
||||||
|
@param tcd TCD handle
|
||||||
|
@param src Source buffer
|
||||||
|
@param len Length of source buffer
|
||||||
|
@param tileno Number that identifies one of the tiles to be decoded
|
||||||
|
*/
|
||||||
|
bool tcd_decode_tile(opj_tcd_t *tcd, unsigned char *src, int len, int tileno, opj_codestream_info_t *cstr_info);
|
||||||
|
/**
|
||||||
|
Free the memory allocated for decoding
|
||||||
|
@param tcd TCD handle
|
||||||
|
*/
|
||||||
|
void tcd_free_decode(opj_tcd_t *tcd);
|
||||||
|
void tcd_free_decode_tile(opj_tcd_t *tcd, int tileno);
|
||||||
|
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __TCD_H */
|
||||||
213
extern/libopenjpeg/tgt.c
vendored
Normal file
213
extern/libopenjpeg/tgt.c
vendored
Normal file
@@ -0,0 +1,213 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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"
|
||||||
|
|
||||||
|
/*
|
||||||
|
==========================================================
|
||||||
|
Tag-tree coder interface
|
||||||
|
==========================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
opj_tgt_tree_t *tgt_create(int numleafsh, int numleafsv) {
|
||||||
|
int nplh[32];
|
||||||
|
int nplv[32];
|
||||||
|
opj_tgt_node_t *node = NULL;
|
||||||
|
opj_tgt_node_t *parentnode = NULL;
|
||||||
|
opj_tgt_node_t *parentnode0 = NULL;
|
||||||
|
opj_tgt_tree_t *tree = NULL;
|
||||||
|
int i, j, k;
|
||||||
|
int numlvls;
|
||||||
|
int n;
|
||||||
|
|
||||||
|
tree = (opj_tgt_tree_t *) opj_malloc(sizeof(opj_tgt_tree_t));
|
||||||
|
if(!tree) return NULL;
|
||||||
|
tree->numleafsh = numleafsh;
|
||||||
|
tree->numleafsv = numleafsv;
|
||||||
|
|
||||||
|
numlvls = 0;
|
||||||
|
nplh[0] = numleafsh;
|
||||||
|
nplv[0] = numleafsv;
|
||||||
|
tree->numnodes = 0;
|
||||||
|
do {
|
||||||
|
n = nplh[numlvls] * nplv[numlvls];
|
||||||
|
nplh[numlvls + 1] = (nplh[numlvls] + 1) / 2;
|
||||||
|
nplv[numlvls + 1] = (nplv[numlvls] + 1) / 2;
|
||||||
|
tree->numnodes += n;
|
||||||
|
++numlvls;
|
||||||
|
} while (n > 1);
|
||||||
|
|
||||||
|
/* ADD */
|
||||||
|
if (tree->numnodes == 0) {
|
||||||
|
opj_free(tree);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
tree->nodes = (opj_tgt_node_t*) opj_calloc(tree->numnodes, sizeof(opj_tgt_node_t));
|
||||||
|
if(!tree->nodes) {
|
||||||
|
opj_free(tree);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
node = tree->nodes;
|
||||||
|
parentnode = &tree->nodes[tree->numleafsh * tree->numleafsv];
|
||||||
|
parentnode0 = parentnode;
|
||||||
|
|
||||||
|
for (i = 0; i < numlvls - 1; ++i) {
|
||||||
|
for (j = 0; j < nplv[i]; ++j) {
|
||||||
|
k = nplh[i];
|
||||||
|
while (--k >= 0) {
|
||||||
|
node->parent = parentnode;
|
||||||
|
++node;
|
||||||
|
if (--k >= 0) {
|
||||||
|
node->parent = parentnode;
|
||||||
|
++node;
|
||||||
|
}
|
||||||
|
++parentnode;
|
||||||
|
}
|
||||||
|
if ((j & 1) || j == nplv[i] - 1) {
|
||||||
|
parentnode0 = parentnode;
|
||||||
|
} else {
|
||||||
|
parentnode = parentnode0;
|
||||||
|
parentnode0 += nplh[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
node->parent = 0;
|
||||||
|
|
||||||
|
tgt_reset(tree);
|
||||||
|
|
||||||
|
return tree;
|
||||||
|
}
|
||||||
|
|
||||||
|
void tgt_destroy(opj_tgt_tree_t *tree) {
|
||||||
|
opj_free(tree->nodes);
|
||||||
|
opj_free(tree);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tgt_reset(opj_tgt_tree_t *tree) {
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (NULL == tree)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (i = 0; i < tree->numnodes; i++) {
|
||||||
|
tree->nodes[i].value = 999;
|
||||||
|
tree->nodes[i].low = 0;
|
||||||
|
tree->nodes[i].known = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void tgt_setvalue(opj_tgt_tree_t *tree, int leafno, int value) {
|
||||||
|
opj_tgt_node_t *node;
|
||||||
|
node = &tree->nodes[leafno];
|
||||||
|
while (node && node->value > value) {
|
||||||
|
node->value = value;
|
||||||
|
node = node->parent;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void tgt_encode(opj_bio_t *bio, opj_tgt_tree_t *tree, int leafno, int threshold) {
|
||||||
|
opj_tgt_node_t *stk[31];
|
||||||
|
opj_tgt_node_t **stkptr;
|
||||||
|
opj_tgt_node_t *node;
|
||||||
|
int low;
|
||||||
|
|
||||||
|
stkptr = stk;
|
||||||
|
node = &tree->nodes[leafno];
|
||||||
|
while (node->parent) {
|
||||||
|
*stkptr++ = node;
|
||||||
|
node = node->parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
low = 0;
|
||||||
|
for (;;) {
|
||||||
|
if (low > node->low) {
|
||||||
|
node->low = low;
|
||||||
|
} else {
|
||||||
|
low = node->low;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (low < threshold) {
|
||||||
|
if (low >= node->value) {
|
||||||
|
if (!node->known) {
|
||||||
|
bio_write(bio, 1, 1);
|
||||||
|
node->known = 1;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
bio_write(bio, 0, 1);
|
||||||
|
++low;
|
||||||
|
}
|
||||||
|
|
||||||
|
node->low = low;
|
||||||
|
if (stkptr == stk)
|
||||||
|
break;
|
||||||
|
node = *--stkptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int tgt_decode(opj_bio_t *bio, opj_tgt_tree_t *tree, int leafno, int threshold) {
|
||||||
|
opj_tgt_node_t *stk[31];
|
||||||
|
opj_tgt_node_t **stkptr;
|
||||||
|
opj_tgt_node_t *node;
|
||||||
|
int low;
|
||||||
|
|
||||||
|
stkptr = stk;
|
||||||
|
node = &tree->nodes[leafno];
|
||||||
|
while (node->parent) {
|
||||||
|
*stkptr++ = node;
|
||||||
|
node = node->parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
low = 0;
|
||||||
|
for (;;) {
|
||||||
|
if (low > node->low) {
|
||||||
|
node->low = low;
|
||||||
|
} else {
|
||||||
|
low = node->low;
|
||||||
|
}
|
||||||
|
while (low < threshold && low < node->value) {
|
||||||
|
if (bio_read(bio, 1)) {
|
||||||
|
node->value = low;
|
||||||
|
} else {
|
||||||
|
++low;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
node->low = low;
|
||||||
|
if (stkptr == stk) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
node = *--stkptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (node->value < threshold) ? 1 : 0;
|
||||||
|
}
|
||||||
114
extern/libopenjpeg/tgt.h
vendored
Normal file
114
extern/libopenjpeg/tgt.h
vendored
Normal file
@@ -0,0 +1,114 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2002-2007, Communications and Remote Sensing Laboratory, Universite catholique de Louvain (UCL), Belgium
|
||||||
|
* Copyright (c) 2002-2007, Professor Benoit Macq
|
||||||
|
* Copyright (c) 2001-2003, David Janssens
|
||||||
|
* Copyright (c) 2002-2003, Yannick Verschueren
|
||||||
|
* Copyright (c) 2003-2007, Francois-Olivier Devaux and Antonin Descampe
|
||||||
|
* Copyright (c) 2005, Herve Drolon, FreeImage Team
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __TGT_H
|
||||||
|
#define __TGT_H
|
||||||
|
/**
|
||||||
|
@file tgt.h
|
||||||
|
@brief Implementation of a tag-tree coder (TGT)
|
||||||
|
|
||||||
|
The functions in TGT.C have for goal to realize a tag-tree coder. The functions in TGT.C
|
||||||
|
are used by some function in T2.C.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @defgroup TGT TGT - Implementation of a tag-tree coder */
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tag node
|
||||||
|
*/
|
||||||
|
typedef struct opj_tgt_node {
|
||||||
|
struct opj_tgt_node *parent;
|
||||||
|
int value;
|
||||||
|
int low;
|
||||||
|
int known;
|
||||||
|
} opj_tgt_node_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tag tree
|
||||||
|
*/
|
||||||
|
typedef struct opj_tgt_tree {
|
||||||
|
int numleafsh;
|
||||||
|
int numleafsv;
|
||||||
|
int numnodes;
|
||||||
|
opj_tgt_node_t *nodes;
|
||||||
|
} opj_tgt_tree_t;
|
||||||
|
|
||||||
|
/** @name Exported functions */
|
||||||
|
/*@{*/
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
Create a tag-tree
|
||||||
|
@param numleafsh Width of the array of leafs of the tree
|
||||||
|
@param numleafsv Height of the array of leafs of the tree
|
||||||
|
@return Returns a new tag-tree if successful, returns NULL otherwise
|
||||||
|
*/
|
||||||
|
opj_tgt_tree_t *tgt_create(int numleafsh, int numleafsv);
|
||||||
|
/**
|
||||||
|
Destroy a tag-tree, liberating memory
|
||||||
|
@param tree Tag-tree to destroy
|
||||||
|
*/
|
||||||
|
void tgt_destroy(opj_tgt_tree_t *tree);
|
||||||
|
/**
|
||||||
|
Reset a tag-tree (set all leaves to 0)
|
||||||
|
@param tree Tag-tree to reset
|
||||||
|
*/
|
||||||
|
void tgt_reset(opj_tgt_tree_t *tree);
|
||||||
|
/**
|
||||||
|
Set the value of a leaf of a tag-tree
|
||||||
|
@param tree Tag-tree to modify
|
||||||
|
@param leafno Number that identifies the leaf to modify
|
||||||
|
@param value New value of the leaf
|
||||||
|
*/
|
||||||
|
void tgt_setvalue(opj_tgt_tree_t *tree, int leafno, int value);
|
||||||
|
/**
|
||||||
|
Encode the value of a leaf of the tag-tree up to a given threshold
|
||||||
|
@param bio Pointer to a BIO handle
|
||||||
|
@param tree Tag-tree to modify
|
||||||
|
@param leafno Number that identifies the leaf to encode
|
||||||
|
@param threshold Threshold to use when encoding value of the leaf
|
||||||
|
*/
|
||||||
|
void tgt_encode(opj_bio_t *bio, opj_tgt_tree_t *tree, int leafno, int threshold);
|
||||||
|
/**
|
||||||
|
Decode the value of a leaf of the tag-tree up to a given threshold
|
||||||
|
@param bio Pointer to a BIO handle
|
||||||
|
@param tree Tag-tree to decode
|
||||||
|
@param leafno Number that identifies the leaf to decode
|
||||||
|
@param threshold Threshold to use when decoding value of the leaf
|
||||||
|
@return Returns 1 if the node's value < threshold, returns 0 otherwise
|
||||||
|
*/
|
||||||
|
int tgt_decode(opj_bio_t *bio, opj_tgt_tree_t *tree, int leafno, int threshold);
|
||||||
|
/* ----------------------------------------------------------------------- */
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
|
#endif /* __TGT_H */
|
||||||
1
extern/libredcode/AUTHOR
vendored
Normal file
1
extern/libredcode/AUTHOR
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Peter Schlaile * peter at schlaile dot de
|
||||||
340
extern/libredcode/LICENSE
vendored
Normal file
340
extern/libredcode/LICENSE
vendored
Normal file
@@ -0,0 +1,340 @@
|
|||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
Version 2, June 1991
|
||||||
|
|
||||||
|
Copyright (C) 1989, 1991 Free Software Foundation, Inc.
|
||||||
|
59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The licenses for most software are designed to take away your
|
||||||
|
freedom to share and change it. By contrast, the GNU General Public
|
||||||
|
License is intended to guarantee your freedom to share and change free
|
||||||
|
software--to make sure the software is free for all its users. This
|
||||||
|
General Public License applies to most of the Free Software
|
||||||
|
Foundation's software and to any other program whose authors commit to
|
||||||
|
using it. (Some other Free Software Foundation software is covered by
|
||||||
|
the GNU Library General Public License instead.) You can apply it to
|
||||||
|
your programs, too.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
this service if you wish), that you receive source code or can get it
|
||||||
|
if you want it, that you can change the software or use pieces of it
|
||||||
|
in new free programs; and that you know you can do these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to make restrictions that forbid
|
||||||
|
anyone to deny you these rights or to ask you to surrender the rights.
|
||||||
|
These restrictions translate to certain responsibilities for you if you
|
||||||
|
distribute copies of the software, or if you modify it.
|
||||||
|
|
||||||
|
For example, if you distribute copies of such a program, whether
|
||||||
|
gratis or for a fee, you must give the recipients all the rights that
|
||||||
|
you have. You must make sure that they, too, receive or can get the
|
||||||
|
source code. And you must show them these terms so they know their
|
||||||
|
rights.
|
||||||
|
|
||||||
|
We protect your rights with two steps: (1) copyright the software, and
|
||||||
|
(2) offer you this license which gives you legal permission to copy,
|
||||||
|
distribute and/or modify the software.
|
||||||
|
|
||||||
|
Also, for each author's protection and ours, we want to make certain
|
||||||
|
that everyone understands that there is no warranty for this free
|
||||||
|
software. If the software is modified by someone else and passed on, we
|
||||||
|
want its recipients to know that what they have is not the original, so
|
||||||
|
that any problems introduced by others will not reflect on the original
|
||||||
|
authors' reputations.
|
||||||
|
|
||||||
|
Finally, any free program is threatened constantly by software
|
||||||
|
patents. We wish to avoid the danger that redistributors of a free
|
||||||
|
program will individually obtain patent licenses, in effect making the
|
||||||
|
program proprietary. To prevent this, we have made it clear that any
|
||||||
|
patent must be licensed for everyone's free use or not licensed at all.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||||
|
|
||||||
|
0. This License applies to any program or other work which contains
|
||||||
|
a notice placed by the copyright holder saying it may be distributed
|
||||||
|
under the terms of this General Public License. The "Program", below,
|
||||||
|
refers to any such program or work, and a "work based on the Program"
|
||||||
|
means either the Program or any derivative work under copyright law:
|
||||||
|
that is to say, a work containing the Program or a portion of it,
|
||||||
|
either verbatim or with modifications and/or translated into another
|
||||||
|
language. (Hereinafter, translation is included without limitation in
|
||||||
|
the term "modification".) Each licensee is addressed as "you".
|
||||||
|
|
||||||
|
Activities other than copying, distribution and modification are not
|
||||||
|
covered by this License; they are outside its scope. The act of
|
||||||
|
running the Program is not restricted, and the output from the Program
|
||||||
|
is covered only if its contents constitute a work based on the
|
||||||
|
Program (independent of having been made by running the Program).
|
||||||
|
Whether that is true depends on what the Program does.
|
||||||
|
|
||||||
|
1. You may copy and distribute verbatim copies of the Program's
|
||||||
|
source code as you receive it, in any medium, provided that you
|
||||||
|
conspicuously and appropriately publish on each copy an appropriate
|
||||||
|
copyright notice and disclaimer of warranty; keep intact all the
|
||||||
|
notices that refer to this License and to the absence of any warranty;
|
||||||
|
and give any other recipients of the Program a copy of this License
|
||||||
|
along with the Program.
|
||||||
|
|
||||||
|
You may charge a fee for the physical act of transferring a copy, and
|
||||||
|
you may at your option offer warranty protection in exchange for a fee.
|
||||||
|
|
||||||
|
2. You may modify your copy or copies of the Program or any portion
|
||||||
|
of it, thus forming a work based on the Program, and copy and
|
||||||
|
distribute such modifications or work under the terms of Section 1
|
||||||
|
above, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) You must cause the modified files to carry prominent notices
|
||||||
|
stating that you changed the files and the date of any change.
|
||||||
|
|
||||||
|
b) You must cause any work that you distribute or publish, that in
|
||||||
|
whole or in part contains or is derived from the Program or any
|
||||||
|
part thereof, to be licensed as a whole at no charge to all third
|
||||||
|
parties under the terms of this License.
|
||||||
|
|
||||||
|
c) If the modified program normally reads commands interactively
|
||||||
|
when run, you must cause it, when started running for such
|
||||||
|
interactive use in the most ordinary way, to print or display an
|
||||||
|
announcement including an appropriate copyright notice and a
|
||||||
|
notice that there is no warranty (or else, saying that you provide
|
||||||
|
a warranty) and that users may redistribute the program under
|
||||||
|
these conditions, and telling the user how to view a copy of this
|
||||||
|
License. (Exception: if the Program itself is interactive but
|
||||||
|
does not normally print such an announcement, your work based on
|
||||||
|
the Program is not required to print an announcement.)
|
||||||
|
|
||||||
|
These requirements apply to the modified work as a whole. If
|
||||||
|
identifiable sections of that work are not derived from the Program,
|
||||||
|
and can be reasonably considered independent and separate works in
|
||||||
|
themselves, then this License, and its terms, do not apply to those
|
||||||
|
sections when you distribute them as separate works. But when you
|
||||||
|
distribute the same sections as part of a whole which is a work based
|
||||||
|
on the Program, the distribution of the whole must be on the terms of
|
||||||
|
this License, whose permissions for other licensees extend to the
|
||||||
|
entire whole, and thus to each and every part regardless of who wrote it.
|
||||||
|
|
||||||
|
Thus, it is not the intent of this section to claim rights or contest
|
||||||
|
your rights to work written entirely by you; rather, the intent is to
|
||||||
|
exercise the right to control the distribution of derivative or
|
||||||
|
collective works based on the Program.
|
||||||
|
|
||||||
|
In addition, mere aggregation of another work not based on the Program
|
||||||
|
with the Program (or with a work based on the Program) on a volume of
|
||||||
|
a storage or distribution medium does not bring the other work under
|
||||||
|
the scope of this License.
|
||||||
|
|
||||||
|
3. You may copy and distribute the Program (or a work based on it,
|
||||||
|
under Section 2) in object code or executable form under the terms of
|
||||||
|
Sections 1 and 2 above provided that you also do one of the following:
|
||||||
|
|
||||||
|
a) Accompany it with the complete corresponding machine-readable
|
||||||
|
source code, which must be distributed under the terms of Sections
|
||||||
|
1 and 2 above on a medium customarily used for software interchange; or,
|
||||||
|
|
||||||
|
b) Accompany it with a written offer, valid for at least three
|
||||||
|
years, to give any third party, for a charge no more than your
|
||||||
|
cost of physically performing source distribution, a complete
|
||||||
|
machine-readable copy of the corresponding source code, to be
|
||||||
|
distributed under the terms of Sections 1 and 2 above on a medium
|
||||||
|
customarily used for software interchange; or,
|
||||||
|
|
||||||
|
c) Accompany it with the information you received as to the offer
|
||||||
|
to distribute corresponding source code. (This alternative is
|
||||||
|
allowed only for noncommercial distribution and only if you
|
||||||
|
received the program in object code or executable form with such
|
||||||
|
an offer, in accord with Subsection b above.)
|
||||||
|
|
||||||
|
The source code for a work means the preferred form of the work for
|
||||||
|
making modifications to it. For an executable work, complete source
|
||||||
|
code means all the source code for all modules it contains, plus any
|
||||||
|
associated interface definition files, plus the scripts used to
|
||||||
|
control compilation and installation of the executable. However, as a
|
||||||
|
special exception, the source code distributed need not include
|
||||||
|
anything that is normally distributed (in either source or binary
|
||||||
|
form) with the major components (compiler, kernel, and so on) of the
|
||||||
|
operating system on which the executable runs, unless that component
|
||||||
|
itself accompanies the executable.
|
||||||
|
|
||||||
|
If distribution of executable or object code is made by offering
|
||||||
|
access to copy from a designated place, then offering equivalent
|
||||||
|
access to copy the source code from the same place counts as
|
||||||
|
distribution of the source code, even though third parties are not
|
||||||
|
compelled to copy the source along with the object code.
|
||||||
|
|
||||||
|
4. You may not copy, modify, sublicense, or distribute the Program
|
||||||
|
except as expressly provided under this License. Any attempt
|
||||||
|
otherwise to copy, modify, sublicense or distribute the Program is
|
||||||
|
void, and will automatically terminate your rights under this License.
|
||||||
|
However, parties who have received copies, or rights, from you under
|
||||||
|
this License will not have their licenses terminated so long as such
|
||||||
|
parties remain in full compliance.
|
||||||
|
|
||||||
|
5. You are not required to accept this License, since you have not
|
||||||
|
signed it. However, nothing else grants you permission to modify or
|
||||||
|
distribute the Program or its derivative works. These actions are
|
||||||
|
prohibited by law if you do not accept this License. Therefore, by
|
||||||
|
modifying or distributing the Program (or any work based on the
|
||||||
|
Program), you indicate your acceptance of this License to do so, and
|
||||||
|
all its terms and conditions for copying, distributing or modifying
|
||||||
|
the Program or works based on it.
|
||||||
|
|
||||||
|
6. Each time you redistribute the Program (or any work based on the
|
||||||
|
Program), the recipient automatically receives a license from the
|
||||||
|
original licensor to copy, distribute or modify the Program subject to
|
||||||
|
these terms and conditions. You may not impose any further
|
||||||
|
restrictions on the recipients' exercise of the rights granted herein.
|
||||||
|
You are not responsible for enforcing compliance by third parties to
|
||||||
|
this License.
|
||||||
|
|
||||||
|
7. If, as a consequence of a court judgment or allegation of patent
|
||||||
|
infringement or for any other reason (not limited to patent issues),
|
||||||
|
conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot
|
||||||
|
distribute so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you
|
||||||
|
may not distribute the Program at all. For example, if a patent
|
||||||
|
license would not permit royalty-free redistribution of the Program by
|
||||||
|
all those who receive copies directly or indirectly through you, then
|
||||||
|
the only way you could satisfy both it and this License would be to
|
||||||
|
refrain entirely from distribution of the Program.
|
||||||
|
|
||||||
|
If any portion of this section is held invalid or unenforceable under
|
||||||
|
any particular circumstance, the balance of the section is intended to
|
||||||
|
apply and the section as a whole is intended to apply in other
|
||||||
|
circumstances.
|
||||||
|
|
||||||
|
It is not the purpose of this section to induce you to infringe any
|
||||||
|
patents or other property right claims or to contest validity of any
|
||||||
|
such claims; this section has the sole purpose of protecting the
|
||||||
|
integrity of the free software distribution system, which is
|
||||||
|
implemented by public license practices. Many people have made
|
||||||
|
generous contributions to the wide range of software distributed
|
||||||
|
through that system in reliance on consistent application of that
|
||||||
|
system; it is up to the author/donor to decide if he or she is willing
|
||||||
|
to distribute software through any other system and a licensee cannot
|
||||||
|
impose that choice.
|
||||||
|
|
||||||
|
This section is intended to make thoroughly clear what is believed to
|
||||||
|
be a consequence of the rest of this License.
|
||||||
|
|
||||||
|
8. If the distribution and/or use of the Program is restricted in
|
||||||
|
certain countries either by patents or by copyrighted interfaces, the
|
||||||
|
original copyright holder who places the Program under this License
|
||||||
|
may add an explicit geographical distribution limitation excluding
|
||||||
|
those countries, so that distribution is permitted only in or among
|
||||||
|
countries not thus excluded. In such case, this License incorporates
|
||||||
|
the limitation as if written in the body of this License.
|
||||||
|
|
||||||
|
9. The Free Software Foundation may publish revised and/or new versions
|
||||||
|
of the General Public License from time to time. Such new versions will
|
||||||
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the Program
|
||||||
|
specifies a version number of this License which applies to it and "any
|
||||||
|
later version", you have the option of following the terms and conditions
|
||||||
|
either of that version or of any later version published by the Free
|
||||||
|
Software Foundation. If the Program does not specify a version number of
|
||||||
|
this License, you may choose any version ever published by the Free Software
|
||||||
|
Foundation.
|
||||||
|
|
||||||
|
10. If you wish to incorporate parts of the Program into other free
|
||||||
|
programs whose distribution conditions are different, write to the author
|
||||||
|
to ask for permission. For software which is copyrighted by the Free
|
||||||
|
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||||
|
make exceptions for this. Our decision will be guided by the two goals
|
||||||
|
of preserving the free status of all derivatives of our free software and
|
||||||
|
of promoting the sharing and reuse of software generally.
|
||||||
|
|
||||||
|
NO WARRANTY
|
||||||
|
|
||||||
|
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||||
|
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||||
|
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||||
|
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||||
|
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||||
|
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||||
|
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||||
|
REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||||
|
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||||
|
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||||
|
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||||
|
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||||
|
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||||
|
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||||
|
POSSIBILITY OF SUCH DAMAGES.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Programs
|
||||||
|
|
||||||
|
If you develop a new program, and you want it to be of the greatest
|
||||||
|
possible use to the public, the best way to achieve this is to make it
|
||||||
|
free software which everyone can redistribute and change under these terms.
|
||||||
|
|
||||||
|
To do so, attach the following notices to the program. It is safest
|
||||||
|
to attach them to the start of each source file to most effectively
|
||||||
|
convey the exclusion of warranty; and each file should have at least
|
||||||
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
This program is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation; either version 2 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
If the program is interactive, make it output a short notice like this
|
||||||
|
when it starts in an interactive mode:
|
||||||
|
|
||||||
|
Gnomovision version 69, Copyright (C) year name of author
|
||||||
|
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
|
This is free software, and you are welcome to redistribute it
|
||||||
|
under certain conditions; type `show c' for details.
|
||||||
|
|
||||||
|
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||||
|
parts of the General Public License. Of course, the commands you use may
|
||||||
|
be called something other than `show w' and `show c'; they could even be
|
||||||
|
mouse-clicks or menu items--whatever suits your program.
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or your
|
||||||
|
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||||
|
necessary. Here is a sample; alter the names:
|
||||||
|
|
||||||
|
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||||
|
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||||
|
|
||||||
|
<signature of Ty Coon>, 1 April 1989
|
||||||
|
Ty Coon, President of Vice
|
||||||
|
|
||||||
|
This General Public License does not permit incorporating your program into
|
||||||
|
proprietary programs. If your program is a subroutine library, you may
|
||||||
|
consider it more useful to permit linking proprietary applications with the
|
||||||
|
library. If this is what you want to do, use the GNU Library General
|
||||||
|
Public License instead of this License.
|
||||||
23
extern/libredcode/NOTES
vendored
Normal file
23
extern/libredcode/NOTES
vendored
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
This is a redcode decoder written solely by looking at some
|
||||||
|
R3D files that are floating around on the web.
|
||||||
|
|
||||||
|
This decoder works with the files I tested, but I don't give
|
||||||
|
any guarantee (especially, if RED decides to change the
|
||||||
|
file format).
|
||||||
|
|
||||||
|
That said: I found it a really nice idea of them, to make
|
||||||
|
such a clear and comprehensible file layout.
|
||||||
|
|
||||||
|
Choosing JPEG2000 as an encoding format and the way, they
|
||||||
|
did YCbCr encoding of the sensor data is simply brilliant.
|
||||||
|
|
||||||
|
My approach of decoding the sensor data is probably not
|
||||||
|
so brilliant, but seems to work rather nicely (and sharply!).
|
||||||
|
|
||||||
|
If someone finds a better way and/or RED decides to release
|
||||||
|
specification, just go ahead and feel free to change it!
|
||||||
|
|
||||||
|
Until that happens: simply enjoy opening RED footage within Blender!
|
||||||
|
|
||||||
|
-- Peter Schlaile, June 2008
|
||||||
|
|
||||||
28
extern/libredcode/SConscript
vendored
Normal file
28
extern/libredcode/SConscript
vendored
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
|
||||||
|
Import('env')
|
||||||
|
|
||||||
|
sources = env.Glob('*.c')
|
||||||
|
incs = '. ../libopenjpeg'
|
||||||
|
|
||||||
|
root = "extern/libredcode"
|
||||||
|
|
||||||
|
if not os.path.isdir(root + "/include"):
|
||||||
|
os.mkdir(root + "/include");
|
||||||
|
if not os.path.isdir(root + "/include/redcode"):
|
||||||
|
os.mkdir(root + "/include/redcode");
|
||||||
|
|
||||||
|
for h in env.Glob('*.h'):
|
||||||
|
shutil.copyfile(root + "/" + h,
|
||||||
|
root + "/include/redcode/" + h)
|
||||||
|
|
||||||
|
|
||||||
|
env.BlenderLib ( libname='extern_redcode',
|
||||||
|
sources=sources, includes=Split(incs),
|
||||||
|
defines=[],
|
||||||
|
libtype=['core','intern','player'],
|
||||||
|
priority=[5, 5, 200], compileflags = [])
|
||||||
141
extern/libredcode/codec.c
vendored
Normal file
141
extern/libredcode/codec.c
vendored
Normal file
@@ -0,0 +1,141 @@
|
|||||||
|
#include "codec.h"
|
||||||
|
#include "format.h"
|
||||||
|
#include "debayer.h"
|
||||||
|
|
||||||
|
#include <openjpeg.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
static void error_callback(const char *msg, void *client_data) {
|
||||||
|
FILE *stream = (FILE*)client_data;
|
||||||
|
fprintf(stream, "[R3D ERR] %s", msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void warning_callback(const char *msg, void *client_data) {
|
||||||
|
FILE *stream = (FILE*)client_data;
|
||||||
|
fprintf(stream, "[R3D WARN] %s", msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void info_callback(const char *msg, void *client_data) {
|
||||||
|
(void)client_data;
|
||||||
|
fprintf(stdout, "[R3D INFO] %s", msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define J2K_CFMT 0
|
||||||
|
#define JP2_CFMT 1
|
||||||
|
#define JPT_CFMT 2
|
||||||
|
|
||||||
|
struct redcode_frame_raw * redcode_decode_video_raw(
|
||||||
|
struct redcode_frame * frame, int scale)
|
||||||
|
{
|
||||||
|
struct redcode_frame_raw * rv = NULL;
|
||||||
|
opj_dparameters_t parameters; /* decompression parameters */
|
||||||
|
opj_event_mgr_t event_mgr; /* event manager */
|
||||||
|
opj_image_t *image = NULL;
|
||||||
|
opj_dinfo_t* dinfo = NULL; /* handle to a decompressor */
|
||||||
|
opj_cio_t *cio = NULL;
|
||||||
|
|
||||||
|
memset(&event_mgr, 0, sizeof(opj_event_mgr_t));
|
||||||
|
event_mgr.error_handler = error_callback;
|
||||||
|
event_mgr.warning_handler = warning_callback;
|
||||||
|
event_mgr.info_handler = info_callback;
|
||||||
|
|
||||||
|
opj_set_default_decoder_parameters(¶meters);
|
||||||
|
|
||||||
|
parameters.decod_format = JP2_CFMT;
|
||||||
|
|
||||||
|
if (scale == 2) {
|
||||||
|
parameters.cp_reduce = 1;
|
||||||
|
} else if (scale == 4) {
|
||||||
|
parameters.cp_reduce = 2;
|
||||||
|
} else if (scale == 8) {
|
||||||
|
parameters.cp_reduce = 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* JPEG 2000 compressed image data */
|
||||||
|
|
||||||
|
/* get a decoder handle */
|
||||||
|
dinfo = opj_create_decompress(CODEC_JP2);
|
||||||
|
|
||||||
|
/* catch events using our callbacks and give a local context */
|
||||||
|
opj_set_event_mgr((opj_common_ptr)dinfo, &event_mgr, stderr);
|
||||||
|
|
||||||
|
/* setup the decoder decoding parameters using the current image
|
||||||
|
and user parameters */
|
||||||
|
opj_setup_decoder(dinfo, ¶meters);
|
||||||
|
|
||||||
|
/* open a byte stream */
|
||||||
|
cio = opj_cio_open((opj_common_ptr)dinfo,
|
||||||
|
frame->data + frame->offset, frame->length);
|
||||||
|
|
||||||
|
image = opj_decode(dinfo, cio);
|
||||||
|
|
||||||
|
if(!image) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"ERROR -> j2k_to_image: failed to decode image!\n");
|
||||||
|
opj_destroy_decompress(dinfo);
|
||||||
|
opj_cio_close(cio);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* close the byte stream */
|
||||||
|
opj_cio_close(cio);
|
||||||
|
|
||||||
|
/* free remaining structures */
|
||||||
|
if(dinfo) {
|
||||||
|
opj_destroy_decompress(dinfo);
|
||||||
|
}
|
||||||
|
|
||||||
|
if((image->numcomps * image->x1 * image->y1) == 0) {
|
||||||
|
opj_image_destroy(image);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
rv = (struct redcode_frame_raw *) calloc(
|
||||||
|
1, sizeof(struct redcode_frame_raw));
|
||||||
|
|
||||||
|
rv->data = image;
|
||||||
|
rv->width = image->comps[0].w;
|
||||||
|
rv->height = image->comps[0].h;
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
int redcode_decode_video_float(struct redcode_frame_raw * frame,
|
||||||
|
float * out, int scale)
|
||||||
|
{
|
||||||
|
int* planes[4];
|
||||||
|
int i;
|
||||||
|
opj_image_t *image = (opj_image_t*) frame->data;
|
||||||
|
|
||||||
|
if (image->numcomps != 4) {
|
||||||
|
fprintf(stderr, "R3D: need 4 planes, but got: %d\n",
|
||||||
|
image->numcomps);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < 4; i++) {
|
||||||
|
planes[i] = image->comps[i].data;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (scale == 1) {
|
||||||
|
redcode_ycbcr2rgb_fullscale(
|
||||||
|
planes, frame->width, frame->height, out);
|
||||||
|
} else if (scale == 2) {
|
||||||
|
redcode_ycbcr2rgb_halfscale(
|
||||||
|
planes, frame->width, frame->height, out);
|
||||||
|
} else if (scale == 4) {
|
||||||
|
redcode_ycbcr2rgb_quarterscale(
|
||||||
|
planes, frame->width, frame->height, out);
|
||||||
|
}
|
||||||
|
|
||||||
|
opj_image_destroy(image);
|
||||||
|
|
||||||
|
free(frame);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
34
extern/libredcode/codec.h
vendored
Normal file
34
extern/libredcode/codec.h
vendored
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
#ifndef __redcode_codec_h_included__
|
||||||
|
#define __redcode_codec_h_included__
|
||||||
|
|
||||||
|
struct redcode_frame;
|
||||||
|
|
||||||
|
struct redcode_frame_raw {
|
||||||
|
void * data;
|
||||||
|
int width;
|
||||||
|
int height;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* do the JPEG2000 decompression into YCbCrY planes */
|
||||||
|
struct redcode_frame_raw * redcode_decode_video_raw(
|
||||||
|
struct redcode_frame * frame, int scale);
|
||||||
|
|
||||||
|
/* finally decode RAW frame into out-buffer (which has to be allocated
|
||||||
|
in advance)
|
||||||
|
|
||||||
|
Keep in mind: frame_raw-width + height is half sized.
|
||||||
|
(one pixel contains 2x2 bayer-sensor data)
|
||||||
|
|
||||||
|
output-buffer should have room for
|
||||||
|
|
||||||
|
scale = 1 : width * height * 4 * 4 * sizeof(float)
|
||||||
|
scale = 2 : width * height * 4 * sizeof(float)
|
||||||
|
scale = 4 : width * height * sizeof(float)
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
int redcode_decode_video_float(struct redcode_frame_raw * frame,
|
||||||
|
float * out, int scale);
|
||||||
|
|
||||||
|
|
||||||
|
#endif
|
||||||
129
extern/libredcode/debayer.c
vendored
Normal file
129
extern/libredcode/debayer.c
vendored
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
#include "debayer.h"
|
||||||
|
|
||||||
|
/* pretty simple but astonishingly very effective "debayer" function
|
||||||
|
*/
|
||||||
|
|
||||||
|
void redcode_ycbcr2rgb_fullscale(
|
||||||
|
int ** planes, int width, int height, float * out)
|
||||||
|
{
|
||||||
|
int x,y;
|
||||||
|
int pix_max = 4096;
|
||||||
|
int mask = pix_max - 1;
|
||||||
|
float Kb = 0.0722;
|
||||||
|
float Kr = 0.2126;
|
||||||
|
float *o;
|
||||||
|
|
||||||
|
for (y = 0; y < height; y++) {
|
||||||
|
for (x = 0; x < width; x++) {
|
||||||
|
int i = x + y*width;
|
||||||
|
int i_p = (y > 0) ? i-width : i;
|
||||||
|
int i_n = (y < (height-1)) ? i + width : i;
|
||||||
|
float y1n = planes[0][i_n] & mask;
|
||||||
|
float y1 = planes[0][i] & mask;
|
||||||
|
float cb = (planes[1][i] & mask) - pix_max/2;
|
||||||
|
float cr = (planes[2][i] & mask) - pix_max/2;
|
||||||
|
float y2 = (planes[3][i] & mask);
|
||||||
|
float y2p = (planes[3][i_p] & mask);
|
||||||
|
|
||||||
|
float b_ = cb * (1.0 - Kb)/(pix_max/2);
|
||||||
|
float r_ = cr * (1.0 - Kr)/(pix_max/2);
|
||||||
|
float g_ = (- Kr * r_ - Kb * b_)/(1.0 - Kr - Kb);
|
||||||
|
|
||||||
|
float y_[4] = {y1 / pix_max,
|
||||||
|
(y2 + y2p)/2 / pix_max,
|
||||||
|
(y1 + y1n)/2 / pix_max,
|
||||||
|
y2 / pix_max};
|
||||||
|
|
||||||
|
int j;
|
||||||
|
int yc = 0;
|
||||||
|
|
||||||
|
o = out + (2*height-1-2*y)*2*4*width
|
||||||
|
+ x*2*4;
|
||||||
|
|
||||||
|
for (j = 0; j < 8; j += 4) {
|
||||||
|
o[j+0] = r_ + y_[yc];
|
||||||
|
o[j+1] = g_ + y_[yc];
|
||||||
|
o[j+2] = b_ + y_[yc];
|
||||||
|
o[j+3] = 1.0;
|
||||||
|
yc++;
|
||||||
|
}
|
||||||
|
|
||||||
|
o = out + (2*height-1-2*y)*2*4*width
|
||||||
|
+ x*2*4 - 2*4*width;
|
||||||
|
|
||||||
|
for (j = 0; j < 8; j += 4) {
|
||||||
|
o[j+0] = r_ + y_[yc];
|
||||||
|
o[j+1] = g_ + y_[yc];
|
||||||
|
o[j+2] = b_ + y_[yc];
|
||||||
|
o[j+3] = 1.0;
|
||||||
|
yc++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void redcode_ycbcr2rgb_halfscale(
|
||||||
|
int ** planes, int width, int height, float * out)
|
||||||
|
{
|
||||||
|
int x,y;
|
||||||
|
int pix_max = 4096;
|
||||||
|
int mask = pix_max - 1;
|
||||||
|
float Kb = 0.0722;
|
||||||
|
float Kr = 0.2126;
|
||||||
|
|
||||||
|
for (y = 0; y < height; y++) {
|
||||||
|
float *o = out + width * (height - y - 1);
|
||||||
|
for (x = 0; x < width; x++) {
|
||||||
|
int i = y*height + x;
|
||||||
|
float y1 = (planes[0][i] & mask);
|
||||||
|
float cb = (planes[1][i] & mask) - pix_max/2;
|
||||||
|
float cr = (planes[2][i] & mask) - pix_max/2;
|
||||||
|
float y2 = (planes[3][i] & mask);
|
||||||
|
|
||||||
|
float b_ = cb * (1.0 - Kb)/(pix_max/2);
|
||||||
|
float r_ = cr * (1.0 - Kr)/(pix_max/2);
|
||||||
|
float g_ = (- Kr * r_ - Kb * b_)/(1.0 - Kr - Kb);
|
||||||
|
|
||||||
|
float y = (y1 + y2)/2 / pix_max;
|
||||||
|
|
||||||
|
*o++ = r_ + y;
|
||||||
|
*o++ = g_ + y;
|
||||||
|
*o++ = b_ + y;
|
||||||
|
*o++ = 1.0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void redcode_ycbcr2rgb_quarterscale(
|
||||||
|
int ** planes, int width, int height, float * out)
|
||||||
|
{
|
||||||
|
int x,y;
|
||||||
|
int pix_max = 4096;
|
||||||
|
int mask = pix_max - 1;
|
||||||
|
float Kb = 0.0722;
|
||||||
|
float Kr = 0.2126;
|
||||||
|
|
||||||
|
for (y = 0; y < height; y += 2) {
|
||||||
|
float *o = out + (width/2) * (height/2 - y/2 - 1);
|
||||||
|
for (x = 0; x < width; x += 2) {
|
||||||
|
int i = y * width + x;
|
||||||
|
float y1 = planes[0][i] & mask;
|
||||||
|
float cb = (planes[1][i] & mask) - pix_max/2;
|
||||||
|
float cr = (planes[2][i] & mask) - pix_max/2;
|
||||||
|
float y2 = planes[3][i] & mask;
|
||||||
|
|
||||||
|
float b_ = cb * (1.0 - Kb)/(pix_max/2);
|
||||||
|
float r_ = cr * (1.0 - Kr)/(pix_max/2);
|
||||||
|
float g_ = (- Kr * r_ - Kb * b_)/(1.0 - Kr - Kb);
|
||||||
|
|
||||||
|
float y = (y1 + y2)/2 / pix_max;
|
||||||
|
|
||||||
|
*o++ = r_ + y;
|
||||||
|
*o++ = g_ + y;
|
||||||
|
*o++ = b_ + y;
|
||||||
|
*o++ = 1.0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
11
extern/libredcode/debayer.h
vendored
Normal file
11
extern/libredcode/debayer.h
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
#ifndef __redcode_debayer_h_included__
|
||||||
|
#define __redcode_debayer_h_included__ 1
|
||||||
|
|
||||||
|
void redcode_ycbcr2rgb_fullscale(
|
||||||
|
int ** planes, int width, int height, float * out);
|
||||||
|
void redcode_ycbcr2rgb_halfscale(
|
||||||
|
int ** planes, int width, int height, float * out);
|
||||||
|
void redcode_ycbcr2rgb_quarterscale(
|
||||||
|
int ** planes, int width, int height, float * out);
|
||||||
|
|
||||||
|
#endif
|
||||||
213
extern/libredcode/format.c
vendored
Normal file
213
extern/libredcode/format.c
vendored
Normal file
@@ -0,0 +1,213 @@
|
|||||||
|
#include <netinet/in.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include "format.h"
|
||||||
|
|
||||||
|
struct red_reob {
|
||||||
|
unsigned long len;
|
||||||
|
char head[4];
|
||||||
|
|
||||||
|
unsigned long rdvo;
|
||||||
|
unsigned long rdvs;
|
||||||
|
unsigned long rdao;
|
||||||
|
unsigned long rdas;
|
||||||
|
|
||||||
|
unsigned long unknown1;
|
||||||
|
unsigned long unknown2;
|
||||||
|
unsigned long totlen;
|
||||||
|
|
||||||
|
unsigned long avgv;
|
||||||
|
unsigned long avgs;
|
||||||
|
|
||||||
|
unsigned long unknown3;
|
||||||
|
unsigned long unknown4;
|
||||||
|
unsigned long unknown5;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct redcode_handle {
|
||||||
|
FILE * fp;
|
||||||
|
struct red_reob * reob;
|
||||||
|
unsigned long * rdvo;
|
||||||
|
unsigned long * rdvs;
|
||||||
|
unsigned long * rdao;
|
||||||
|
unsigned long * rdas;
|
||||||
|
long cfra;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
static unsigned char* read_packet(FILE * fp, char * expect)
|
||||||
|
{
|
||||||
|
unsigned long len;
|
||||||
|
char head[5];
|
||||||
|
unsigned char * rv;
|
||||||
|
|
||||||
|
fread(&len, 4, 1, fp);
|
||||||
|
fread(&head, 4, 1, fp);
|
||||||
|
|
||||||
|
head[4] = 0;
|
||||||
|
|
||||||
|
len = ntohl(len);
|
||||||
|
|
||||||
|
if (strcmp(expect, head) != 0) {
|
||||||
|
fprintf(stderr, "Read: %s, expect: %s\n", head, expect);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
rv = (unsigned char*) malloc(len + 8);
|
||||||
|
|
||||||
|
memcpy(rv, &len, 4);
|
||||||
|
memcpy(rv + 4, &head, 4);
|
||||||
|
|
||||||
|
fread(rv + 8, len, 1, fp);
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
static unsigned long * read_index_packet(FILE * fp, char * expect)
|
||||||
|
{
|
||||||
|
unsigned long * rv = (unsigned long*) read_packet(fp, expect);
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!rv) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 2; i < rv[0]/4; i++) {
|
||||||
|
rv[i] = ntohl(rv[i]);
|
||||||
|
}
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct red_reob * read_reob(FILE * fp)
|
||||||
|
{
|
||||||
|
fseek(fp, -0x38, SEEK_END);
|
||||||
|
|
||||||
|
return (struct red_reob *) read_index_packet(fp, "REOB");
|
||||||
|
}
|
||||||
|
|
||||||
|
static unsigned long * read_index(FILE * fp, unsigned long i, char * expect)
|
||||||
|
{
|
||||||
|
fseek(fp, i, SEEK_SET);
|
||||||
|
|
||||||
|
return (unsigned long*) read_index_packet(fp, expect);
|
||||||
|
}
|
||||||
|
|
||||||
|
static unsigned char * read_data(FILE * fp, unsigned long i, char * expect)
|
||||||
|
{
|
||||||
|
fseek(fp, i, SEEK_SET);
|
||||||
|
|
||||||
|
return read_packet(fp, expect);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct redcode_handle * redcode_open(const char * fname)
|
||||||
|
{
|
||||||
|
struct redcode_handle * rv = NULL;
|
||||||
|
struct red_reob * reob = NULL;
|
||||||
|
|
||||||
|
FILE * fp = fopen(fname, "rb");
|
||||||
|
|
||||||
|
if (!fp) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
reob = read_reob(fp);
|
||||||
|
if (!reob) {
|
||||||
|
fclose(fp);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
rv = (struct redcode_handle*) calloc(1, sizeof(struct redcode_handle));
|
||||||
|
|
||||||
|
rv->fp = fp;
|
||||||
|
rv->reob = reob;
|
||||||
|
rv->rdvo = read_index(fp, reob->rdvo, "RDVO");
|
||||||
|
rv->rdvs = read_index(fp, reob->rdvs, "RDVS");
|
||||||
|
rv->rdao = read_index(fp, reob->rdao, "RDAO");
|
||||||
|
rv->rdas = read_index(fp, reob->rdas, "RDAS");
|
||||||
|
|
||||||
|
if (!rv->rdvo || !rv->rdvs || !rv->rdao || !rv->rdas) {
|
||||||
|
redcode_close(rv);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
void redcode_close(struct redcode_handle * handle)
|
||||||
|
{
|
||||||
|
if (handle->reob) {
|
||||||
|
free(handle->reob);
|
||||||
|
}
|
||||||
|
if (handle->rdvo) {
|
||||||
|
free(handle->rdvo);
|
||||||
|
}
|
||||||
|
if (handle->rdvs) {
|
||||||
|
free(handle->rdvs);
|
||||||
|
}
|
||||||
|
if (handle->rdao) {
|
||||||
|
free(handle->rdao);
|
||||||
|
}
|
||||||
|
if (handle->rdas) {
|
||||||
|
free(handle->rdas);
|
||||||
|
}
|
||||||
|
fclose(handle->fp);
|
||||||
|
free(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
long redcode_get_length(struct redcode_handle * handle)
|
||||||
|
{
|
||||||
|
return handle->rdvo[0]/4;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct redcode_frame * redcode_read_video_frame(
|
||||||
|
struct redcode_handle * handle, long frame)
|
||||||
|
{
|
||||||
|
struct redcode_frame * rv;
|
||||||
|
unsigned char * data;
|
||||||
|
|
||||||
|
if (frame > handle->rdvo[0]/4 || handle->rdvo[frame + 2] == 0) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
data = read_data(handle->fp, handle->rdvo[frame + 2], "REDV");
|
||||||
|
if (!data) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
rv = (struct redcode_frame*) calloc(1, sizeof(struct redcode_frame));
|
||||||
|
|
||||||
|
rv->offset = 12+8;
|
||||||
|
rv->length = *(unsigned long*)data - rv->offset;
|
||||||
|
rv->data = data;
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct redcode_frame * redcode_read_audio_frame(
|
||||||
|
struct redcode_handle * handle, long frame)
|
||||||
|
{
|
||||||
|
struct redcode_frame * rv;
|
||||||
|
unsigned char * data;
|
||||||
|
|
||||||
|
if (frame > handle->rdao[0]/4 || handle->rdao[frame + 2] == 0) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
data = read_data(handle->fp, handle->rdao[frame+2], "REDA");
|
||||||
|
if (!data) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
rv = (struct redcode_frame*) calloc(1, sizeof(struct redcode_frame));
|
||||||
|
|
||||||
|
rv->offset = 24+8;
|
||||||
|
rv->length = *(unsigned long*)data - rv->offset;
|
||||||
|
rv->data = data;
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
void redcode_free_frame(struct redcode_frame * frame)
|
||||||
|
{
|
||||||
|
free(frame->data);
|
||||||
|
free(frame);
|
||||||
|
}
|
||||||
24
extern/libredcode/format.h
vendored
Normal file
24
extern/libredcode/format.h
vendored
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
#ifndef __redcode_format_h_included__
|
||||||
|
#define __redcode_format_h_included__
|
||||||
|
|
||||||
|
struct redcode_handle;
|
||||||
|
struct redcode_frame {
|
||||||
|
unsigned long length;
|
||||||
|
unsigned long offset;
|
||||||
|
unsigned char * data;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct redcode_handle * redcode_open(const char * fname);
|
||||||
|
void redcode_close(struct redcode_handle * handle);
|
||||||
|
|
||||||
|
long redcode_get_length(struct redcode_handle * handle);
|
||||||
|
|
||||||
|
struct redcode_frame * redcode_read_video_frame(
|
||||||
|
struct redcode_handle * handle, long frame);
|
||||||
|
struct redcode_frame * redcode_read_audio_frame(
|
||||||
|
struct redcode_handle * handle, long frame);
|
||||||
|
|
||||||
|
void redcode_free_frame(struct redcode_frame * frame);
|
||||||
|
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -8,5 +8,5 @@ incs = 'intern ../container ../moto/include ../memutil'
|
|||||||
if (env['OURPLATFORM'] == 'win32-mingw'):
|
if (env['OURPLATFORM'] == 'win32-mingw'):
|
||||||
env.BlenderLib ('blender_BSP', sources, Split(incs), [], libtype=['common','intern'], priority=[26,26] )
|
env.BlenderLib ('blender_BSP', sources, Split(incs), [], libtype=['common','intern'], priority=[26,26] )
|
||||||
else:
|
else:
|
||||||
env.BlenderLib ('blender_BSP', sources, Split(incs), [], libtype='core', priority=15 )
|
env.BlenderLib ('blender_BSP', sources, Split(incs), [], libtype='core', priority=26 )
|
||||||
|
|
||||||
|
|||||||
Binary file not shown.
|
Before Width: | Height: | Size: 71 KiB After Width: | Height: | Size: 93 KiB |
File diff suppressed because it is too large
Load Diff
@@ -1,129 +0,0 @@
|
|||||||
#!BPY
|
|
||||||
"""
|
|
||||||
Name: 'UVs from unselected adjacent'
|
|
||||||
Blender: 242
|
|
||||||
Group: 'UVCalculation'
|
|
||||||
Tooltip: 'Assign UVs to selected faces from surrounding unselected faces.'
|
|
||||||
"""
|
|
||||||
__author__ = "Campbell Barton"
|
|
||||||
__url__ = ("blender", "elysiun")
|
|
||||||
__version__ = "1.0 2006/02/07"
|
|
||||||
|
|
||||||
__bpydoc__ = """\
|
|
||||||
This script sets the UV mapping and image of selected faces from adjacent unselected faces.
|
|
||||||
|
|
||||||
Use this script in face select mode for texturing between textured faces.
|
|
||||||
"""
|
|
||||||
|
|
||||||
# ***** BEGIN GPL LICENSE BLOCK *****
|
|
||||||
#
|
|
||||||
# Script copyright (C) Campbell J Barton
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or
|
|
||||||
# modify it under the terms of the GNU General Public License
|
|
||||||
# as published by the Free Software Foundation; either version 2
|
|
||||||
# of the License, or (at your option) any later version.
|
|
||||||
#
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
# GNU General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with this program; if not, write to the Free Software Foundation,
|
|
||||||
# Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
||||||
#
|
|
||||||
# ***** END GPL LICENCE BLOCK *****
|
|
||||||
# --------------------------------------------------------------------------
|
|
||||||
|
|
||||||
|
|
||||||
from Blender import *
|
|
||||||
import bpy
|
|
||||||
|
|
||||||
def mostUsedImage(imageList): # Returns the image most used in the list.
|
|
||||||
if not imageList:
|
|
||||||
return None
|
|
||||||
elif len(imageList) < 3:
|
|
||||||
return imageList[0]
|
|
||||||
|
|
||||||
# 3+ Images, Get the most used image for surrounding faces.
|
|
||||||
imageCount = {}
|
|
||||||
for image in imageList:
|
|
||||||
if image:
|
|
||||||
image_key= image.name
|
|
||||||
else:
|
|
||||||
image_key = None
|
|
||||||
|
|
||||||
try:
|
|
||||||
imageCount[image_key]['imageCount'] +=1 # an extra user of this image
|
|
||||||
except:
|
|
||||||
imageCount[image_key] = {'imageCount':1, 'blenderImage':image} # start with 1 user.
|
|
||||||
|
|
||||||
# Now a list of tuples, (imageName, {imageCount, image})
|
|
||||||
imageCount = imageCount.items()
|
|
||||||
|
|
||||||
try: imageCount.sort(key=lambda a: a[1])
|
|
||||||
except: imageCount.sort(lambda a,b: cmp(a[1], b[1]))
|
|
||||||
|
|
||||||
|
|
||||||
return imageCount[-1][1]['blenderImage']
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
sce = bpy.data.scenes.active
|
|
||||||
ob = sce.objects.active
|
|
||||||
|
|
||||||
if ob == None or ob.type != 'Mesh':
|
|
||||||
Draw.PupMenu('ERROR: No mesh object in face select mode.')
|
|
||||||
return
|
|
||||||
me = ob.getData(mesh=1)
|
|
||||||
|
|
||||||
if not me.faceUV:
|
|
||||||
Draw.PupMenu('ERROR: No mesh object in face select mode.')
|
|
||||||
return
|
|
||||||
|
|
||||||
selfaces = [f for f in me.faces if f.sel]
|
|
||||||
unselfaces = [f for f in me.faces if not f.sel]
|
|
||||||
|
|
||||||
|
|
||||||
# Gather per Vert UV and Image, store in vertUvAverage
|
|
||||||
vertUvAverage = [[[],[]] for i in xrange(len(me.verts))]
|
|
||||||
|
|
||||||
for f in unselfaces: # Unselected faces only.
|
|
||||||
fuv = f.uv
|
|
||||||
for i,v in enumerate(f):
|
|
||||||
vertUvAverage[v.index][0].append(fuv[i])
|
|
||||||
vertUvAverage[v.index][1].append(f.image)
|
|
||||||
|
|
||||||
# Average per vectex UV coords
|
|
||||||
for vertUvData in vertUvAverage:
|
|
||||||
uvList = vertUvData[0]
|
|
||||||
if uvList:
|
|
||||||
# Convert from a list of vectors into 1 vector.
|
|
||||||
vertUvData[0] = reduce(lambda a,b: a+b, uvList, Mathutils.Vector(0,0)) * (1.0/len(uvList))
|
|
||||||
else:
|
|
||||||
vertUvData[0] = None
|
|
||||||
|
|
||||||
# Assign to selected faces
|
|
||||||
TEX_FLAG = Mesh.FaceModes['TEX']
|
|
||||||
for f in selfaces:
|
|
||||||
uvlist = []
|
|
||||||
imageList = []
|
|
||||||
for i,v in enumerate(f):
|
|
||||||
uv, vImages = vertUvAverage[v.index]
|
|
||||||
uvlist.append( uv )
|
|
||||||
imageList.extend(vImages)
|
|
||||||
|
|
||||||
if None not in uvlist:
|
|
||||||
# all the faces images used by this faces vert. some faces will be added twice but thats ok.
|
|
||||||
# Get the most used image and assign to the face.
|
|
||||||
image = mostUsedImage(imageList)
|
|
||||||
f.uv = uvlist
|
|
||||||
|
|
||||||
if image:
|
|
||||||
f.image = image
|
|
||||||
f.mode |= TEX_FLAG
|
|
||||||
Window.RedrawAll()
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
|
||||||
main()
|
|
||||||
@@ -1,12 +1,31 @@
|
|||||||
#!BPY
|
#!BPY
|
||||||
"""
|
"""
|
||||||
Name: 'Seams from Islands'
|
Name: 'Seams from Islands'
|
||||||
Blender: 243
|
Blender: 246
|
||||||
Group: 'UV'
|
Group: 'UV'
|
||||||
Tooltip: 'Add seams onto the mesh at the bounds of UV islands'
|
Tooltip: 'Add seams onto the mesh at the bounds of UV islands'
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Add a licence here if you wish to re-distribute, we recommend the GPL
|
# ***** BEGIN GPL LICENSE BLOCK *****
|
||||||
|
#
|
||||||
|
# Script copyright (C) Campbell Barton
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the terms of the GNU General Public License
|
||||||
|
# as published by the Free Software Foundation; either version 2
|
||||||
|
# of the License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program; if not, write to the Free Software Foundation,
|
||||||
|
# Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||||||
|
#
|
||||||
|
# ***** END GPL LICENCE BLOCK *****
|
||||||
|
# --------------------------------------------------------------------------
|
||||||
|
|
||||||
from Blender import Scene, Mesh, Window, sys
|
from Blender import Scene, Mesh, Window, sys
|
||||||
import BPyMessages
|
import BPyMessages
|
||||||
@@ -37,8 +56,11 @@ def seams_from_islands(me):
|
|||||||
# add seams
|
# add seams
|
||||||
SEAM = Mesh.EdgeFlags.SEAM
|
SEAM = Mesh.EdgeFlags.SEAM
|
||||||
for ed in me.edges:
|
for ed in me.edges:
|
||||||
|
try: # the edge might not be in a face
|
||||||
if len(set(edge_uvs[ed.key])) > 1:
|
if len(set(edge_uvs[ed.key])) > 1:
|
||||||
ed.flag |= SEAM
|
ed.flag |= SEAM
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
|
|
||||||
|
|||||||
@@ -41,6 +41,12 @@ import BPyMesh
|
|||||||
|
|
||||||
from math import sqrt
|
from math import sqrt
|
||||||
|
|
||||||
|
def AngleBetweenVecs(a1,a2):
|
||||||
|
try:
|
||||||
|
return Mathutils.AngleBetweenVecs(a1,a2)
|
||||||
|
except:
|
||||||
|
return 180.0
|
||||||
|
|
||||||
class prettyface(object):
|
class prettyface(object):
|
||||||
__slots__ = 'uv', 'width', 'height', 'children', 'xoff', 'yoff', 'has_parent', 'rot'
|
__slots__ = 'uv', 'width', 'height', 'children', 'xoff', 'yoff', 'has_parent', 'rot'
|
||||||
def __init__(self, data):
|
def __init__(self, data):
|
||||||
@@ -148,9 +154,9 @@ class prettyface(object):
|
|||||||
if len(uv) == 2:
|
if len(uv) == 2:
|
||||||
# match the order of angle sizes of the 3d verts with the UV angles and rotate.
|
# match the order of angle sizes of the 3d verts with the UV angles and rotate.
|
||||||
def get_tri_angles(v1,v2,v3):
|
def get_tri_angles(v1,v2,v3):
|
||||||
a1= Mathutils.AngleBetweenVecs(v2-v1,v3-v1)
|
a1= AngleBetweenVecs(v2-v1,v3-v1)
|
||||||
a2= Mathutils.AngleBetweenVecs(v1-v2,v3-v2)
|
a2= AngleBetweenVecs(v1-v2,v3-v2)
|
||||||
a3 = 180 - (a1+a2) #a3= Mathutils.AngleBetweenVecs(v2-v3,v1-v3)
|
a3 = 180 - (a1+a2) #a3= AngleBetweenVecs(v2-v3,v1-v3)
|
||||||
|
|
||||||
|
|
||||||
return [(a1,0),(a2,1),(a3,2)]
|
return [(a1,0),(a2,1),(a3,2)]
|
||||||
@@ -237,8 +243,17 @@ PREF_MARGIN_DIV= 512):
|
|||||||
face_groups.append(faces)
|
face_groups.append(faces)
|
||||||
|
|
||||||
if PREF_NEW_UVLAYER:
|
if PREF_NEW_UVLAYER:
|
||||||
me.addUVLayer('lightmap')
|
uvname_org = uvname = 'lightmap'
|
||||||
me.activeUVLayer = 'lightmap'
|
uvnames = me.getUVLayerNames()
|
||||||
|
i = 1
|
||||||
|
while uvname in uvnames:
|
||||||
|
uvname = '%s.%03d' % (uvname_org, i)
|
||||||
|
i+=1
|
||||||
|
|
||||||
|
me.addUVLayer(uvname)
|
||||||
|
me.activeUVLayer = uvname
|
||||||
|
|
||||||
|
del uvnames, uvname_org, uvname
|
||||||
|
|
||||||
for face_sel in face_groups:
|
for face_sel in face_groups:
|
||||||
print "\nStarting unwrap"
|
print "\nStarting unwrap"
|
||||||
@@ -402,11 +417,14 @@ PREF_MARGIN_DIV= 512):
|
|||||||
# ...limiting this is needed or you end up with bug unused texture spaces
|
# ...limiting this is needed or you end up with bug unused texture spaces
|
||||||
# ...however if its too high, boxpacking is way too slow for high poly meshes.
|
# ...however if its too high, boxpacking is way too slow for high poly meshes.
|
||||||
float_to_int_factor = lengths_to_ints[0][0]
|
float_to_int_factor = lengths_to_ints[0][0]
|
||||||
|
if float_to_int_factor > 0:
|
||||||
max_int_dimension = int(((side_len / float_to_int_factor)) / PREF_BOX_DIV)
|
max_int_dimension = int(((side_len / float_to_int_factor)) / PREF_BOX_DIV)
|
||||||
|
ok = True
|
||||||
|
else:
|
||||||
|
max_int_dimension = 0.0 # wont be used
|
||||||
|
ok = False
|
||||||
|
|
||||||
# RECURSIVE prettyface grouping
|
# RECURSIVE prettyface grouping
|
||||||
ok = True
|
|
||||||
while ok:
|
while ok:
|
||||||
ok = False
|
ok = False
|
||||||
|
|
||||||
|
|||||||
@@ -32,7 +32,7 @@ Name "Blender VERSION"
|
|||||||
|
|
||||||
!insertmacro MUI_PAGE_DIRECTORY
|
!insertmacro MUI_PAGE_DIRECTORY
|
||||||
Page custom DataLocation DataLocationOnLeave
|
Page custom DataLocation DataLocationOnLeave
|
||||||
Page custom AppDataChoice AppDataChoiceOnLeave
|
;Page custom AppDataChoice AppDataChoiceOnLeave
|
||||||
Page custom PreMigrateUserSettings MigrateUserSettings
|
Page custom PreMigrateUserSettings MigrateUserSettings
|
||||||
!insertmacro MUI_PAGE_INSTFILES
|
!insertmacro MUI_PAGE_INSTFILES
|
||||||
!insertmacro MUI_PAGE_FINISH
|
!insertmacro MUI_PAGE_FINISH
|
||||||
@@ -271,7 +271,9 @@ Function DataLocationOnLeave
|
|||||||
StrCpy $SETUSERCONTEXT "false"
|
StrCpy $SETUSERCONTEXT "false"
|
||||||
${NSD_GetState} $HWND_APPDATA $R0
|
${NSD_GetState} $HWND_APPDATA $R0
|
||||||
${If} $R0 == "1"
|
${If} $R0 == "1"
|
||||||
StrCpy $SETUSERCONTEXT "true"
|
; FIXME: disabled 'all users' until fully multi-user compatible
|
||||||
|
;StrCpy $SETUSERCONTEXT "true"
|
||||||
|
Call SetWinXPPathCurrentUser
|
||||||
${Else}
|
${Else}
|
||||||
${NSD_GetState} $HWND_INSTDIR $R0
|
${NSD_GetState} $HWND_INSTDIR $R0
|
||||||
${If} $R0 == "1"
|
${If} $R0 == "1"
|
||||||
|
|||||||
@@ -40,16 +40,35 @@
|
|||||||
#include "DNA_listBase.h"
|
#include "DNA_listBase.h"
|
||||||
#include "BLI_ghash.h"
|
#include "BLI_ghash.h"
|
||||||
#include "BLI_memarena.h"
|
#include "BLI_memarena.h"
|
||||||
#include "DNA_customdata_types.h"
|
#include "DNA_image_types.h"
|
||||||
#include "BLI_editVert.h"
|
#include "BLI_editVert.h"
|
||||||
#include "BKE_DerivedMesh.h"
|
#include "BKE_DerivedMesh.h"
|
||||||
#include "transform.h"
|
#include "transform.h"
|
||||||
|
#include "BKE_bmeshCustomData.h"
|
||||||
|
|
||||||
|
/*forward declerations*/
|
||||||
struct BME_Vert;
|
struct BME_Vert;
|
||||||
struct BME_Edge;
|
struct BME_Edge;
|
||||||
struct BME_Poly;
|
struct BME_Poly;
|
||||||
struct BME_Loop;
|
struct BME_Loop;
|
||||||
|
|
||||||
|
|
||||||
|
/*structure for fast memory allocation/frees*/
|
||||||
|
typedef struct BME_mempool{
|
||||||
|
struct ListBase chunks;
|
||||||
|
int esize, csize, pchunk; /*size of elements and chunks in bytes and number of elements per chunk*/
|
||||||
|
struct BME_freenode *free; /*free element list. Interleaved into chunk datas.*/
|
||||||
|
}BME_mempool;
|
||||||
|
|
||||||
|
/*Notes on further structure Cleanup:
|
||||||
|
-Remove the tflags, they belong in custom data layers
|
||||||
|
-Remove the eflags completely, they are mostly not used
|
||||||
|
-Remove the selection/vis/bevel weight flag/values ect and move them to custom data
|
||||||
|
-Remove EID member and move to custom data
|
||||||
|
-Add a radial cycle length, disk cycle length and loop cycle lenght attributes to custom data and have eulers maintain/use them if present.
|
||||||
|
-Move data such as vertex coordinates/normals to custom data and leave pointers in structures to active layer data.
|
||||||
|
-Remove BME_CycleNode structure?
|
||||||
|
*/
|
||||||
typedef struct BME_CycleNode{
|
typedef struct BME_CycleNode{
|
||||||
struct BME_CycleNode *next, *prev;
|
struct BME_CycleNode *next, *prev;
|
||||||
void *data;
|
void *data;
|
||||||
@@ -57,16 +76,21 @@ typedef struct BME_CycleNode{
|
|||||||
|
|
||||||
typedef struct BME_Mesh
|
typedef struct BME_Mesh
|
||||||
{
|
{
|
||||||
ListBase verts, edges, polys, loops;
|
ListBase verts, edges, polys;
|
||||||
int totvert, totedge, totpoly, totloop; /*record keeping*/
|
/*memory pools used for storing mesh elements*/
|
||||||
int nextv, nexte, nextp, nextl; /*Next element ID for verts/edges/faces/loops. Never reused*/
|
struct BME_mempool *vpool;
|
||||||
struct CustomData vdata, edata, pdata, ldata; /*Custom Data Layer information*/
|
struct BME_mempool *epool;
|
||||||
|
struct BME_mempool *ppool;
|
||||||
|
struct BME_mempool *lpool;
|
||||||
/*some scratch arrays used by eulers*/
|
/*some scratch arrays used by eulers*/
|
||||||
struct BME_Vert **vtar;
|
struct BME_Vert **vtar;
|
||||||
struct BME_Edge **edar;
|
struct BME_Edge **edar;
|
||||||
struct BME_Loop **lpar;
|
struct BME_Loop **lpar;
|
||||||
struct BME_Poly **plar;
|
struct BME_Poly **plar;
|
||||||
int vtarlen, edarlen, lparlen, plarlen;
|
int vtarlen, edarlen, lparlen, plarlen;
|
||||||
|
int totvert, totedge, totpoly, totloop; /*record keeping*/
|
||||||
|
int nextv, nexte, nextp, nextl; /*Next element ID for verts/edges/faces/loops. Never reused*/
|
||||||
|
struct BME_CustomData vdata, edata, pdata, ldata; /*Custom Data Layer information*/
|
||||||
} BME_Mesh;
|
} BME_Mesh;
|
||||||
|
|
||||||
typedef struct BME_Vert
|
typedef struct BME_Vert
|
||||||
@@ -102,7 +126,6 @@ typedef struct BME_Loop
|
|||||||
struct BME_Loop *next, *prev; /*circularly linked list around face*/
|
struct BME_Loop *next, *prev; /*circularly linked list around face*/
|
||||||
int EID;
|
int EID;
|
||||||
struct BME_CycleNode radial; /*circularly linked list used to find faces around an edge*/
|
struct BME_CycleNode radial; /*circularly linked list used to find faces around an edge*/
|
||||||
struct BME_CycleNode *gref; /*pointer to loop ref. Nasty.*/
|
|
||||||
struct BME_Vert *v; /*vertex that this loop starts at.*/
|
struct BME_Vert *v; /*vertex that this loop starts at.*/
|
||||||
struct BME_Edge *e; /*edge this loop belongs to*/
|
struct BME_Edge *e; /*edge this loop belongs to*/
|
||||||
struct BME_Poly *f; /*face this loop belongs to*/
|
struct BME_Poly *f; /*face this loop belongs to*/
|
||||||
@@ -124,7 +147,7 @@ typedef struct BME_Poly
|
|||||||
unsigned short flag, h, mat_nr;
|
unsigned short flag, h, mat_nr;
|
||||||
} BME_Poly;
|
} BME_Poly;
|
||||||
|
|
||||||
//*EDGE UTILITIES*/
|
/*EDGE UTILITIES*/
|
||||||
int BME_verts_in_edge(struct BME_Vert *v1, struct BME_Vert *v2, struct BME_Edge *e);
|
int BME_verts_in_edge(struct BME_Vert *v1, struct BME_Vert *v2, struct BME_Edge *e);
|
||||||
int BME_vert_in_edge(struct BME_Edge *e, BME_Vert *v);
|
int BME_vert_in_edge(struct BME_Edge *e, BME_Vert *v);
|
||||||
struct BME_Vert *BME_edge_getothervert(struct BME_Edge *e, struct BME_Vert *v);
|
struct BME_Vert *BME_edge_getothervert(struct BME_Edge *e, struct BME_Vert *v);
|
||||||
@@ -146,7 +169,7 @@ int BME_radial_find_face(struct BME_Edge *e,struct BME_Poly *f);
|
|||||||
struct BME_Loop *BME_loop_find_loop(struct BME_Poly *f, struct BME_Vert *v);
|
struct BME_Loop *BME_loop_find_loop(struct BME_Poly *f, struct BME_Vert *v);
|
||||||
|
|
||||||
/*MESH CREATION/DESTRUCTION*/
|
/*MESH CREATION/DESTRUCTION*/
|
||||||
struct BME_Mesh *BME_make_mesh(void);
|
struct BME_Mesh *BME_make_mesh(int allocsize[4], struct BME_CustomDataInit init[4]);
|
||||||
void BME_free_mesh(struct BME_Mesh *bm);
|
void BME_free_mesh(struct BME_Mesh *bm);
|
||||||
/*FULL MESH VALIDATION*/
|
/*FULL MESH VALIDATION*/
|
||||||
int BME_validate_mesh(struct BME_Mesh *bm, int halt);
|
int BME_validate_mesh(struct BME_Mesh *bm, int halt);
|
||||||
@@ -230,8 +253,8 @@ float *BME_bevel_calc_polynormal(struct BME_Poly *f, struct BME_TransData_Head *
|
|||||||
struct BME_Mesh *BME_bevel(struct BME_Mesh *bm, float value, int res, int options, int defgrp_index, float angle, BME_TransData_Head **rtd);
|
struct BME_Mesh *BME_bevel(struct BME_Mesh *bm, float value, int res, int options, int defgrp_index, float angle, BME_TransData_Head **rtd);
|
||||||
|
|
||||||
/*CONVERSION FUNCTIONS*/
|
/*CONVERSION FUNCTIONS*/
|
||||||
struct BME_Mesh *BME_editmesh_to_bmesh(EditMesh *em, struct BME_Mesh *bm);
|
struct BME_Mesh *BME_editmesh_to_bmesh(EditMesh *em);
|
||||||
struct EditMesh *BME_bmesh_to_editmesh(struct BME_Mesh *bm, BME_TransData_Head *td);
|
struct EditMesh *BME_bmesh_to_editmesh(struct BME_Mesh *bm, BME_TransData_Head *td);
|
||||||
struct BME_Mesh *BME_derivedmesh_to_bmesh(struct DerivedMesh *dm, struct BME_Mesh *bm);
|
struct BME_Mesh *BME_derivedmesh_to_bmesh(struct DerivedMesh *dm);
|
||||||
struct DerivedMesh *BME_bmesh_to_derivedmesh(struct BME_Mesh *bm, struct DerivedMesh *dm);
|
struct DerivedMesh *BME_bmesh_to_derivedmesh(struct BME_Mesh *bm, struct DerivedMesh *dm);
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
108
source/blender/blenkernel/BKE_bmeshCustomData.h
Normal file
108
source/blender/blenkernel/BKE_bmeshCustomData.h
Normal file
@@ -0,0 +1,108 @@
|
|||||||
|
/**
|
||||||
|
* BKE_bmesh.h jan 2007
|
||||||
|
*
|
||||||
|
* BMesh modeler structure and functions.
|
||||||
|
*
|
||||||
|
* $Id: BKE_bmesh.h,v 1.00 2007/01/17 17:42:01 Briggs Exp $
|
||||||
|
*
|
||||||
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version. The Blender
|
||||||
|
* Foundation also sells licenses for use in proprietary software under
|
||||||
|
* the Blender License. See http://www.blender.org/BL/ for information
|
||||||
|
* about this.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||||||
|
*
|
||||||
|
* The Original Code is Copyright (C) 2004 Blender Foundation.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* The Original Code is: all of this file.
|
||||||
|
*
|
||||||
|
* Contributor(s): Geoffrey Bantle.
|
||||||
|
*
|
||||||
|
* ***** END GPL LICENSE BLOCK *****
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef BKE_BMESHCUSTOMDATA_H
|
||||||
|
#define BKE_BMESHCUSTOMDATA_H
|
||||||
|
|
||||||
|
struct BME_mempool;
|
||||||
|
|
||||||
|
/*Custom Data Types and defines
|
||||||
|
Eventual plan is to move almost everything to custom data and let caller
|
||||||
|
decide when making the mesh what layers they want to store in the mesh
|
||||||
|
|
||||||
|
This stuff should probably go in a seperate file....
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define BME_CD_FACETEX 0 /*Image texture/texface*/
|
||||||
|
#define BME_CD_LOOPTEX 1 /*UV coordinates*/
|
||||||
|
#define BME_CD_LOOPCOL 2 /*Vcolors*/
|
||||||
|
#define BME_CD_DEFORMVERT 3 /*Vertex Group/Weights*/
|
||||||
|
#define BME_CD_NUMTYPES 4
|
||||||
|
|
||||||
|
typedef struct BME_CustomDataLayer {
|
||||||
|
int type; /* type of data in layer */
|
||||||
|
int offset; /* offset of layer in block */
|
||||||
|
int active; /* offset of active layer*/
|
||||||
|
char name[32]; /* layer name */
|
||||||
|
} BME_CustomDataLayer;
|
||||||
|
|
||||||
|
typedef struct BME_CustomData {
|
||||||
|
struct BME_CustomDataLayer *layers; /*Custom Data Layers*/
|
||||||
|
struct BME_mempool *pool; /*pool for alloc of blocks*/
|
||||||
|
int totlayer, totsize; /*total layers and total size in bytes of each block*/
|
||||||
|
} BME_CustomData;
|
||||||
|
|
||||||
|
typedef struct BME_CustomDataInit{
|
||||||
|
int layout[BME_CD_NUMTYPES];
|
||||||
|
int active[BME_CD_NUMTYPES];
|
||||||
|
int totlayers;
|
||||||
|
char *nametemplate;
|
||||||
|
} BME_CustomDataInit;
|
||||||
|
|
||||||
|
/*Custom data types*/
|
||||||
|
typedef struct BME_DeformWeight {
|
||||||
|
int def_nr;
|
||||||
|
float weight;
|
||||||
|
} BME_DeformWeight;
|
||||||
|
|
||||||
|
typedef struct BME_DeformVert {
|
||||||
|
struct BME_DeformWeight *dw;
|
||||||
|
int totweight;
|
||||||
|
} BME_DeformVert;
|
||||||
|
|
||||||
|
typedef struct BME_facetex{
|
||||||
|
struct Image *tpage;
|
||||||
|
char flag, transp;
|
||||||
|
short mode, tile, unwrap;
|
||||||
|
}BME_facetex;
|
||||||
|
|
||||||
|
typedef struct BME_looptex{
|
||||||
|
float u, v;
|
||||||
|
}BME_looptex;
|
||||||
|
|
||||||
|
typedef struct BME_loopcol{
|
||||||
|
char r, g, b, a;
|
||||||
|
}BME_loopcol;
|
||||||
|
|
||||||
|
/*CUSTOM DATA API*/
|
||||||
|
void BME_CD_Create(struct BME_CustomData *data, struct BME_CustomDataInit *init, int initalloc);
|
||||||
|
void BME_CD_Free(struct BME_CustomData *data);
|
||||||
|
void BME_CD_free_block(struct BME_CustomData *data, void **block);
|
||||||
|
void BME_CD_copy_data(const struct BME_CustomData *source, struct BME_CustomData *dest, void *src_block, void **dest_block);
|
||||||
|
void BME_CD_set_default(struct BME_CustomData *data, void **block);
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -24,14 +24,14 @@
|
|||||||
*
|
*
|
||||||
* The Original Code is: all of this file.
|
* The Original Code is: all of this file.
|
||||||
*
|
*
|
||||||
* Contributor(s): none yet.
|
* Contributor(s): Daniel Genrich
|
||||||
*
|
*
|
||||||
* ***** END GPL LICENSE BLOCK *****
|
* ***** END GPL LICENSE BLOCK *****
|
||||||
*/
|
*/
|
||||||
#ifndef BKE_CLOTH_H
|
#ifndef BKE_CLOTH_H
|
||||||
#define BKE_CLOTH_H
|
#define BKE_CLOTH_H
|
||||||
|
|
||||||
#include "float.h"
|
#include <float.h>
|
||||||
|
|
||||||
#include "BLI_linklist.h"
|
#include "BLI_linklist.h"
|
||||||
#include "BKE_customdata.h"
|
#include "BKE_customdata.h"
|
||||||
@@ -102,7 +102,8 @@ typedef struct Cloth
|
|||||||
unsigned char old_solver_type; /* unused, only 1 solver here */
|
unsigned char old_solver_type; /* unused, only 1 solver here */
|
||||||
unsigned char pad2;
|
unsigned char pad2;
|
||||||
short pad3;
|
short pad3;
|
||||||
struct BVH *tree; /* collision tree for this cloth object */
|
struct BVHTree *bvhtree; /* collision tree for this cloth object */
|
||||||
|
struct BVHTree *bvhselftree; /* collision tree for this cloth object */
|
||||||
struct MFace *mfaces;
|
struct MFace *mfaces;
|
||||||
struct Implicit_Data *implicit; /* our implicit solver connects to this pointer */
|
struct Implicit_Data *implicit; /* our implicit solver connects to this pointer */
|
||||||
struct Implicit_Data *implicitEM; /* our implicit solver connects to this pointer */
|
struct Implicit_Data *implicitEM; /* our implicit solver connects to this pointer */
|
||||||
@@ -171,17 +172,10 @@ ClothSpring;
|
|||||||
/* These are the bits used in SimSettings.flags. */
|
/* These are the bits used in SimSettings.flags. */
|
||||||
typedef enum
|
typedef enum
|
||||||
{
|
{
|
||||||
//CLOTH_SIMSETTINGS_FLAG_RESET = ( 1 << 1 ), // The CM object requires a reinitializaiton.
|
|
||||||
CLOTH_SIMSETTINGS_FLAG_COLLOBJ = ( 1 << 2 ),// object is only collision object, no cloth simulation is done
|
CLOTH_SIMSETTINGS_FLAG_COLLOBJ = ( 1 << 2 ),// object is only collision object, no cloth simulation is done
|
||||||
CLOTH_SIMSETTINGS_FLAG_GOAL = ( 1 << 3 ), // we have goals enabled
|
CLOTH_SIMSETTINGS_FLAG_GOAL = ( 1 << 3 ), // we have goals enabled
|
||||||
CLOTH_SIMSETTINGS_FLAG_TEARING = ( 1 << 4 ),// true if tearing is enabled
|
CLOTH_SIMSETTINGS_FLAG_TEARING = ( 1 << 4 ),// true if tearing is enabled
|
||||||
//CLOTH_SIMSETTINGS_FLAG_CCACHE_PROTECT = ( 1 << 5 ), // true if tearing is enabled
|
|
||||||
//CLOTH_SIMSETTINGS_FLAG_EDITMODE = ( 1 << 6 ), // are we in editmode? -several things disabled
|
|
||||||
//CLOTH_SIMSETTINGS_FLAG_CCACHE_FFREE = ( 1 << 7 ), /* force cache freeing */
|
|
||||||
CLOTH_SIMSETTINGS_FLAG_SCALING = ( 1 << 8 ), /* is advanced scaling active? */
|
CLOTH_SIMSETTINGS_FLAG_SCALING = ( 1 << 8 ), /* is advanced scaling active? */
|
||||||
//CLOTH_SIMSETTINGS_FLAG_LOADED = ( 1 << 9 ), /* did we just got load? */
|
|
||||||
//CLOTH_SIMSETTINGS_FLAG_AUTOPROTECT = ( 1 << 10 ), /* is autoprotect enabled? */
|
|
||||||
//CLOTH_SIMSETTINGS_FLAG_CCACHE_OUTDATED = (1 << 11), /* while protected, did cache get outdated? */
|
|
||||||
CLOTH_SIMSETTINGS_FLAG_CCACHE_EDIT = (1 << 12) /* edit cache in editmode */
|
CLOTH_SIMSETTINGS_FLAG_CCACHE_EDIT = (1 << 12) /* edit cache in editmode */
|
||||||
} CLOTH_SIMSETTINGS_FLAGS;
|
} CLOTH_SIMSETTINGS_FLAGS;
|
||||||
|
|
||||||
@@ -208,6 +202,7 @@ typedef enum
|
|||||||
CLOTH_SPRING_FLAG_NEEDED = ( 1 << 2 ), // springs has values to be applied
|
CLOTH_SPRING_FLAG_NEEDED = ( 1 << 2 ), // springs has values to be applied
|
||||||
} CLOTH_SPRINGS_FLAGS;
|
} CLOTH_SPRINGS_FLAGS;
|
||||||
|
|
||||||
|
|
||||||
/////////////////////////////////////////////////
|
/////////////////////////////////////////////////
|
||||||
// collision.c
|
// collision.c
|
||||||
////////////////////////////////////////////////
|
////////////////////////////////////////////////
|
||||||
@@ -246,7 +241,8 @@ DerivedMesh *clothModifier_do ( ClothModifierData *clmd,Object *ob, DerivedMesh
|
|||||||
void cloth_update_normals ( ClothVertex *verts, int nVerts, MFace *face, int totface );
|
void cloth_update_normals ( ClothVertex *verts, int nVerts, MFace *face, int totface );
|
||||||
|
|
||||||
// needed for collision.c
|
// needed for collision.c
|
||||||
void bvh_update_from_cloth ( ClothModifierData *clmd, int moving );
|
void bvhtree_update_from_cloth ( ClothModifierData *clmd, int moving );
|
||||||
|
void bvhselftree_update_from_cloth ( ClothModifierData *clmd, int moving );
|
||||||
|
|
||||||
// needed for editmesh.c
|
// needed for editmesh.c
|
||||||
void cloth_write_cache ( Object *ob, ClothModifierData *clmd, float framenr );
|
void cloth_write_cache ( Object *ob, ClothModifierData *clmd, float framenr );
|
||||||
@@ -261,11 +257,6 @@ int cloth_add_spring ( ClothModifierData *clmd, unsigned int indexA, unsigned in
|
|||||||
////////////////////////////////////////////////
|
////////////////////////////////////////////////
|
||||||
|
|
||||||
|
|
||||||
/* Typedefs for function pointers we need for solvers and collision detection. */
|
|
||||||
typedef void ( *CM_COLLISION_SELF ) ( ClothModifierData *clmd, int step );
|
|
||||||
typedef void ( *CM_COLLISION_OBJ ) ( ClothModifierData *clmd, int step, CM_COLLISION_RESPONSE collision_response );
|
|
||||||
|
|
||||||
|
|
||||||
/* This enum provides the IDs for our solvers. */
|
/* This enum provides the IDs for our solvers. */
|
||||||
// only one available in the moment
|
// only one available in the moment
|
||||||
typedef enum
|
typedef enum
|
||||||
@@ -286,15 +277,6 @@ typedef struct
|
|||||||
}
|
}
|
||||||
CM_SOLVER_DEF;
|
CM_SOLVER_DEF;
|
||||||
|
|
||||||
/* used for caching in implicit.c */
|
|
||||||
typedef struct Frame
|
|
||||||
{
|
|
||||||
ClothVertex *verts;
|
|
||||||
ClothSpring *springs;
|
|
||||||
unsigned int numverts, numsprings;
|
|
||||||
float time; /* we need float since we want to support sub-frames */
|
|
||||||
}
|
|
||||||
Frame;
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -24,7 +24,7 @@
|
|||||||
*
|
*
|
||||||
* The Original Code is: all of this file.
|
* The Original Code is: all of this file.
|
||||||
*
|
*
|
||||||
* Contributor(s): none yet.
|
* Contributor(s): Daniel Genrich
|
||||||
*
|
*
|
||||||
* ***** END GPL LICENSE BLOCK *****
|
* ***** END GPL LICENSE BLOCK *****
|
||||||
*/
|
*/
|
||||||
@@ -32,7 +32,7 @@
|
|||||||
#define BKE_COLLISIONS_H
|
#define BKE_COLLISIONS_H
|
||||||
|
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
#include "float.h"
|
#include <float.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
@@ -47,68 +47,27 @@
|
|||||||
#include "DNA_modifier_types.h"
|
#include "DNA_modifier_types.h"
|
||||||
#include "DNA_object_types.h"
|
#include "DNA_object_types.h"
|
||||||
|
|
||||||
|
#include "BLI_kdopbvh.h"
|
||||||
|
|
||||||
struct Object;
|
struct Object;
|
||||||
struct Cloth;
|
struct Cloth;
|
||||||
struct MFace;
|
struct MFace;
|
||||||
struct DerivedMesh;
|
struct DerivedMesh;
|
||||||
struct ClothModifierData;
|
struct ClothModifierData;
|
||||||
struct CollisionTree;
|
|
||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////
|
////////////////////////////////////////
|
||||||
// used in kdop.c and collision.c
|
// used for collisions in collision.c
|
||||||
////////////////////////////////////////
|
////////////////////////////////////////
|
||||||
typedef struct CollisionTree
|
|
||||||
|
/* COLLISION FLAGS */
|
||||||
|
typedef enum
|
||||||
{
|
{
|
||||||
struct CollisionTree *nodes[4]; // 4 children --> quad-tree
|
COLLISION_IN_FUTURE = ( 1 << 1 ),
|
||||||
struct CollisionTree *parent;
|
} COLLISION_FLAGS;
|
||||||
struct CollisionTree *nextLeaf;
|
|
||||||
struct CollisionTree *prevLeaf;
|
|
||||||
float bv[26]; // Bounding volume of all nodes / we have 7 axes on a 14-DOP
|
|
||||||
unsigned int tri_index; // this saves the index of the face
|
|
||||||
// int point_index[4]; // supports up to 4 points in a leaf
|
|
||||||
int count_nodes; // how many nodes are used
|
|
||||||
int traversed; // how many nodes already traversed until this level?
|
|
||||||
int isleaf;
|
|
||||||
float alpha; /* for selfcollision */
|
|
||||||
float normal[3]; /* for selfcollision */
|
|
||||||
}
|
|
||||||
CollisionTree;
|
|
||||||
|
|
||||||
typedef struct BVH
|
|
||||||
{
|
|
||||||
unsigned int numfaces;
|
|
||||||
unsigned int numverts;
|
|
||||||
MVert *current_x; // e.g. txold in clothvertex
|
|
||||||
MVert *current_xold; // e.g. tx in clothvertex
|
|
||||||
MFace *mfaces; // just a pointer to the original datastructure
|
|
||||||
struct LinkNode *tree;
|
|
||||||
CollisionTree *root; // TODO: saving the root --> is this really needed? YES!
|
|
||||||
CollisionTree *leaf_tree; /* Tail of the leaf linked list. */
|
|
||||||
CollisionTree *leaf_root; /* Head of the leaf linked list. */
|
|
||||||
float epsilon; /* epslion is used for inflation of the k-dop */
|
|
||||||
int flags; /* bvhFlags */
|
|
||||||
}
|
|
||||||
BVH;
|
|
||||||
////////////////////////////////////////
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////
|
|
||||||
// kdop.c
|
|
||||||
////////////////////////////////////////
|
|
||||||
|
|
||||||
// needed for collision.c
|
|
||||||
typedef void ( *CM_COLLISION_RESPONSE ) ( ModifierData *md1, ModifierData *md2, CollisionTree *tree1, CollisionTree *tree2 );
|
|
||||||
|
|
||||||
// needed for collision.c
|
|
||||||
int bvh_traverse ( ModifierData * md1, ModifierData * md2, CollisionTree * tree1, CollisionTree * tree2, float step, CM_COLLISION_RESPONSE collision_response, int selfcollision);
|
|
||||||
|
|
||||||
////////////////////////////////////////
|
|
||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////
|
////////////////////////////////////////
|
||||||
// used for collisions in kdop.c and also collision.c
|
// used for collisions in collision.c
|
||||||
////////////////////////////////////////
|
////////////////////////////////////////
|
||||||
/* used for collisions in collision.c */
|
/* used for collisions in collision.c */
|
||||||
typedef struct CollPair
|
typedef struct CollPair
|
||||||
@@ -119,10 +78,10 @@ typedef struct CollPair
|
|||||||
float normal[3];
|
float normal[3];
|
||||||
float vector[3]; // unnormalized collision vector: p2-p1
|
float vector[3]; // unnormalized collision vector: p2-p1
|
||||||
float pa[3], pb[3]; // collision point p1 on face1, p2 on face2
|
float pa[3], pb[3]; // collision point p1 on face1, p2 on face2
|
||||||
int lastsign; // indicates if the distance sign has changed, unused itm
|
int flag;
|
||||||
float time; // collision time, from 0 up to 1
|
float time; // collision time, from 0 up to 1
|
||||||
unsigned int ap1, ap2, ap3, bp1, bp2, bp3;
|
int ap1, ap2, ap3, bp1, bp2, bp3;
|
||||||
unsigned int pointsb[4];
|
int pointsb[4];
|
||||||
}
|
}
|
||||||
CollPair;
|
CollPair;
|
||||||
|
|
||||||
@@ -157,22 +116,12 @@ FaceCollPair;
|
|||||||
// forward declarations
|
// forward declarations
|
||||||
/////////////////////////////////////////////////
|
/////////////////////////////////////////////////
|
||||||
|
|
||||||
// NOTICE: mvert-routines for building + update the BVH are the most native ones
|
/////////////////////////////////////////////////
|
||||||
|
// used in modifier.c from collision.c
|
||||||
// builds bounding volume hierarchy
|
/////////////////////////////////////////////////
|
||||||
void bvh_build (BVH *bvh);
|
BVHTree *bvhtree_build_from_mvert ( MFace *mfaces, unsigned int numfaces, MVert *x, unsigned int numverts, float epsilon );
|
||||||
BVH *bvh_build_from_mvert (MFace *mfaces, unsigned int numfaces, MVert *x, unsigned int numverts, float epsilon);
|
void bvhtree_update_from_mvert ( BVHTree * bvhtree, MFace *faces, int numfaces, MVert *x, MVert *xnew, int numverts, int moving );
|
||||||
|
/////////////////////////////////////////////////
|
||||||
// frees the same
|
|
||||||
void bvh_free ( BVH * bvh );
|
|
||||||
|
|
||||||
// checks two bounding volume hierarchies for potential collisions and returns some list with those
|
|
||||||
|
|
||||||
|
|
||||||
// update bounding volumes, needs updated positions in bvh->current_xold (static)
|
|
||||||
// and also bvh->current_x if moving==1
|
|
||||||
void bvh_update_from_mvert(BVH * bvh, MVert *x, unsigned int numverts, MVert *xnew, int moving);
|
|
||||||
void bvh_update(BVH * bvh, int moving);
|
|
||||||
|
|
||||||
LinkNode *BLI_linklist_append_fast ( LinkNode **listp, void *ptr );
|
LinkNode *BLI_linklist_append_fast ( LinkNode **listp, void *ptr );
|
||||||
|
|
||||||
|
|||||||
@@ -39,8 +39,8 @@ struct ListBase;
|
|||||||
struct BezTriple;
|
struct BezTriple;
|
||||||
struct BevList;
|
struct BevList;
|
||||||
|
|
||||||
#define KNOTSU(nu) ( (nu)->orderu+ (nu)->pntsu+ (nu->orderu-1)*((nu)->flagu & 1) )
|
#define KNOTSU(nu) ( (nu)->orderu+ (nu)->pntsu+ (nu->orderu-1)*((nu)->flagu & CU_CYCLIC) )
|
||||||
#define KNOTSV(nu) ( (nu)->orderv+ (nu)->pntsv+ (nu->orderv-1)*((nu)->flagv & 1) )
|
#define KNOTSV(nu) ( (nu)->orderv+ (nu)->pntsv+ (nu->orderv-1)*((nu)->flagv & CU_CYCLIC) )
|
||||||
|
|
||||||
|
|
||||||
void unlink_curve( struct Curve *cu);
|
void unlink_curve( struct Curve *cu);
|
||||||
@@ -84,5 +84,12 @@ void switchdirectionNurb( struct Nurb *nu);
|
|||||||
float (*curve_getVertexCos(struct Curve *cu, struct ListBase *lb, int *numVerts_r))[3];
|
float (*curve_getVertexCos(struct Curve *cu, struct ListBase *lb, int *numVerts_r))[3];
|
||||||
void curve_applyVertexCos(struct Curve *cu, struct ListBase *lb, float (*vertexCos)[3]);
|
void curve_applyVertexCos(struct Curve *cu, struct ListBase *lb, float (*vertexCos)[3]);
|
||||||
|
|
||||||
|
/* nurb checks if they can be drawn, also clamp order func */
|
||||||
|
int check_valid_nurb_u( struct Nurb *nu);
|
||||||
|
int check_valid_nurb_v( struct Nurb *nu);
|
||||||
|
|
||||||
|
int clamp_nurb_order_u( struct Nurb *nu);
|
||||||
|
int clamp_nurb_order_v( struct Nurb *nu);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
198
source/blender/blenkernel/intern/BME_Customdata.c
Normal file
198
source/blender/blenkernel/intern/BME_Customdata.c
Normal file
@@ -0,0 +1,198 @@
|
|||||||
|
/**
|
||||||
|
* BME_customdata.c jan 2007
|
||||||
|
*
|
||||||
|
* Custom Data functions for Bmesh
|
||||||
|
*
|
||||||
|
* $Id: BKE_bmesh.h,v 1.00 2007/01/17 17:42:01 Briggs Exp $
|
||||||
|
*
|
||||||
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version. The Blender
|
||||||
|
* Foundation also sells licenses for use in proprietary software under
|
||||||
|
* the Blender License. See http://www.blender.org/BL/ for information
|
||||||
|
* about this.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||||||
|
*
|
||||||
|
* The Original Code is Copyright (C) 2004 Blender Foundation.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* The Original Code is: all of this file.
|
||||||
|
*
|
||||||
|
* Contributor(s): Geoffrey Bantle, Brecht Van Lommel, Ben Batt
|
||||||
|
*
|
||||||
|
* ***** END GPL LICENSE BLOCK *****
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include "BKE_bmesh.h"
|
||||||
|
#include "BKE_bmeshCustomData.h"
|
||||||
|
#include "bmesh_private.h"
|
||||||
|
#include <string.h>
|
||||||
|
#include "MEM_guardedalloc.h"
|
||||||
|
|
||||||
|
/********************* Layer type information **********************/
|
||||||
|
typedef struct BME_LayerTypeInfo {
|
||||||
|
int size;
|
||||||
|
char *defaultname;
|
||||||
|
void (*copy)(const void *source, void *dest, int count);
|
||||||
|
void (*free)(void *data, int count, int size);
|
||||||
|
void (*interp)(void **sources, float *weights, float *sub_weights, int count, void *dest);
|
||||||
|
void (*set_default)(void *data, int count);
|
||||||
|
} BME_LayerTypeInfo;
|
||||||
|
const BME_LayerTypeInfo BMELAYERTYPEINFO[BME_CD_NUMTYPES] = {
|
||||||
|
{sizeof(BME_facetex), "TexFace", NULL, NULL, NULL, NULL},
|
||||||
|
{sizeof(BME_looptex), "UV", NULL, NULL, NULL, NULL},
|
||||||
|
{sizeof(BME_loopcol), "VCol", NULL, NULL, NULL, NULL},
|
||||||
|
{sizeof(BME_DeformVert), "Group", NULL, NULL, NULL, NULL}
|
||||||
|
};
|
||||||
|
static const BME_LayerTypeInfo *BME_layerType_getInfo(int type)
|
||||||
|
{
|
||||||
|
if(type < 0 || type >= CD_NUMTYPES) return NULL;
|
||||||
|
|
||||||
|
return &BMELAYERTYPEINFO[type];
|
||||||
|
}
|
||||||
|
void BME_CD_Create(BME_CustomData *data, BME_CustomDataInit *init, int initalloc)
|
||||||
|
{
|
||||||
|
int i, j, offset=0;
|
||||||
|
const BME_LayerTypeInfo *info;
|
||||||
|
|
||||||
|
/*initialize data members*/
|
||||||
|
data->layers = NULL;
|
||||||
|
data->pool = NULL;
|
||||||
|
data->totlayer = 0;
|
||||||
|
data->totsize = 0;
|
||||||
|
|
||||||
|
/*first count how many layers to alloc*/
|
||||||
|
for(i=0; i < BME_CD_NUMTYPES; i++){
|
||||||
|
info = BME_layerType_getInfo(i);
|
||||||
|
data->totlayer += init->layout[i];
|
||||||
|
data->totsize += (init->layout[i] * info->size);
|
||||||
|
}
|
||||||
|
/*alloc our layers*/
|
||||||
|
if(data->totlayer){
|
||||||
|
/*alloc memory*/
|
||||||
|
data->layers = MEM_callocN(sizeof(BME_CustomDataLayer)*data->totlayer, "BMesh Custom Data Layers");
|
||||||
|
data->pool = BME_mempool_create(data->totsize, initalloc, initalloc);
|
||||||
|
/*initialize layer data*/
|
||||||
|
for(i=0; i < BME_CD_NUMTYPES; i++){
|
||||||
|
if(init->layout[i]){
|
||||||
|
info = BME_layerType_getInfo(i);
|
||||||
|
for(j=0; j < init->layout[i]; j++){
|
||||||
|
if(j==0) data->layers[j+i].active = init->active[i];
|
||||||
|
data->layers[j+i].type = i;
|
||||||
|
data->layers[j+i].offset = offset;
|
||||||
|
strcpy(data->layers[j+i].name, &(init->nametemplate[j+i]));
|
||||||
|
offset += info->size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BME_CD_Free(BME_CustomData *data)
|
||||||
|
{
|
||||||
|
if(data->pool) BME_mempool_destroy(data->pool);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*Block level ops*/
|
||||||
|
void BME_CD_free_block(BME_CustomData *data, void **block)
|
||||||
|
{
|
||||||
|
const BME_LayerTypeInfo *typeInfo;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if(!*block) return;
|
||||||
|
for(i = 0; i < data->totlayer; ++i) {
|
||||||
|
typeInfo = BME_layerType_getInfo(data->layers[i].type);
|
||||||
|
if(typeInfo->free) {
|
||||||
|
int offset = data->layers[i].offset;
|
||||||
|
typeInfo->free((char*)*block + offset, 1, typeInfo->size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BME_mempool_free(data->pool, *block);
|
||||||
|
*block = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void BME_CD_alloc_block(BME_CustomData *data, void **block)
|
||||||
|
{
|
||||||
|
|
||||||
|
if (*block) BME_CD_free_block(data, block); //if we copy layers that have their own free functions like deformverts
|
||||||
|
|
||||||
|
if (data->totsize > 0)
|
||||||
|
*block = BME_mempool_alloc(data->pool);
|
||||||
|
else
|
||||||
|
*block = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void BME_CD_copy_data(const BME_CustomData *source, BME_CustomData *dest,
|
||||||
|
void *src_block, void **dest_block)
|
||||||
|
{
|
||||||
|
const BME_LayerTypeInfo *typeInfo;
|
||||||
|
int dest_i, src_i;
|
||||||
|
|
||||||
|
if (!*dest_block) /*for addXXXlist functions!*/
|
||||||
|
BME_CD_alloc_block(dest, dest_block);
|
||||||
|
|
||||||
|
/* copies a layer at a time */
|
||||||
|
dest_i = 0;
|
||||||
|
for(src_i = 0; src_i < source->totlayer; ++src_i) {
|
||||||
|
|
||||||
|
/* find the first dest layer with type >= the source type
|
||||||
|
* (this should work because layers are ordered by type)
|
||||||
|
*/
|
||||||
|
while(dest_i < dest->totlayer
|
||||||
|
&& dest->layers[dest_i].type < source->layers[src_i].type)
|
||||||
|
++dest_i;
|
||||||
|
|
||||||
|
/* if there are no more dest layers, we're done */
|
||||||
|
if(dest_i >= dest->totlayer) return;
|
||||||
|
|
||||||
|
/* if we found a matching layer, copy the data */
|
||||||
|
if(dest->layers[dest_i].type == source->layers[src_i].type &&
|
||||||
|
strcmp(dest->layers[dest_i].name, source->layers[src_i].name) == 0) {
|
||||||
|
char *src_data = (char*)src_block + source->layers[src_i].offset;
|
||||||
|
char *dest_data = (char*)*dest_block + dest->layers[dest_i].offset;
|
||||||
|
|
||||||
|
typeInfo = BME_layerType_getInfo(source->layers[src_i].type);
|
||||||
|
|
||||||
|
if(typeInfo->copy)
|
||||||
|
typeInfo->copy(src_data, dest_data, 1);
|
||||||
|
else
|
||||||
|
memcpy(dest_data, src_data, typeInfo->size);
|
||||||
|
|
||||||
|
/* if there are multiple source & dest layers of the same type,
|
||||||
|
* we don't want to copy all source layers to the same dest, so
|
||||||
|
* increment dest_i
|
||||||
|
*/
|
||||||
|
++dest_i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void BME_CD_set_default(BME_CustomData *data, void **block)
|
||||||
|
{
|
||||||
|
const BME_LayerTypeInfo *typeInfo;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!*block)
|
||||||
|
BME_CD_alloc_block(data, block); //for addXXXlist functions...
|
||||||
|
|
||||||
|
for(i = 0; i < data->totlayer; ++i) {
|
||||||
|
int offset = data->layers[i].offset;
|
||||||
|
|
||||||
|
typeInfo = BME_layerType_getInfo(data->layers[i].type);
|
||||||
|
|
||||||
|
if(typeInfo->set_default)
|
||||||
|
typeInfo->set_default((char*)*block + offset, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -55,7 +55,10 @@
|
|||||||
|
|
||||||
#include "BSE_edit.h"
|
#include "BSE_edit.h"
|
||||||
|
|
||||||
BME_Mesh *BME_editmesh_to_bmesh(EditMesh *em, BME_Mesh *bm) {
|
BME_Mesh *BME_editmesh_to_bmesh(EditMesh *em) {
|
||||||
|
BME_Mesh *bm;
|
||||||
|
int allocsize[4] = {512,512,2048,512};
|
||||||
|
BME_CustomDataInit *init = MEM_callocN(sizeof(BME_CustomDataInit) * 4, "Bmesh custom data init");
|
||||||
BME_Vert *v1, *v2;
|
BME_Vert *v1, *v2;
|
||||||
BME_Edge *e, *edar[4];
|
BME_Edge *e, *edar[4];
|
||||||
BME_Poly *f;
|
BME_Poly *f;
|
||||||
@@ -65,7 +68,7 @@ BME_Mesh *BME_editmesh_to_bmesh(EditMesh *em, BME_Mesh *bm) {
|
|||||||
EditFace *efa;
|
EditFace *efa;
|
||||||
|
|
||||||
int len;
|
int len;
|
||||||
|
bm = BME_make_mesh(allocsize,init);
|
||||||
BME_model_begin(bm);
|
BME_model_begin(bm);
|
||||||
|
|
||||||
/*add verts*/
|
/*add verts*/
|
||||||
@@ -134,6 +137,7 @@ BME_Mesh *BME_editmesh_to_bmesh(EditMesh *em, BME_Mesh *bm) {
|
|||||||
efa = efa->next;
|
efa = efa->next;
|
||||||
}
|
}
|
||||||
BME_model_end(bm);
|
BME_model_end(bm);
|
||||||
|
MEM_freeN(init);
|
||||||
return bm;
|
return bm;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -224,21 +228,24 @@ EditMesh *BME_bmesh_to_editmesh(BME_Mesh *bm, BME_TransData_Head *td) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Adds the geometry found in dm to bm
|
/* Adds the geometry found in dm to bm
|
||||||
* NOTE: it does not allocate a new BME_Mesh!
|
|
||||||
*/
|
*/
|
||||||
BME_Mesh *BME_derivedmesh_to_bmesh(DerivedMesh *dm, BME_Mesh *bm)
|
BME_Mesh *BME_derivedmesh_to_bmesh(DerivedMesh *dm)
|
||||||
{
|
{
|
||||||
|
|
||||||
|
BME_Mesh *bm;
|
||||||
|
int allocsize[4] = {512,512,2048,512};
|
||||||
|
BME_CustomDataInit *init = MEM_callocN(sizeof(BME_CustomDataInit) * 4, "Bmesh custom data init");
|
||||||
MVert *mvert, *mv;
|
MVert *mvert, *mv;
|
||||||
MEdge *medge, *me;
|
MEdge *medge, *me;
|
||||||
MFace *mface, *mf;
|
MFace *mface, *mf;
|
||||||
int totface,totedge,totvert,i,len;
|
int totface,totedge,totvert,i,len;
|
||||||
|
|
||||||
BME_Vert *v1=NULL,*v2=NULL, **vert_array;
|
BME_Vert *v1=NULL,*v2=NULL, **vert_array;
|
||||||
BME_Edge *e=NULL;
|
BME_Edge *e=NULL;
|
||||||
BME_Poly *f=NULL;
|
BME_Poly *f=NULL;
|
||||||
|
|
||||||
EdgeHash *edge_hash = BLI_edgehash_new();
|
EdgeHash *edge_hash = BLI_edgehash_new();
|
||||||
|
|
||||||
|
bm = BME_make_mesh(allocsize,init);
|
||||||
totvert = dm->getNumVerts(dm);
|
totvert = dm->getNumVerts(dm);
|
||||||
totedge = dm->getNumEdges(dm);
|
totedge = dm->getNumEdges(dm);
|
||||||
totface = dm->getNumFaces(dm);
|
totface = dm->getNumFaces(dm);
|
||||||
@@ -293,6 +300,7 @@ BME_Mesh *BME_derivedmesh_to_bmesh(DerivedMesh *dm, BME_Mesh *bm)
|
|||||||
BME_model_end(bm);
|
BME_model_end(bm);
|
||||||
BLI_edgehash_free(edge_hash, NULL);
|
BLI_edgehash_free(edge_hash, NULL);
|
||||||
MEM_freeN(vert_array);
|
MEM_freeN(vert_array);
|
||||||
|
MEM_freeN(init);
|
||||||
return bm;
|
return bm;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -59,14 +59,39 @@
|
|||||||
/*
|
/*
|
||||||
* BME MAKE MESH
|
* BME MAKE MESH
|
||||||
*
|
*
|
||||||
* Allocates a new BME_Mesh structure
|
* Allocates a new BME_Mesh structure.
|
||||||
|
* The arguments are two arrays, one of type int
|
||||||
|
* and another of type BME_CustomDataInit. The first array
|
||||||
|
* contains the allocation size for each element pool in
|
||||||
|
* the mesh. For instance allocsize[0] contains the number
|
||||||
|
* of vertices to allocate at a time for the vertex pool.
|
||||||
|
*
|
||||||
|
* The second array contains structures describing the layout
|
||||||
|
* of custom data for each element type in the mesh. So init[0]
|
||||||
|
* contains the custom data layout information for vertices, init[1]
|
||||||
|
* the layout information for edges and so on.
|
||||||
|
*
|
||||||
|
* Returns -
|
||||||
|
* Pointer to a Bmesh
|
||||||
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
BME_Mesh *BME_make_mesh(void){
|
BME_Mesh *BME_make_mesh(int allocsize[4], BME_CustomDataInit init[4])
|
||||||
|
{
|
||||||
|
/*allocate the structure*/
|
||||||
BME_Mesh *bm = MEM_callocN(sizeof(BME_Mesh),"BMesh");
|
BME_Mesh *bm = MEM_callocN(sizeof(BME_Mesh),"BMesh");
|
||||||
|
/*allocate the memory pools for the mesh elements*/
|
||||||
|
bm->vpool = BME_mempool_create(sizeof(BME_Vert), allocsize[0], allocsize[0]);
|
||||||
|
bm->epool = BME_mempool_create(sizeof(BME_Edge), allocsize[1], allocsize[1]);
|
||||||
|
bm->lpool = BME_mempool_create(sizeof(BME_Loop), allocsize[2], allocsize[2]);
|
||||||
|
bm->ppool = BME_mempool_create(sizeof(BME_Poly), allocsize[3], allocsize[3]);
|
||||||
|
/*Setup custom data layers*/
|
||||||
|
BME_CD_Create(&bm->vdata, &init[0], allocsize[0]);
|
||||||
|
BME_CD_Create(&bm->edata, &init[1], allocsize[1]);
|
||||||
|
BME_CD_Create(&bm->ldata, &init[2], allocsize[2]);
|
||||||
|
BME_CD_Create(&bm->pdata, &init[3], allocsize[3]);
|
||||||
return bm;
|
return bm;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* BME FREE MESH
|
* BME FREE MESH
|
||||||
*
|
*
|
||||||
@@ -75,45 +100,31 @@ BME_Mesh *BME_make_mesh(void){
|
|||||||
|
|
||||||
void BME_free_mesh(BME_Mesh *bm)
|
void BME_free_mesh(BME_Mesh *bm)
|
||||||
{
|
{
|
||||||
BME_Poly *bf, *nextf;
|
BME_Vert *v;
|
||||||
BME_Edge *be, *nexte;
|
BME_Edge *e;
|
||||||
BME_Vert *bv, *nextv;
|
BME_Loop *l;
|
||||||
BME_CycleNode *loopref;
|
BME_Poly *f;
|
||||||
|
|
||||||
/*destroy polygon data*/
|
for(v=bm->verts.first; v; v=v->next) BME_CD_free_block(&bm->vdata, &v->data);
|
||||||
bf = bm->polys.first;
|
for(e=bm->edges.first; e; e=e->next) BME_CD_free_block(&bm->edata, &e->data);
|
||||||
while(bf){
|
for(f=bm->polys.first; f; f=f->next){
|
||||||
nextf = bf->next;
|
BME_CD_free_block(&bm->pdata, &f->data);
|
||||||
BLI_remlink(&(bm->polys), bf);
|
l = f->loopbase;
|
||||||
BME_free_poly(bm, bf);
|
do{
|
||||||
|
BME_CD_free_block(&bm->ldata, &l->data);
|
||||||
bf = nextf;
|
l = l->next;
|
||||||
|
}while(l!=f->loopbase);
|
||||||
}
|
}
|
||||||
/*destroy edge data*/
|
/*destroy element pools*/
|
||||||
be = bm->edges.first;
|
BME_mempool_destroy(bm->vpool);
|
||||||
while(be){
|
BME_mempool_destroy(bm->epool);
|
||||||
nexte = be->next;
|
BME_mempool_destroy(bm->ppool);
|
||||||
BLI_remlink(&(bm->edges), be);
|
BME_mempool_destroy(bm->lpool);
|
||||||
BME_free_edge(bm, be);
|
/*free custom data pools*/
|
||||||
be = nexte;
|
BME_CD_Free(&bm->vdata);
|
||||||
}
|
BME_CD_Free(&bm->edata);
|
||||||
/*destroy vert data*/
|
BME_CD_Free(&bm->ldata);
|
||||||
bv = bm->verts.first;
|
BME_CD_Free(&bm->pdata);
|
||||||
while(bv){
|
|
||||||
nextv = bv->next;
|
|
||||||
BLI_remlink(&(bm->verts), bv);
|
|
||||||
BME_free_vert(bm, bv);
|
|
||||||
bv = nextv;
|
|
||||||
}
|
|
||||||
|
|
||||||
for(loopref=bm->loops.first;loopref;loopref=loopref->next) BME_delete_loop(bm,loopref->data);
|
|
||||||
BLI_freelistN(&(bm->loops));
|
|
||||||
|
|
||||||
//CustomData_free(&bm->vdata, 0);
|
|
||||||
//CustomData_free(&bm->edata, 0);
|
|
||||||
//CustomData_free(&bm->ldata, 0);
|
|
||||||
//CustomData_free(&bm->pdata, 0);
|
|
||||||
|
|
||||||
MEM_freeN(bm);
|
MEM_freeN(bm);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -124,17 +135,12 @@ void BME_free_mesh(BME_Mesh *bm)
|
|||||||
* must begin with a call to BME_model_end() and finish with a call to BME_model_end().
|
* must begin with a call to BME_model_end() and finish with a call to BME_model_end().
|
||||||
* No modification of mesh data is allowed except in between these two calls.
|
* No modification of mesh data is allowed except in between these two calls.
|
||||||
*
|
*
|
||||||
* TODO
|
* The purpose of these calls is allow for housekeeping tasks to be performed,
|
||||||
* FOR BME_MODEL_BEGIN:
|
* such as allocating/freeing scratch arrays or performing debug validation of
|
||||||
* -integrate euler undo system.
|
* the mesh structure.
|
||||||
* -make full copy of structure to safely recover from errors.
|
|
||||||
* -accept a toolname string.
|
|
||||||
* -accept param to turn off full copy if just selection tool. (perhaps check for this in eulers...)
|
|
||||||
*
|
*
|
||||||
* BME_MODEL_END:
|
* Returns -
|
||||||
* -full mesh validation if debugging turned on
|
* Nothing
|
||||||
* -free structure copy or use it to restore.
|
|
||||||
* -do euler undo push.
|
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@@ -151,12 +157,11 @@ int BME_model_begin(BME_Mesh *bm){
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BME_model_end(BME_Mesh *bm){
|
void BME_model_end(BME_Mesh *bm){
|
||||||
int meshok, totvert, totedge, totpoly, totloop;
|
int meshok, totvert, totedge, totpoly;
|
||||||
|
|
||||||
totvert = BLI_countlist(&(bm->verts));
|
totvert = BLI_countlist(&(bm->verts));
|
||||||
totedge = BLI_countlist(&(bm->edges));
|
totedge = BLI_countlist(&(bm->edges));
|
||||||
totpoly = BLI_countlist(&(bm->polys));
|
totpoly = BLI_countlist(&(bm->polys));
|
||||||
totloop = BLI_countlist(&(bm->loops));
|
|
||||||
|
|
||||||
if(bm->vtar) MEM_freeN(bm->vtar);
|
if(bm->vtar) MEM_freeN(bm->vtar);
|
||||||
if(bm->edar) MEM_freeN(bm->edar);
|
if(bm->edar) MEM_freeN(bm->edar);
|
||||||
@@ -167,10 +172,10 @@ void BME_model_end(BME_Mesh *bm){
|
|||||||
bm->edar = NULL;
|
bm->edar = NULL;
|
||||||
bm->lpar = NULL;
|
bm->lpar = NULL;
|
||||||
bm->plar = NULL;
|
bm->plar = NULL;
|
||||||
bm->vtarlen = bm->edarlen = bm->lparlen = bm->plarlen = 1024;
|
bm->vtarlen = bm->edarlen = bm->lparlen = bm->plarlen = 0;
|
||||||
|
|
||||||
|
|
||||||
if(bm->totvert!=totvert || bm->totedge!=totedge || bm->totpoly!=totpoly || bm->totloop!=totloop)
|
if(bm->totvert!=totvert || bm->totedge!=totedge || bm->totpoly!=totpoly)
|
||||||
BME_error();
|
BME_error();
|
||||||
|
|
||||||
meshok = BME_validate_mesh(bm, 1);
|
meshok = BME_validate_mesh(bm, 1);
|
||||||
@@ -194,7 +199,7 @@ void BME_model_end(BME_Mesh *bm){
|
|||||||
*
|
*
|
||||||
* TODO
|
* TODO
|
||||||
*
|
*
|
||||||
* -Write a full mesh validation function for debugging purposes.
|
* -Make this only part of debug builds
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define VHALT(halt) {BME_error(); if(halt) return 0;}
|
#define VHALT(halt) {BME_error(); if(halt) return 0;}
|
||||||
|
|||||||
@@ -43,6 +43,99 @@
|
|||||||
#include "BLI_ghash.h"
|
#include "BLI_ghash.h"
|
||||||
|
|
||||||
#include "BKE_customdata.h"
|
#include "BKE_customdata.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
Simple, fast memory allocator for allocating many elements of the same size.
|
||||||
|
*/
|
||||||
|
typedef struct BME_mempool_chunk{
|
||||||
|
struct BME_mempool_chunk *next, *prev;
|
||||||
|
void *data;
|
||||||
|
}BME_mempool_chunk;
|
||||||
|
|
||||||
|
/*this is just to make things prettier*/
|
||||||
|
typedef struct BME_freenode{
|
||||||
|
struct BME_freenode *next;
|
||||||
|
}BME_freenode;
|
||||||
|
|
||||||
|
BME_mempool *BME_mempool_create(int esize, int tote, int pchunk)
|
||||||
|
{ BME_mempool *pool = NULL;
|
||||||
|
BME_freenode *lasttail = NULL, *curnode = NULL;
|
||||||
|
int i,j, maxchunks;
|
||||||
|
char *addr;
|
||||||
|
|
||||||
|
/*allocate the pool structure*/
|
||||||
|
pool = MEM_mallocN(sizeof(BME_mempool),"memory pool");
|
||||||
|
pool->esize = esize;
|
||||||
|
pool->pchunk = pchunk;
|
||||||
|
pool->csize = esize * pchunk;
|
||||||
|
pool->chunks.first = pool->chunks.last = NULL;
|
||||||
|
|
||||||
|
maxchunks = tote / pchunk;
|
||||||
|
|
||||||
|
/*allocate the actual chunks*/
|
||||||
|
for(i=0; i < maxchunks; i++){
|
||||||
|
BME_mempool_chunk *mpchunk = MEM_mallocN(sizeof(BME_mempool_chunk), "BME_Mempool Chunk");
|
||||||
|
mpchunk->next = mpchunk->prev = NULL;
|
||||||
|
mpchunk->data = MEM_mallocN(pool->csize, "BME Mempool Chunk Data");
|
||||||
|
BLI_addtail(&(pool->chunks), mpchunk);
|
||||||
|
|
||||||
|
if(i==0) pool->free = mpchunk->data; /*start of the list*/
|
||||||
|
/*loop through the allocated data, building the pointer structures*/
|
||||||
|
for(addr = mpchunk->data, j=0; j < pool->pchunk; j++){
|
||||||
|
curnode = ((BME_freenode*)addr);
|
||||||
|
addr += pool->esize;
|
||||||
|
curnode->next = (BME_freenode*)addr;
|
||||||
|
}
|
||||||
|
/*final pointer in the previously allocated chunk is wrong.*/
|
||||||
|
if(lasttail) lasttail->next = mpchunk->data;
|
||||||
|
/*set the end of this chunks memory to the new tail for next iteration*/
|
||||||
|
lasttail = curnode;
|
||||||
|
}
|
||||||
|
/*terminate the list*/
|
||||||
|
curnode->next = NULL;
|
||||||
|
return pool;
|
||||||
|
}
|
||||||
|
|
||||||
|
void *BME_mempool_alloc(BME_mempool *pool){
|
||||||
|
void *retval=NULL;
|
||||||
|
BME_freenode *curnode=NULL;
|
||||||
|
char *addr=NULL;
|
||||||
|
int j;
|
||||||
|
|
||||||
|
if(!(pool->free)){
|
||||||
|
/*need to allocate a new chunk*/
|
||||||
|
BME_mempool_chunk *mpchunk = MEM_mallocN(sizeof(BME_mempool_chunk), "BME_Mempool Chunk");
|
||||||
|
mpchunk->next = mpchunk->prev = NULL;
|
||||||
|
mpchunk->data = MEM_mallocN(pool->csize, "BME_Mempool Chunk Data");
|
||||||
|
BLI_addtail(&(pool->chunks), mpchunk);
|
||||||
|
|
||||||
|
pool->free = mpchunk->data; /*start of the list*/
|
||||||
|
for(addr = mpchunk->data, j=0; j < pool->pchunk; j++){
|
||||||
|
curnode = ((BME_freenode*)addr);
|
||||||
|
addr += pool->esize;
|
||||||
|
curnode->next = (BME_freenode*)addr;
|
||||||
|
}
|
||||||
|
curnode->next = NULL; /*terminate the list*/
|
||||||
|
}
|
||||||
|
|
||||||
|
retval = pool->free;
|
||||||
|
pool->free = pool->free->next;
|
||||||
|
//memset(retval, 0, pool->esize);
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
|
void BME_mempool_free(BME_mempool *pool, void *addr){ //doesnt protect against double frees, dont be stupid!
|
||||||
|
BME_freenode *newhead = addr;
|
||||||
|
newhead->next = pool->free;
|
||||||
|
pool->free = newhead;
|
||||||
|
}
|
||||||
|
void BME_mempool_destroy(BME_mempool *pool)
|
||||||
|
{
|
||||||
|
BME_mempool_chunk *mpchunk=NULL;
|
||||||
|
for(mpchunk = pool->chunks.first; mpchunk; mpchunk = mpchunk->next) MEM_freeN(mpchunk->data);
|
||||||
|
BLI_freelistN(&(pool->chunks));
|
||||||
|
MEM_freeN(pool);
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* MISC utility functions.
|
* MISC utility functions.
|
||||||
*
|
*
|
||||||
@@ -86,78 +179,98 @@ int BME_edge_swapverts(BME_Edge *e, BME_Vert *orig, BME_Vert *new){
|
|||||||
|
|
||||||
BME_Vert *BME_addvertlist(BME_Mesh *bm, BME_Vert *example){
|
BME_Vert *BME_addvertlist(BME_Mesh *bm, BME_Vert *example){
|
||||||
BME_Vert *v=NULL;
|
BME_Vert *v=NULL;
|
||||||
v = MEM_callocN(sizeof(BME_Vert), "BME Vertex");
|
v = BME_mempool_alloc(bm->vpool);
|
||||||
BLI_addtail(&(bm->verts), v);
|
v->next = v->prev = NULL;
|
||||||
v->EID = bm->nextv;
|
v->EID = bm->nextv;
|
||||||
|
v->co[0] = v->co[1] = v->co[2] = 0.0f;
|
||||||
|
v->no[0] = v->no[1] = v->no[2] = 0.0f;
|
||||||
|
v->edge = NULL;
|
||||||
|
v->data = NULL;
|
||||||
|
v->eflag1 = v->eflag2 = v->tflag1 = v->tflag2 = 0;
|
||||||
|
v->flag = v->h = 0;
|
||||||
|
v->bweight = 0.0f;
|
||||||
|
BLI_addtail(&(bm->verts), v);
|
||||||
bm->nextv++;
|
bm->nextv++;
|
||||||
bm->totvert++;
|
bm->totvert++;
|
||||||
|
|
||||||
if(example)
|
if(example){
|
||||||
VECCOPY(v->co,example->co);
|
VECCOPY(v->co,example->co);
|
||||||
//if(example)
|
BME_CD_copy_data(&bm->vdata, &bm->vdata, example->data, &v->data);
|
||||||
// CustomData_em_copy_data(&bm->vdata, &bm->vdata, example->data, &v->data);
|
}
|
||||||
//else
|
else
|
||||||
// CustomData_em_set_default(&bm->vdata, &v->data);
|
BME_CD_set_default(&bm->vdata, &v->data);
|
||||||
|
|
||||||
return v;
|
return v;
|
||||||
}
|
}
|
||||||
BME_Edge *BME_addedgelist(BME_Mesh *bm, BME_Vert *v1, BME_Vert *v2, BME_Edge *example){
|
BME_Edge *BME_addedgelist(BME_Mesh *bm, BME_Vert *v1, BME_Vert *v2, BME_Edge *example){
|
||||||
BME_Edge *e=NULL;
|
BME_Edge *e=NULL;
|
||||||
e = MEM_callocN(sizeof(BME_Edge), "BME_Edge");
|
e = BME_mempool_alloc(bm->epool);
|
||||||
|
e->next = e->prev = NULL;
|
||||||
|
e->EID = bm->nexte;
|
||||||
e->v1 = v1;
|
e->v1 = v1;
|
||||||
e->v2 = v2;
|
e->v2 = v2;
|
||||||
|
e->d1.next = e->d1.prev = e->d2.next = e->d2.prev = NULL;
|
||||||
e->d1.data = e;
|
e->d1.data = e;
|
||||||
e->d2.data = e;
|
e->d2.data = e;
|
||||||
e->EID = bm->nexte;
|
e->loop = NULL;
|
||||||
|
e->data = NULL;
|
||||||
|
e->eflag1 = e->eflag2 = e->tflag1 = e->tflag2 = 0;
|
||||||
|
e->flag = e->h = 0;
|
||||||
|
e->crease = e->bweight = 0.0f;
|
||||||
bm->nexte++;
|
bm->nexte++;
|
||||||
bm->totedge++;
|
bm->totedge++;
|
||||||
BLI_addtail(&(bm->edges), e);
|
BLI_addtail(&(bm->edges), e);
|
||||||
|
|
||||||
//if(example)
|
if(example)
|
||||||
// CustomData_em_copy_data(&bm->edata, &bm->edata, example->data, &e->data);
|
BME_CD_copy_data(&bm->edata, &bm->edata, example->data, &e->data);
|
||||||
//else
|
else
|
||||||
// CustomData_em_set_default(&bm->edata, &e->data);
|
BME_CD_set_default(&bm->edata, &e->data);
|
||||||
|
|
||||||
|
|
||||||
return e;
|
return e;
|
||||||
}
|
}
|
||||||
BME_Loop *BME_create_loop(BME_Mesh *bm, BME_Vert *v, BME_Edge *e, BME_Poly *f, BME_Loop *example){
|
BME_Loop *BME_create_loop(BME_Mesh *bm, BME_Vert *v, BME_Edge *e, BME_Poly *f, BME_Loop *example){
|
||||||
/*allocate a BME_Loop and add it to the loophash*/
|
|
||||||
BME_Loop *l=NULL;
|
BME_Loop *l=NULL;
|
||||||
BME_CycleNode *loopnode = MEM_callocN(sizeof(BME_CycleNode),"BME Loop Reference");
|
l = BME_mempool_alloc(bm->lpool);
|
||||||
l = MEM_callocN(sizeof(BME_Loop), "BME_Loop");
|
l->next = l->prev = NULL;
|
||||||
|
l->EID = bm->nextl;
|
||||||
|
l->radial.next = l->radial.prev = NULL;
|
||||||
l->radial.data = l;
|
l->radial.data = l;
|
||||||
l->v = v;
|
l->v = v;
|
||||||
l->e = e;
|
l->e = e;
|
||||||
l->f = f;
|
l->f = f;
|
||||||
l->EID = bm->nextl;
|
l->data = NULL;
|
||||||
l->gref = loopnode;
|
l->eflag1 = l->eflag2 = l->tflag1 = l->tflag2 = 0;
|
||||||
loopnode->data = l;
|
l->flag = l->h = 0; //stupid waste!
|
||||||
BLI_addtail(&(bm->loops),loopnode);
|
|
||||||
bm->nextl++;
|
bm->nextl++;
|
||||||
bm->totloop++;
|
bm->totloop++;
|
||||||
|
|
||||||
|
if(example)
|
||||||
/* if(example)
|
BME_CD_copy_data(&bm->ldata, &bm->ldata, example->data, &l->data);
|
||||||
BME_CustomData_copy_data(&bm->ldata, &bm->ldata, example->data, &l->data);
|
|
||||||
else
|
else
|
||||||
BME_CustomData_set_default(&bm->ldata, &l->data);
|
BME_CD_set_default(&bm->ldata, &l->data);
|
||||||
*/
|
|
||||||
return l;
|
return l;
|
||||||
}
|
}
|
||||||
|
|
||||||
BME_Poly *BME_addpolylist(BME_Mesh *bm, BME_Poly *example){
|
BME_Poly *BME_addpolylist(BME_Mesh *bm, BME_Poly *example){
|
||||||
BME_Poly *f = NULL;
|
BME_Poly *f = NULL;
|
||||||
f= MEM_callocN(sizeof(BME_Poly),"BME_Poly");
|
f = BME_mempool_alloc(bm->ppool);
|
||||||
BLI_addtail(&(bm->polys),f);
|
f->next = f->prev = NULL;
|
||||||
f->EID = bm->nextp;
|
f->EID = bm->nextp;
|
||||||
|
f->loopbase = NULL;
|
||||||
|
f->len = 0;
|
||||||
|
f->data = NULL;
|
||||||
|
f->eflag1 = f->eflag2 = f->tflag1 = f->tflag2 = 0;
|
||||||
|
f->flag = f->h = f->mat_nr;
|
||||||
|
BLI_addtail(&(bm->polys),f);
|
||||||
bm->nextp++;
|
bm->nextp++;
|
||||||
bm->totpoly++;
|
bm->totpoly++;
|
||||||
|
|
||||||
//if(example)
|
if(example)
|
||||||
// CustomData_em_copy_data(&bm->pdata, &bm->pdata, example->data, &f->data);
|
BME_CD_copy_data(&bm->pdata, &bm->pdata, example->data, &f->data);
|
||||||
//else
|
else
|
||||||
// CustomData_em_set_default(&bm->pdata, &f->data);
|
BME_CD_set_default(&bm->pdata, &f->data);
|
||||||
|
|
||||||
|
|
||||||
return f;
|
return f;
|
||||||
@@ -168,31 +281,24 @@ BME_Poly *BME_addpolylist(BME_Mesh *bm, BME_Poly *example){
|
|||||||
*/
|
*/
|
||||||
void BME_free_vert(BME_Mesh *bm, BME_Vert *v){
|
void BME_free_vert(BME_Mesh *bm, BME_Vert *v){
|
||||||
bm->totvert--;
|
bm->totvert--;
|
||||||
//CustomData_em_free_block(&bm->vdata, &v->data);
|
BME_CD_free_block(&bm->vdata, &v->data);
|
||||||
MEM_freeN(v);
|
BME_mempool_free(bm->vpool, v);
|
||||||
}
|
}
|
||||||
void BME_free_edge(BME_Mesh *bm, BME_Edge *e){
|
void BME_free_edge(BME_Mesh *bm, BME_Edge *e){
|
||||||
bm->totedge--;
|
bm->totedge--;
|
||||||
//CustomData_em_free_block(&bm->edata, &e->data);
|
BME_CD_free_block(&bm->edata, &e->data);
|
||||||
MEM_freeN(e);
|
BME_mempool_free(bm->epool, e);
|
||||||
}
|
}
|
||||||
void BME_free_poly(BME_Mesh *bm, BME_Poly *f){
|
void BME_free_poly(BME_Mesh *bm, BME_Poly *f){
|
||||||
bm->totpoly--;
|
bm->totpoly--;
|
||||||
//CustomData_em_free_block(&bm->pdata, &f->data);
|
BME_CD_free_block(&bm->pdata, &f->data);
|
||||||
MEM_freeN(f);
|
BME_mempool_free(bm->ppool, f);
|
||||||
}
|
|
||||||
void BME_delete_loop(BME_Mesh *bm, BME_Loop *l){
|
|
||||||
bm->totloop--;
|
|
||||||
//CustomData_em_free_block(&bm->ldata, &l->data);
|
|
||||||
MEM_freeN(l);
|
|
||||||
}
|
}
|
||||||
void BME_free_loop(BME_Mesh *bm, BME_Loop *l){
|
void BME_free_loop(BME_Mesh *bm, BME_Loop *l){
|
||||||
BME_CycleNode *loopref = l->gref;
|
bm->totloop--;
|
||||||
BLI_freelinkN(&(bm->loops),loopref);
|
BME_CD_free_block(&bm->ldata, &l->data);
|
||||||
BME_delete_loop(bm,l);
|
BME_mempool_free(bm->lpool, l);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* BMESH CYCLES
|
* BMESH CYCLES
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -811,7 +811,18 @@ static float BME_bevel_get_angle(BME_Mesh *bm, BME_Edge *e, BME_Vert *v) {
|
|||||||
|
|
||||||
return Inpf(vec3,vec4);
|
return Inpf(vec3,vec4);
|
||||||
}
|
}
|
||||||
|
static int BME_face_sharededges(BME_Poly *f1, BME_Poly *f2){
|
||||||
|
BME_Loop *l;
|
||||||
|
int count = 0;
|
||||||
|
|
||||||
|
l = f1->loopbase;
|
||||||
|
do{
|
||||||
|
if(BME_radial_find_face(l->e,f2)) count++;
|
||||||
|
l = l->next;
|
||||||
|
}while(l != f1->loopbase);
|
||||||
|
|
||||||
|
return count;
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* BME_bevel_initialize
|
* BME_bevel_initialize
|
||||||
*
|
*
|
||||||
@@ -990,6 +1001,17 @@ static BME_Mesh *BME_bevel_initialize(BME_Mesh *bm, int options, int defgrp_inde
|
|||||||
/* face pass */
|
/* face pass */
|
||||||
for (f=bm->polys.first; f; f=f->next) f->tflag1 = BME_BEVEL_ORIG;
|
for (f=bm->polys.first; f; f=f->next) f->tflag1 = BME_BEVEL_ORIG;
|
||||||
|
|
||||||
|
/*clean up edges with 2 faces that share more than one edge*/
|
||||||
|
for (e=bm->edges.first; e; e=e->next){
|
||||||
|
if(e->tflag1 & BME_BEVEL_BEVEL){
|
||||||
|
int count = 0;
|
||||||
|
count = BME_face_sharededges(e->loop->f, ((BME_Loop*)e->loop->radial.next->data)->f);
|
||||||
|
if(count > 1){
|
||||||
|
e->tflag1 &= ~BME_BEVEL_BEVEL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return bm;
|
return bm;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1075,7 +1097,7 @@ static BME_Mesh *BME_bevel_mesh(BME_Mesh *bm, float value, int res, int options,
|
|||||||
v = BME_bevel_wire(bm, v, value, res, options, td);
|
v = BME_bevel_wire(bm, v, value, res, options, td);
|
||||||
}
|
}
|
||||||
else if (res && ((v->tflag1 & BME_BEVEL_BEVEL) && (v->tflag1 & BME_BEVEL_ORIG))) {
|
else if (res && ((v->tflag1 & BME_BEVEL_BEVEL) && (v->tflag1 & BME_BEVEL_ORIG))) {
|
||||||
|
int count = 0;
|
||||||
/* first, make sure we're not sitting on an edge to be removed */
|
/* first, make sure we're not sitting on an edge to be removed */
|
||||||
oe = v->edge;
|
oe = v->edge;
|
||||||
e = BME_disk_nextedge(oe,v);
|
e = BME_disk_nextedge(oe,v);
|
||||||
@@ -1089,6 +1111,7 @@ static BME_Mesh *BME_bevel_mesh(BME_Mesh *bm, float value, int res, int options,
|
|||||||
/* look for original edges, and remove them */
|
/* look for original edges, and remove them */
|
||||||
oe = e;
|
oe = e;
|
||||||
while ( (e = BME_disk_next_edgeflag(oe, v, 0, BME_BEVEL_ORIG | BME_BEVEL_BEVEL)) ) {
|
while ( (e = BME_disk_next_edgeflag(oe, v, 0, BME_BEVEL_ORIG | BME_BEVEL_BEVEL)) ) {
|
||||||
|
count++;
|
||||||
/* join the faces (we'll split them later) */
|
/* join the faces (we'll split them later) */
|
||||||
f = BME_JFKE_safe(bm,e->loop->f,((BME_Loop*)e->loop->radial.next->data)->f,e);
|
f = BME_JFKE_safe(bm,e->loop->f,((BME_Loop*)e->loop->radial.next->data)->f,e);
|
||||||
if (!f){
|
if (!f){
|
||||||
@@ -1096,6 +1119,9 @@ static BME_Mesh *BME_bevel_mesh(BME_Mesh *bm, float value, int res, int options,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*need to do double check *before* you bevel to make sure that manifold edges are for two faces that share only *one* edge to make sure it doesnt hang here!*/
|
||||||
|
|
||||||
|
|
||||||
/* all original edges marked to be beveled have been removed;
|
/* all original edges marked to be beveled have been removed;
|
||||||
* now we need to link up the edges for this "corner" */
|
* now we need to link up the edges for this "corner" */
|
||||||
len = BME_cycle_length(BME_disk_getpointer(v->edge, v));
|
len = BME_cycle_length(BME_disk_getpointer(v->edge, v));
|
||||||
|
|||||||
@@ -39,6 +39,11 @@
|
|||||||
|
|
||||||
#include "BKE_bmesh.h"
|
#include "BKE_bmesh.h"
|
||||||
|
|
||||||
|
struct BME_mempool *BME_mempool_create(int esize, int tote, int pchunk);
|
||||||
|
void BME_mempool_destroy(struct BME_mempool *pool);
|
||||||
|
void *BME_mempool_alloc(struct BME_mempool *pool);
|
||||||
|
void BME_mempool_free(struct BME_mempool *pool, void *address);
|
||||||
|
|
||||||
/*ALLOCATION/DEALLOCATION*/
|
/*ALLOCATION/DEALLOCATION*/
|
||||||
struct BME_Vert *BME_addvertlist(struct BME_Mesh *bm, struct BME_Vert *example);
|
struct BME_Vert *BME_addvertlist(struct BME_Mesh *bm, struct BME_Vert *example);
|
||||||
struct BME_Edge *BME_addedgelist(struct BME_Mesh *bm, struct BME_Vert *v1, struct BME_Vert *v2, struct BME_Edge *example);
|
struct BME_Edge *BME_addedgelist(struct BME_Mesh *bm, struct BME_Vert *v1, struct BME_Vert *v2, struct BME_Edge *example);
|
||||||
@@ -49,7 +54,7 @@ void BME_free_vert(struct BME_Mesh *bm, struct BME_Vert *v);
|
|||||||
void BME_free_edge(struct BME_Mesh *bm, struct BME_Edge *e);
|
void BME_free_edge(struct BME_Mesh *bm, struct BME_Edge *e);
|
||||||
void BME_free_poly(struct BME_Mesh *bm, struct BME_Poly *f);
|
void BME_free_poly(struct BME_Mesh *bm, struct BME_Poly *f);
|
||||||
void BME_free_loop(struct BME_Mesh *bm, struct BME_Loop *l);
|
void BME_free_loop(struct BME_Mesh *bm, struct BME_Loop *l);
|
||||||
void BME_delete_loop(struct BME_Mesh *bm, struct BME_Loop *l);
|
//void BME_delete_loop(struct BME_Mesh *bm, struct BME_Loop *l);
|
||||||
|
|
||||||
/*DOUBLE CIRCULAR LINKED LIST FUNCTIONS*/
|
/*DOUBLE CIRCULAR LINKED LIST FUNCTIONS*/
|
||||||
void BME_cycle_append(void *h, void *nt);
|
void BME_cycle_append(void *h, void *nt);
|
||||||
|
|||||||
@@ -45,6 +45,8 @@
|
|||||||
|
|
||||||
#include "BKE_pointcache.h"
|
#include "BKE_pointcache.h"
|
||||||
|
|
||||||
|
#include "BLI_kdopbvh.h"
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
void tstart ( void )
|
void tstart ( void )
|
||||||
{}
|
{}
|
||||||
@@ -151,13 +153,14 @@ void cloth_init ( ClothModifierData *clmd )
|
|||||||
clmd->sim_parms->goalfrict = 0.0f;
|
clmd->sim_parms->goalfrict = 0.0f;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BVHTree *bvhselftree_build_from_cloth (ClothModifierData *clmd, float epsilon)
|
||||||
BVH *bvh_build_from_cloth (ClothModifierData *clmd, float epsilon)
|
|
||||||
{
|
{
|
||||||
unsigned int i = 0;
|
int i;
|
||||||
BVH *bvh=NULL;
|
BVHTree *bvhtree;
|
||||||
Cloth *cloth = clmd->clothObject;
|
Cloth *cloth = clmd->clothObject;
|
||||||
ClothVertex *verts = NULL;
|
ClothVertex *verts;
|
||||||
|
MFace *mfaces;
|
||||||
|
float co[12];
|
||||||
|
|
||||||
if(!clmd)
|
if(!clmd)
|
||||||
return NULL;
|
return NULL;
|
||||||
@@ -168,69 +171,171 @@ BVH *bvh_build_from_cloth (ClothModifierData *clmd, float epsilon)
|
|||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
verts = cloth->verts;
|
verts = cloth->verts;
|
||||||
|
mfaces = cloth->mfaces;
|
||||||
|
|
||||||
|
// in the moment, return zero if no faces there
|
||||||
|
if(!cloth->numverts)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
// create quadtree with k=26
|
||||||
|
bvhtree = BLI_bvhtree_new(cloth->numverts, epsilon, 4, 6);
|
||||||
|
|
||||||
|
// fill tree
|
||||||
|
for(i = 0; i < cloth->numverts; i++, verts++)
|
||||||
|
{
|
||||||
|
VECCOPY(&co[0*3], verts->xold);
|
||||||
|
|
||||||
|
BLI_bvhtree_insert(bvhtree, i, co, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// balance tree
|
||||||
|
BLI_bvhtree_balance(bvhtree);
|
||||||
|
|
||||||
|
return bvhtree;
|
||||||
|
}
|
||||||
|
|
||||||
|
BVHTree *bvhtree_build_from_cloth (ClothModifierData *clmd, float epsilon)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
BVHTree *bvhtree;
|
||||||
|
Cloth *cloth = clmd->clothObject;
|
||||||
|
ClothVertex *verts;
|
||||||
|
MFace *mfaces;
|
||||||
|
float co[12];
|
||||||
|
|
||||||
|
if(!clmd)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
cloth = clmd->clothObject;
|
||||||
|
|
||||||
|
if(!cloth)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
verts = cloth->verts;
|
||||||
|
mfaces = cloth->mfaces;
|
||||||
|
|
||||||
// in the moment, return zero if no faces there
|
// in the moment, return zero if no faces there
|
||||||
if(!cloth->numfaces)
|
if(!cloth->numfaces)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
bvh = MEM_callocN(sizeof(BVH), "BVH");
|
// create quadtree with k=26
|
||||||
if (bvh == NULL)
|
bvhtree = BLI_bvhtree_new(cloth->numfaces, epsilon, 4, 26);
|
||||||
|
|
||||||
|
// fill tree
|
||||||
|
for(i = 0; i < cloth->numfaces; i++, mfaces++)
|
||||||
{
|
{
|
||||||
printf("bvh: Out of memory.\n");
|
VECCOPY(&co[0*3], verts[mfaces->v1].xold);
|
||||||
return NULL;
|
VECCOPY(&co[1*3], verts[mfaces->v2].xold);
|
||||||
|
VECCOPY(&co[2*3], verts[mfaces->v3].xold);
|
||||||
|
|
||||||
|
if(mfaces->v4)
|
||||||
|
VECCOPY(&co[3*3], verts[mfaces->v4].xold);
|
||||||
|
|
||||||
|
BLI_bvhtree_insert(bvhtree, i, co, (mfaces->v4 ? 4 : 3));
|
||||||
}
|
}
|
||||||
|
|
||||||
// springs = cloth->springs;
|
// balance tree
|
||||||
// numsprings = cloth->numsprings;
|
BLI_bvhtree_balance(bvhtree);
|
||||||
|
|
||||||
bvh->epsilon = epsilon;
|
return bvhtree;
|
||||||
bvh->numfaces = cloth->numfaces;
|
|
||||||
bvh->mfaces = cloth->mfaces;
|
|
||||||
|
|
||||||
bvh->numverts = cloth->numverts;
|
|
||||||
|
|
||||||
bvh->current_x = MEM_callocN ( sizeof ( MVert ) * bvh->numverts, "bvh->current_x" );
|
|
||||||
|
|
||||||
if (bvh->current_x == NULL)
|
|
||||||
{
|
|
||||||
printf("bvh: Out of memory.\n");
|
|
||||||
MEM_freeN(bvh);
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for(i = 0; i < bvh->numverts; i++)
|
void bvhtree_update_from_cloth(ClothModifierData *clmd, int moving)
|
||||||
{
|
|
||||||
VECCOPY(bvh->current_x[i].co, verts[i].tx);
|
|
||||||
}
|
|
||||||
|
|
||||||
bvh_build (bvh);
|
|
||||||
|
|
||||||
return bvh;
|
|
||||||
}
|
|
||||||
|
|
||||||
void bvh_update_from_cloth(ClothModifierData *clmd, int moving)
|
|
||||||
{
|
{
|
||||||
unsigned int i = 0;
|
unsigned int i = 0;
|
||||||
Cloth *cloth = clmd->clothObject;
|
Cloth *cloth = clmd->clothObject;
|
||||||
BVH *bvh = cloth->tree;
|
BVHTree *bvhtree = cloth->bvhtree;
|
||||||
ClothVertex *verts = cloth->verts;
|
ClothVertex *verts = cloth->verts;
|
||||||
|
MFace *mfaces;
|
||||||
|
float co[12], co_moving[12];
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
if(!bvh)
|
if(!bvhtree)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if(cloth->numverts!=bvh->numverts)
|
mfaces = cloth->mfaces;
|
||||||
return;
|
|
||||||
|
|
||||||
if(cloth->verts)
|
// update vertex position in bvh tree
|
||||||
|
if(verts && mfaces)
|
||||||
{
|
{
|
||||||
for(i = 0; i < bvh->numverts; i++)
|
for(i = 0; i < cloth->numfaces; i++, mfaces++)
|
||||||
{
|
{
|
||||||
VECCOPY(bvh->current_x[i].co, verts[i].tx);
|
VECCOPY(&co[0*3], verts[mfaces->v1].txold);
|
||||||
VECCOPY(bvh->current_xold[i].co, verts[i].txold);
|
VECCOPY(&co[1*3], verts[mfaces->v2].txold);
|
||||||
|
VECCOPY(&co[2*3], verts[mfaces->v3].txold);
|
||||||
|
|
||||||
|
if(mfaces->v4)
|
||||||
|
VECCOPY(&co[3*3], verts[mfaces->v4].txold);
|
||||||
|
|
||||||
|
// copy new locations into array
|
||||||
|
if(moving)
|
||||||
|
{
|
||||||
|
// update moving positions
|
||||||
|
VECCOPY(&co_moving[0*3], verts[mfaces->v1].tx);
|
||||||
|
VECCOPY(&co_moving[1*3], verts[mfaces->v2].tx);
|
||||||
|
VECCOPY(&co_moving[2*3], verts[mfaces->v3].tx);
|
||||||
|
|
||||||
|
if(mfaces->v4)
|
||||||
|
VECCOPY(&co_moving[3*3], verts[mfaces->v4].tx);
|
||||||
|
|
||||||
|
ret = BLI_bvhtree_update_node(bvhtree, i, co, co_moving, (mfaces->v4 ? 4 : 3));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ret = BLI_bvhtree_update_node(bvhtree, i, co, NULL, (mfaces->v4 ? 4 : 3));
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if tree is already full
|
||||||
|
if(!ret)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
BLI_bvhtree_update_tree(bvhtree);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bvh_update(bvh, moving);
|
void bvhselftree_update_from_cloth(ClothModifierData *clmd, int moving)
|
||||||
|
{
|
||||||
|
unsigned int i = 0;
|
||||||
|
Cloth *cloth = clmd->clothObject;
|
||||||
|
BVHTree *bvhtree = cloth->bvhselftree;
|
||||||
|
ClothVertex *verts = cloth->verts;
|
||||||
|
MFace *mfaces;
|
||||||
|
float co[12], co_moving[12];
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
|
if(!bvhtree)
|
||||||
|
return;
|
||||||
|
|
||||||
|
mfaces = cloth->mfaces;
|
||||||
|
|
||||||
|
// update vertex position in bvh tree
|
||||||
|
if(verts && mfaces)
|
||||||
|
{
|
||||||
|
for(i = 0; i < cloth->numverts; i++, verts++)
|
||||||
|
{
|
||||||
|
VECCOPY(&co[0*3], verts->txold);
|
||||||
|
|
||||||
|
// copy new locations into array
|
||||||
|
if(moving)
|
||||||
|
{
|
||||||
|
// update moving positions
|
||||||
|
VECCOPY(&co_moving[0*3], verts->tx);
|
||||||
|
|
||||||
|
ret = BLI_bvhtree_update_node(bvhtree, i, co, co_moving, 1);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ret = BLI_bvhtree_update_node(bvhtree, i, co, NULL, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if tree is already full
|
||||||
|
if(!ret)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
BLI_bvhtree_update_tree(bvhtree);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int modifiers_indexInObject(Object *ob, ModifierData *md_seek);
|
int modifiers_indexInObject(Object *ob, ModifierData *md_seek);
|
||||||
@@ -541,8 +646,11 @@ void cloth_free_modifier ( Object *ob, ClothModifierData *clmd )
|
|||||||
cloth->numsprings = 0;
|
cloth->numsprings = 0;
|
||||||
|
|
||||||
// free BVH collision tree
|
// free BVH collision tree
|
||||||
if ( cloth->tree )
|
if ( cloth->bvhtree )
|
||||||
bvh_free ( ( BVH * ) cloth->tree );
|
BLI_bvhtree_free ( cloth->bvhtree );
|
||||||
|
|
||||||
|
if ( cloth->bvhselftree )
|
||||||
|
BLI_bvhtree_free ( cloth->bvhselftree );
|
||||||
|
|
||||||
// we save our faces for collision objects
|
// we save our faces for collision objects
|
||||||
if ( cloth->mfaces )
|
if ( cloth->mfaces )
|
||||||
@@ -611,8 +719,11 @@ void cloth_free_modifier_extern ( ClothModifierData *clmd )
|
|||||||
cloth->numsprings = 0;
|
cloth->numsprings = 0;
|
||||||
|
|
||||||
// free BVH collision tree
|
// free BVH collision tree
|
||||||
if ( cloth->tree )
|
if ( cloth->bvhtree )
|
||||||
bvh_free ( ( BVH * ) cloth->tree );
|
BLI_bvhtree_free ( cloth->bvhtree );
|
||||||
|
|
||||||
|
if ( cloth->bvhselftree )
|
||||||
|
BLI_bvhtree_free ( cloth->bvhselftree );
|
||||||
|
|
||||||
// we save our faces for collision objects
|
// we save our faces for collision objects
|
||||||
if ( cloth->mfaces )
|
if ( cloth->mfaces )
|
||||||
@@ -751,6 +862,7 @@ static int cloth_from_object(Object *ob, ClothModifierData *clmd, DerivedMesh *d
|
|||||||
ClothVertex *verts = NULL;
|
ClothVertex *verts = NULL;
|
||||||
float tnull[3] = {0,0,0};
|
float tnull[3] = {0,0,0};
|
||||||
Cloth *cloth = NULL;
|
Cloth *cloth = NULL;
|
||||||
|
float maxdist = 0;
|
||||||
|
|
||||||
// If we have a clothObject, free it.
|
// If we have a clothObject, free it.
|
||||||
if ( clmd->clothObject != NULL )
|
if ( clmd->clothObject != NULL )
|
||||||
@@ -810,6 +922,7 @@ static int cloth_from_object(Object *ob, ClothModifierData *clmd, DerivedMesh *d
|
|||||||
VECCOPY ( verts->xold, verts->x );
|
VECCOPY ( verts->xold, verts->x );
|
||||||
VECCOPY ( verts->xconst, verts->x );
|
VECCOPY ( verts->xconst, verts->x );
|
||||||
VECCOPY ( verts->txold, verts->x );
|
VECCOPY ( verts->txold, verts->x );
|
||||||
|
VECCOPY ( verts->tx, verts->x );
|
||||||
VecMulf ( verts->v, 0.0f );
|
VecMulf ( verts->v, 0.0f );
|
||||||
|
|
||||||
verts->impulse_count = 0;
|
verts->impulse_count = 0;
|
||||||
@@ -820,7 +933,6 @@ static int cloth_from_object(Object *ob, ClothModifierData *clmd, DerivedMesh *d
|
|||||||
// has to be happen before springs are build!
|
// has to be happen before springs are build!
|
||||||
cloth_apply_vgroup (clmd, dm);
|
cloth_apply_vgroup (clmd, dm);
|
||||||
|
|
||||||
|
|
||||||
if ( !cloth_build_springs ( clmd, dm ) )
|
if ( !cloth_build_springs ( clmd, dm ) )
|
||||||
{
|
{
|
||||||
cloth_free_modifier ( ob, clmd );
|
cloth_free_modifier ( ob, clmd );
|
||||||
@@ -845,12 +957,18 @@ static int cloth_from_object(Object *ob, ClothModifierData *clmd, DerivedMesh *d
|
|||||||
if(!first)
|
if(!first)
|
||||||
implicit_set_positions(clmd);
|
implicit_set_positions(clmd);
|
||||||
|
|
||||||
clmd->clothObject->tree = bvh_build_from_cloth ( clmd, clmd->coll_parms->epsilon );
|
clmd->clothObject->bvhtree = bvhtree_build_from_cloth ( clmd, clmd->coll_parms->epsilon );
|
||||||
|
|
||||||
|
for(i = 0; i < dm->getNumVerts(dm); i++)
|
||||||
|
{
|
||||||
|
maxdist = MAX2(maxdist, clmd->coll_parms->selfepsilon* ( cloth->verts[i].avg_spring_len*2.0));
|
||||||
|
}
|
||||||
|
|
||||||
|
clmd->clothObject->bvhselftree = bvhselftree_build_from_cloth ( clmd, maxdist );
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void cloth_from_mesh ( Object *ob, ClothModifierData *clmd, DerivedMesh *dm )
|
static void cloth_from_mesh ( Object *ob, ClothModifierData *clmd, DerivedMesh *dm )
|
||||||
{
|
{
|
||||||
unsigned int numverts = dm->getNumVerts ( dm );
|
unsigned int numverts = dm->getNumVerts ( dm );
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1882,7 +1882,7 @@ static void pycon_get_tarmat (bConstraint *con, bConstraintOb *cob, bConstraintT
|
|||||||
{
|
{
|
||||||
bPythonConstraint *data= con->data;
|
bPythonConstraint *data= con->data;
|
||||||
|
|
||||||
if (VALID_CONS_TARGET(ct)) {
|
if ((G.f & G_DOSCRIPTLINKS) && VALID_CONS_TARGET(ct)) {
|
||||||
/* special exception for curves - depsgraph issues */
|
/* special exception for curves - depsgraph issues */
|
||||||
if (ct->tar->type == OB_CURVE) {
|
if (ct->tar->type == OB_CURVE) {
|
||||||
Curve *cu= ct->tar->data;
|
Curve *cu= ct->tar->data;
|
||||||
@@ -1906,6 +1906,8 @@ static void pycon_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targ
|
|||||||
{
|
{
|
||||||
bPythonConstraint *data= con->data;
|
bPythonConstraint *data= con->data;
|
||||||
|
|
||||||
|
if ((G.f & G_DOSCRIPTLINKS)==0) return;
|
||||||
|
|
||||||
/* currently removed, until I this can be re-implemented for multiple targets */
|
/* currently removed, until I this can be re-implemented for multiple targets */
|
||||||
#if 0
|
#if 0
|
||||||
/* Firstly, run the 'driver' function which has direct access to the objects involved
|
/* Firstly, run the 'driver' function which has direct access to the objects involved
|
||||||
|
|||||||
@@ -1278,7 +1278,7 @@ void *CustomData_em_get_n(const CustomData *data, void *block, int type, int n)
|
|||||||
int layer_index;
|
int layer_index;
|
||||||
|
|
||||||
/* get the layer index of the first layer of type */
|
/* get the layer index of the first layer of type */
|
||||||
layer_index = CustomData_get_active_layer_index(data, type);
|
layer_index = CustomData_get_layer_index(data, type);
|
||||||
if(layer_index < 0) return NULL;
|
if(layer_index < 0) return NULL;
|
||||||
|
|
||||||
return (char *)block + data->layers[layer_index+n].offset;
|
return (char *)block + data->layers[layer_index+n].offset;
|
||||||
|
|||||||
@@ -440,7 +440,7 @@ static void build_underline(Curve *cu, float x1, float y1, float x2, float y2, i
|
|||||||
if (nu2 == NULL) return;
|
if (nu2 == NULL) return;
|
||||||
nu2->resolu= cu->resolu;
|
nu2->resolu= cu->resolu;
|
||||||
nu2->bezt = NULL;
|
nu2->bezt = NULL;
|
||||||
nu2->knotsu = nu2->knotsv = 0;
|
nu2->knotsu = nu2->knotsv = NULL;
|
||||||
nu2->flag= 0;
|
nu2->flag= 0;
|
||||||
nu2->charidx = charidx+1000;
|
nu2->charidx = charidx+1000;
|
||||||
if (mat_nr > 0) nu2->mat_nr= mat_nr-1;
|
if (mat_nr > 0) nu2->mat_nr= mat_nr-1;
|
||||||
@@ -529,7 +529,7 @@ static void buildchar(Curve *cu, unsigned long character, CharInfo *info, float
|
|||||||
memcpy(nu2, nu1, sizeof(struct Nurb));
|
memcpy(nu2, nu1, sizeof(struct Nurb));
|
||||||
nu2->resolu= cu->resolu;
|
nu2->resolu= cu->resolu;
|
||||||
nu2->bp = 0;
|
nu2->bp = 0;
|
||||||
nu2->knotsu = nu2->knotsv = 0;
|
nu2->knotsu = nu2->knotsv = NULL;
|
||||||
nu2->flag= CU_SMOOTH;
|
nu2->flag= CU_SMOOTH;
|
||||||
nu2->charidx = charidx;
|
nu2->charidx = charidx;
|
||||||
if (info->mat_nr) {
|
if (info->mat_nr) {
|
||||||
|
|||||||
@@ -1,860 +0,0 @@
|
|||||||
/* kdop.c
|
|
||||||
*
|
|
||||||
*
|
|
||||||
* ***** BEGIN GPL LICENSE BLOCK *****
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU General Public License
|
|
||||||
* as published by the Free Software Foundation; either version 2
|
|
||||||
* of the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program; if not, write to the Free Software Foundation,
|
|
||||||
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
||||||
*
|
|
||||||
* The Original Code is Copyright (C) Blender Foundation
|
|
||||||
* All rights reserved.
|
|
||||||
*
|
|
||||||
* The Original Code is: all of this file.
|
|
||||||
*
|
|
||||||
* Contributor(s): none yet.
|
|
||||||
*
|
|
||||||
* ***** END GPL LICENSE BLOCK *****
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "MEM_guardedalloc.h"
|
|
||||||
|
|
||||||
#include "BKE_cloth.h"
|
|
||||||
|
|
||||||
#include "DNA_cloth_types.h"
|
|
||||||
#include "DNA_mesh_types.h"
|
|
||||||
#include "DNA_scene_types.h"
|
|
||||||
|
|
||||||
#include "BKE_deform.h"
|
|
||||||
#include "BKE_DerivedMesh.h"
|
|
||||||
#include "BKE_cdderivedmesh.h"
|
|
||||||
#include "BKE_effect.h"
|
|
||||||
#include "BKE_global.h"
|
|
||||||
#include "BKE_object.h"
|
|
||||||
#include "BKE_modifier.h"
|
|
||||||
#include "BKE_utildefines.h"
|
|
||||||
|
|
||||||
#ifdef _OPENMP
|
|
||||||
#include <omp.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////
|
|
||||||
// Additional fastened appending function
|
|
||||||
// It uses the link to the last inserted node as start value
|
|
||||||
// for searching the end of the list
|
|
||||||
// NEW: in compare to the original function, this one returns
|
|
||||||
// the reference to the last inserted node
|
|
||||||
////////////////////////////////////////////////////////////////////////
|
|
||||||
LinkNode *BLI_linklist_append_fast(LinkNode **listp, void *ptr) {
|
|
||||||
LinkNode *nlink= MEM_mallocN(sizeof(*nlink), "nlink");
|
|
||||||
LinkNode *node = *listp;
|
|
||||||
|
|
||||||
nlink->link = ptr;
|
|
||||||
nlink->next = NULL;
|
|
||||||
|
|
||||||
if(node == NULL){
|
|
||||||
*listp = nlink;
|
|
||||||
} else {
|
|
||||||
while(node->next != NULL){
|
|
||||||
node = node->next;
|
|
||||||
}
|
|
||||||
node->next = nlink;
|
|
||||||
}
|
|
||||||
return nlink;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////
|
|
||||||
// Bounding Volume Hierarchy Definition
|
|
||||||
//
|
|
||||||
// Notes: From OBB until 26-DOP --> all bounding volumes possible, just choose type below
|
|
||||||
// Notes: You have to choose the type at compile time ITM
|
|
||||||
// Notes: You can choose the tree type --> binary, quad, octree, choose below
|
|
||||||
////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
static float KDOP_AXES[13][3] =
|
|
||||||
{ {1.0, 0, 0}, {0, 1.0, 0}, {0, 0, 1.0}, {1.0, 1.0, 1.0}, {1.0, -1.0, 1.0}, {1.0, 1.0, -1.0},
|
|
||||||
{1.0, -1.0, -1.0}, {1.0, 1.0, 0}, {1.0, 0, 1.0}, {0, 1.0, 1.0}, {1.0, -1.0, 0}, {1.0, 0, -1.0},
|
|
||||||
{0, 1.0, -1.0}
|
|
||||||
};
|
|
||||||
|
|
||||||
///////////// choose bounding volume here! /////////////
|
|
||||||
|
|
||||||
#define KDOP_26
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#ifdef KDOP_26
|
|
||||||
#define KDOP_END 13
|
|
||||||
#define KDOP_START 0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef KDOP_18
|
|
||||||
#define KDOP_END 13
|
|
||||||
#define KDOP_START 7
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef KDOP_14
|
|
||||||
#define KDOP_END 7
|
|
||||||
#define KDOP_START 0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// this is basicly some AABB
|
|
||||||
#ifdef KDOP_8
|
|
||||||
#define KDOP_END 4
|
|
||||||
#define KDOP_START 0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// this is basicly some OBB
|
|
||||||
#ifdef KDOP_6
|
|
||||||
#define KDOP_END 3
|
|
||||||
#define KDOP_START 0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Introsort
|
|
||||||
// with permission deriven from the following Java code:
|
|
||||||
// http://ralphunden.net/content/tutorials/a-guide-to-introsort/
|
|
||||||
// and he derived it from the SUN STL
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
static int size_threshold = 16;
|
|
||||||
/*
|
|
||||||
* Common methods for all algorithms
|
|
||||||
*/
|
|
||||||
DO_INLINE void bvh_exchange(CollisionTree **a, int i, int j)
|
|
||||||
{
|
|
||||||
CollisionTree *t=a[i];
|
|
||||||
a[i]=a[j];
|
|
||||||
a[j]=t;
|
|
||||||
}
|
|
||||||
DO_INLINE int floor_lg(int a)
|
|
||||||
{
|
|
||||||
return (int)(floor(log(a)/log(2)));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Insertion sort algorithm
|
|
||||||
*/
|
|
||||||
void bvh_insertionsort(CollisionTree **a, int lo, int hi, int axis)
|
|
||||||
{
|
|
||||||
int i,j;
|
|
||||||
CollisionTree *t;
|
|
||||||
for (i=lo; i < hi; i++)
|
|
||||||
{
|
|
||||||
j=i;
|
|
||||||
t = a[i];
|
|
||||||
while((j!=lo) && (t->bv[axis] < (a[j-1])->bv[axis]))
|
|
||||||
{
|
|
||||||
a[j] = a[j-1];
|
|
||||||
j--;
|
|
||||||
}
|
|
||||||
a[j] = t;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static int bvh_partition(CollisionTree **a, int lo, int hi, CollisionTree * x, int axis)
|
|
||||||
{
|
|
||||||
int i=lo, j=hi;
|
|
||||||
while (1)
|
|
||||||
{
|
|
||||||
while ((a[i])->bv[axis] < x->bv[axis]) i++;
|
|
||||||
j=j-1;
|
|
||||||
while (x->bv[axis] < (a[j])->bv[axis]) j=j-1;
|
|
||||||
if(!(i < j))
|
|
||||||
return i;
|
|
||||||
bvh_exchange(a, i,j);
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Heapsort algorithm
|
|
||||||
*/
|
|
||||||
static void bvh_downheap(CollisionTree **a, int i, int n, int lo, int axis)
|
|
||||||
{
|
|
||||||
CollisionTree * d = a[lo+i-1];
|
|
||||||
int child;
|
|
||||||
while (i<=n/2)
|
|
||||||
{
|
|
||||||
child = 2*i;
|
|
||||||
if ((child < n) && ((a[lo+child-1])->bv[axis] < (a[lo+child])->bv[axis]))
|
|
||||||
{
|
|
||||||
child++;
|
|
||||||
}
|
|
||||||
if (!(d->bv[axis] < (a[lo+child-1])->bv[axis])) break;
|
|
||||||
a[lo+i-1] = a[lo+child-1];
|
|
||||||
i = child;
|
|
||||||
}
|
|
||||||
a[lo+i-1] = d;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void bvh_heapsort(CollisionTree **a, int lo, int hi, int axis)
|
|
||||||
{
|
|
||||||
int n = hi-lo, i;
|
|
||||||
for (i=n/2; i>=1; i=i-1)
|
|
||||||
{
|
|
||||||
bvh_downheap(a, i,n,lo, axis);
|
|
||||||
}
|
|
||||||
for (i=n; i>1; i=i-1)
|
|
||||||
{
|
|
||||||
bvh_exchange(a, lo,lo+i-1);
|
|
||||||
bvh_downheap(a, 1,i-1,lo, axis);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static CollisionTree *bvh_medianof3(CollisionTree **a, int lo, int mid, int hi, int axis) // returns Sortable
|
|
||||||
{
|
|
||||||
if ((a[mid])->bv[axis] < (a[lo])->bv[axis])
|
|
||||||
{
|
|
||||||
if ((a[hi])->bv[axis] < (a[mid])->bv[axis])
|
|
||||||
return a[mid];
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if ((a[hi])->bv[axis] < (a[lo])->bv[axis])
|
|
||||||
return a[hi];
|
|
||||||
else
|
|
||||||
return a[lo];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if ((a[hi])->bv[axis] < (a[mid])->bv[axis])
|
|
||||||
{
|
|
||||||
if ((a[hi])->bv[axis] < (a[lo])->bv[axis])
|
|
||||||
return a[lo];
|
|
||||||
else
|
|
||||||
return a[hi];
|
|
||||||
}
|
|
||||||
else
|
|
||||||
return a[mid];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/*
|
|
||||||
* Quicksort algorithm modified for Introsort
|
|
||||||
*/
|
|
||||||
void bvh_introsort_loop (CollisionTree **a, int lo, int hi, int depth_limit, int axis)
|
|
||||||
{
|
|
||||||
int p;
|
|
||||||
|
|
||||||
while (hi-lo > size_threshold)
|
|
||||||
{
|
|
||||||
if (depth_limit == 0)
|
|
||||||
{
|
|
||||||
bvh_heapsort(a, lo, hi, axis);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
depth_limit=depth_limit-1;
|
|
||||||
p=bvh_partition(a, lo, hi, bvh_medianof3(a, lo, lo+((hi-lo)/2)+1, hi-1, axis), axis);
|
|
||||||
bvh_introsort_loop(a, p, hi, depth_limit, axis);
|
|
||||||
hi=p;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
DO_INLINE void bvh_sort(CollisionTree **a0, int begin, int end, int axis)
|
|
||||||
{
|
|
||||||
if (begin < end)
|
|
||||||
{
|
|
||||||
CollisionTree **a=a0;
|
|
||||||
bvh_introsort_loop(a, begin, end, 2*floor_lg(end-begin), axis);
|
|
||||||
bvh_insertionsort(a, begin, end, axis);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
DO_INLINE void bvh_sort_along_axis(CollisionTree **face_list, int start, int end, int axis)
|
|
||||||
{
|
|
||||||
bvh_sort(face_list, start, end, axis);
|
|
||||||
}
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
void bvh_free(BVH * bvh)
|
|
||||||
{
|
|
||||||
LinkNode *search = NULL;
|
|
||||||
CollisionTree *tree = NULL;
|
|
||||||
|
|
||||||
if (bvh)
|
|
||||||
{
|
|
||||||
|
|
||||||
search = bvh->tree;
|
|
||||||
|
|
||||||
while(search)
|
|
||||||
{
|
|
||||||
LinkNode *next= search->next;
|
|
||||||
tree = search->link;
|
|
||||||
|
|
||||||
MEM_freeN(tree);
|
|
||||||
|
|
||||||
search = next;
|
|
||||||
}
|
|
||||||
|
|
||||||
BLI_linklist_free(bvh->tree,NULL);
|
|
||||||
bvh->tree = NULL;
|
|
||||||
|
|
||||||
if(bvh->current_x)
|
|
||||||
MEM_freeN(bvh->current_x);
|
|
||||||
if(bvh->current_xold)
|
|
||||||
MEM_freeN(bvh->current_xold);
|
|
||||||
|
|
||||||
MEM_freeN(bvh);
|
|
||||||
bvh = NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// only supports x,y,z axis in the moment
|
|
||||||
// but we should use a plain and simple function here for speed sake
|
|
||||||
DO_INLINE int bvh_largest_axis(float *bv)
|
|
||||||
{
|
|
||||||
float middle_point[3];
|
|
||||||
|
|
||||||
middle_point[0] = (bv[1]) - (bv[0]); // x axis
|
|
||||||
middle_point[1] = (bv[3]) - (bv[2]); // y axis
|
|
||||||
middle_point[2] = (bv[5]) - (bv[4]); // z axis
|
|
||||||
if (middle_point[0] > middle_point[1])
|
|
||||||
{
|
|
||||||
if (middle_point[0] > middle_point[2])
|
|
||||||
return 1; // max x axis
|
|
||||||
else
|
|
||||||
return 5; // max z axis
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (middle_point[1] > middle_point[2])
|
|
||||||
return 3; // max y axis
|
|
||||||
else
|
|
||||||
return 5; // max z axis
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// depends on the fact that the BVH's for each face is already build
|
|
||||||
DO_INLINE void bvh_calc_DOP_hull_from_faces(BVH * bvh, CollisionTree **tri, int numfaces, float *bv)
|
|
||||||
{
|
|
||||||
float newmin,newmax;
|
|
||||||
int i, j;
|
|
||||||
|
|
||||||
if(numfaces >0)
|
|
||||||
{
|
|
||||||
// for all Axes.
|
|
||||||
for (i = KDOP_START; i < KDOP_END; i++)
|
|
||||||
{
|
|
||||||
bv[(2 * i)] = (tri [0])->bv[(2 * i)];
|
|
||||||
bv[(2 * i) + 1] = (tri [0])->bv[(2 * i) + 1];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (j = 0; j < numfaces; j++)
|
|
||||||
{
|
|
||||||
// for all Axes.
|
|
||||||
for (i = KDOP_START; i < KDOP_END; i++)
|
|
||||||
{
|
|
||||||
newmin = (tri [j])->bv[(2 * i)];
|
|
||||||
if ((newmin < bv[(2 * i)]))
|
|
||||||
{
|
|
||||||
bv[(2 * i)] = newmin;
|
|
||||||
}
|
|
||||||
|
|
||||||
newmax = (tri [j])->bv[(2 * i) + 1];
|
|
||||||
if ((newmax > bv[(2 * i) + 1]))
|
|
||||||
{
|
|
||||||
bv[(2 * i) + 1] = newmax;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
DO_INLINE void bvh_calc_DOP_hull_static(BVH * bvh, CollisionTree **tri, int numfaces, float *bv, CollisionTree *tree)
|
|
||||||
{
|
|
||||||
MFace *tempMFace = bvh->mfaces;
|
|
||||||
float *tempBV = bv;
|
|
||||||
float newminmax;
|
|
||||||
int i, j, k;
|
|
||||||
|
|
||||||
for (j = 0; j < numfaces; j++)
|
|
||||||
{
|
|
||||||
tempMFace = bvh->mfaces + (tri [j])->tri_index;
|
|
||||||
// 3 or 4 vertices per face.
|
|
||||||
for (k = 0; k < 4; k++)
|
|
||||||
{
|
|
||||||
int temp = 0;
|
|
||||||
// If this is a triangle.
|
|
||||||
if (k == 3 && !tempMFace->v4)
|
|
||||||
continue;
|
|
||||||
// TODO: other name for "temp" this gets all vertices of a face
|
|
||||||
if (k == 0)
|
|
||||||
temp = tempMFace->v1;
|
|
||||||
else if (k == 1)
|
|
||||||
temp = tempMFace->v2;
|
|
||||||
else if (k == 2)
|
|
||||||
temp = tempMFace->v3;
|
|
||||||
else if (k == 3)
|
|
||||||
temp = tempMFace->v4;
|
|
||||||
// for all Axes.
|
|
||||||
for (i = KDOP_START; i < KDOP_END; i++)
|
|
||||||
{
|
|
||||||
newminmax = INPR(bvh->current_xold[temp].co, KDOP_AXES[i]);
|
|
||||||
if ((newminmax < tempBV[(2 * i)]) || (k == 0 && j == 0))
|
|
||||||
tempBV[(2 * i)] = newminmax;
|
|
||||||
if ((newminmax > tempBV[(2 * i) + 1])|| (k == 0 && j == 0))
|
|
||||||
tempBV[(2 * i) + 1] = newminmax;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* calculate normal of this face */
|
|
||||||
/* (code copied from cdderivedmesh.c) */
|
|
||||||
/*
|
|
||||||
if(tempMFace->v4)
|
|
||||||
CalcNormFloat4(bvh->current_xold[tempMFace->v1].co, bvh->current_xold[tempMFace->v2].co,
|
|
||||||
bvh->current_xold[tempMFace->v3].co, bvh->current_xold[tempMFace->v4].co, tree->normal);
|
|
||||||
else
|
|
||||||
CalcNormFloat(bvh->current_xold[tempMFace->v1].co, bvh->current_xold[tempMFace->v2].co,
|
|
||||||
bvh->current_xold[tempMFace->v3].co, tree->normal);
|
|
||||||
|
|
||||||
tree->alpha = 0;
|
|
||||||
*/
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
DO_INLINE void bvh_calc_DOP_hull_moving(BVH * bvh, CollisionTree **tri, int numfaces, float *bv, CollisionTree *tree)
|
|
||||||
{
|
|
||||||
MFace *tempMFace = bvh->mfaces;
|
|
||||||
float *tempBV = bv;
|
|
||||||
float newminmax;
|
|
||||||
int i, j, k;
|
|
||||||
|
|
||||||
/* TODO: calculate normals */
|
|
||||||
|
|
||||||
for (j = 0; j < numfaces; j++)
|
|
||||||
{
|
|
||||||
tempMFace = bvh->mfaces + (tri [j])->tri_index;
|
|
||||||
// 3 or 4 vertices per face.
|
|
||||||
for (k = 0; k < 4; k++)
|
|
||||||
{
|
|
||||||
int temp = 0;
|
|
||||||
// If this is a triangle.
|
|
||||||
if (k == 3 && !tempMFace->v4)
|
|
||||||
continue;
|
|
||||||
// TODO: other name for "temp" this gets all vertices of a face
|
|
||||||
if (k == 0)
|
|
||||||
temp = tempMFace->v1;
|
|
||||||
else if (k == 1)
|
|
||||||
temp = tempMFace->v2;
|
|
||||||
else if (k == 2)
|
|
||||||
temp = tempMFace->v3;
|
|
||||||
else if (k == 3)
|
|
||||||
temp = tempMFace->v4;
|
|
||||||
// for all Axes.
|
|
||||||
for (i = KDOP_START; i < KDOP_END; i++)
|
|
||||||
{
|
|
||||||
newminmax = INPR(bvh->current_xold[temp].co, KDOP_AXES[i]);
|
|
||||||
if ((newminmax < tempBV[(2 * i)]) || (k == 0 && j == 0))
|
|
||||||
tempBV[(2 * i)] = newminmax;
|
|
||||||
if ((newminmax > tempBV[(2 * i) + 1])|| (k == 0 && j == 0))
|
|
||||||
tempBV[(2 * i) + 1] = newminmax;
|
|
||||||
|
|
||||||
newminmax = INPR(bvh->current_x[temp].co, KDOP_AXES[i]);
|
|
||||||
if ((newminmax < tempBV[(2 * i)]) || (k == 0 && j == 0))
|
|
||||||
tempBV[(2 * i)] = newminmax;
|
|
||||||
if ((newminmax > tempBV[(2 * i) + 1])|| (k == 0 && j == 0))
|
|
||||||
tempBV[(2 * i) + 1] = newminmax;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void bvh_div_env_node(BVH *bvh, CollisionTree *tree, CollisionTree **face_list, unsigned int start, unsigned int end, int lastaxis, LinkNode *nlink)
|
|
||||||
{
|
|
||||||
int i = 0;
|
|
||||||
CollisionTree *newtree = NULL;
|
|
||||||
int laxis = 0, max_nodes=4;
|
|
||||||
unsigned int tstart, tend;
|
|
||||||
LinkNode *nlink1 = nlink;
|
|
||||||
LinkNode *tnlink;
|
|
||||||
tree->traversed = 0;
|
|
||||||
// Determine which axis to split along
|
|
||||||
laxis = bvh_largest_axis(tree->bv);
|
|
||||||
|
|
||||||
// Sort along longest axis
|
|
||||||
if(laxis!=lastaxis)
|
|
||||||
bvh_sort_along_axis(face_list, start, end, laxis);
|
|
||||||
|
|
||||||
// maximum is 4 since we have a quad tree
|
|
||||||
max_nodes = MIN2((end-start + 1 ),4);
|
|
||||||
|
|
||||||
for (i = 0; i < max_nodes; i++)
|
|
||||||
{
|
|
||||||
tree->count_nodes++;
|
|
||||||
|
|
||||||
if(end-start+1 > 4)
|
|
||||||
{
|
|
||||||
int quarter = ((float)((float)(end - start + 1) / 4.0f));
|
|
||||||
tstart = start + i * quarter;
|
|
||||||
tend = tstart + quarter - 1;
|
|
||||||
|
|
||||||
// be sure that we get all faces
|
|
||||||
if(i==3)
|
|
||||||
{
|
|
||||||
tend = end;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
tend = tstart = start + i;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Build tree until 4 node left.
|
|
||||||
if ((tend-tstart + 1 ) > 1)
|
|
||||||
{
|
|
||||||
newtree = (CollisionTree *)MEM_callocN(sizeof(CollisionTree), "CollisionTree");
|
|
||||||
tnlink = BLI_linklist_append_fast(&nlink1->next, newtree);
|
|
||||||
|
|
||||||
newtree->nodes[0] = newtree->nodes[1] = newtree->nodes[2] = newtree->nodes[3] = NULL;
|
|
||||||
newtree->count_nodes = 0;
|
|
||||||
newtree->parent = tree;
|
|
||||||
newtree->isleaf = 0;
|
|
||||||
|
|
||||||
tree->nodes[i] = newtree;
|
|
||||||
|
|
||||||
nlink1 = tnlink;
|
|
||||||
|
|
||||||
bvh_calc_DOP_hull_from_faces(bvh, &face_list[tstart], tend-tstart + 1, tree->nodes[i]->bv);
|
|
||||||
|
|
||||||
bvh_div_env_node(bvh, tree->nodes[i], face_list, tstart, tend, laxis, nlink1);
|
|
||||||
}
|
|
||||||
else // ok, we have 1 left for this node
|
|
||||||
{
|
|
||||||
CollisionTree *tnode = face_list[tstart];
|
|
||||||
tree->nodes[i] = tnode;
|
|
||||||
tree->nodes[i]->parent = tree;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* function cannot be directly called - needs alloced bvh */
|
|
||||||
void bvh_build (BVH *bvh)
|
|
||||||
{
|
|
||||||
unsigned int i = 0, j = 0, k = 0;
|
|
||||||
CollisionTree **face_list=NULL;
|
|
||||||
CollisionTree *tree=NULL;
|
|
||||||
LinkNode *nlink = NULL;
|
|
||||||
|
|
||||||
bvh->flags = 0;
|
|
||||||
bvh->leaf_tree = NULL;
|
|
||||||
bvh->leaf_root = NULL;
|
|
||||||
bvh->tree = NULL;
|
|
||||||
|
|
||||||
if(!bvh->current_x)
|
|
||||||
{
|
|
||||||
bvh_free(bvh);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
bvh->current_xold = MEM_dupallocN(bvh->current_x);
|
|
||||||
|
|
||||||
tree = (CollisionTree *)MEM_callocN(sizeof(CollisionTree), "CollisionTree");
|
|
||||||
|
|
||||||
if (tree == NULL)
|
|
||||||
{
|
|
||||||
printf("bvh_build: Out of memory for nodes.\n");
|
|
||||||
bvh_free(bvh);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
BLI_linklist_append(&bvh->tree, tree);
|
|
||||||
|
|
||||||
nlink = bvh->tree;
|
|
||||||
|
|
||||||
bvh->root = bvh->tree->link;
|
|
||||||
bvh->root->isleaf = 0;
|
|
||||||
bvh->root->parent = NULL;
|
|
||||||
bvh->root->nodes[0] = bvh->root->nodes[1] = bvh->root->nodes[1] = bvh->root->nodes[3] = NULL;
|
|
||||||
|
|
||||||
if(bvh->numfaces<=1)
|
|
||||||
{
|
|
||||||
bvh->root->tri_index = 0; // Why that? --> only one face there
|
|
||||||
bvh->root->isleaf = 1;
|
|
||||||
bvh->root->traversed = 0;
|
|
||||||
bvh->root->count_nodes = 0;
|
|
||||||
bvh->leaf_root = bvh->root;
|
|
||||||
bvh->leaf_tree = bvh->root;
|
|
||||||
bvh->root->nextLeaf = NULL;
|
|
||||||
bvh->root->prevLeaf = NULL;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// create face boxes
|
|
||||||
face_list = MEM_callocN (bvh->numfaces * sizeof (CollisionTree *), "CollisionTree");
|
|
||||||
if (face_list == NULL)
|
|
||||||
{
|
|
||||||
printf("bvh_build: Out of memory for face_list.\n");
|
|
||||||
bvh_free(bvh);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// create face boxes
|
|
||||||
for(i = 0, k = 0; i < bvh->numfaces; i++)
|
|
||||||
{
|
|
||||||
LinkNode *tnlink;
|
|
||||||
|
|
||||||
tree = (CollisionTree *)MEM_callocN(sizeof(CollisionTree), "CollisionTree");
|
|
||||||
// TODO: check succesfull alloc
|
|
||||||
|
|
||||||
tnlink = BLI_linklist_append_fast(&nlink->next, tree);
|
|
||||||
|
|
||||||
face_list[i] = tree;
|
|
||||||
tree->tri_index = i;
|
|
||||||
tree->isleaf = 1;
|
|
||||||
tree->nextLeaf = NULL;
|
|
||||||
tree->prevLeaf = bvh->leaf_tree;
|
|
||||||
tree->parent = NULL;
|
|
||||||
tree->count_nodes = 0;
|
|
||||||
|
|
||||||
if(i==0)
|
|
||||||
{
|
|
||||||
bvh->leaf_tree = bvh->leaf_root = tree;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
bvh->leaf_tree->nextLeaf = tree;
|
|
||||||
bvh->leaf_tree = bvh->leaf_tree->nextLeaf;
|
|
||||||
}
|
|
||||||
|
|
||||||
tree->nodes[0] = tree->nodes[1] = tree->nodes[2] = tree->nodes[3] = NULL;
|
|
||||||
|
|
||||||
bvh_calc_DOP_hull_static(bvh, &face_list[i], 1, tree->bv, tree);
|
|
||||||
|
|
||||||
// inflate the bv with some epsilon
|
|
||||||
for (j = KDOP_START; j < KDOP_END; j++)
|
|
||||||
{
|
|
||||||
tree->bv[(2 * j)] -= bvh->epsilon; // minimum
|
|
||||||
tree->bv[(2 * j) + 1] += bvh->epsilon; // maximum
|
|
||||||
}
|
|
||||||
|
|
||||||
nlink = tnlink;
|
|
||||||
}
|
|
||||||
|
|
||||||
// build root bvh
|
|
||||||
bvh_calc_DOP_hull_from_faces(bvh, face_list, bvh->numfaces, bvh->root->bv);
|
|
||||||
|
|
||||||
// This is the traversal function.
|
|
||||||
bvh_div_env_node(bvh, bvh->root, face_list, 0, bvh->numfaces-1, 0, nlink);
|
|
||||||
if (face_list)
|
|
||||||
MEM_freeN(face_list);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// bvh_overlap - is it possbile for 2 bv's to collide ?
|
|
||||||
DO_INLINE int bvh_overlap(float *bv1, float *bv2)
|
|
||||||
{
|
|
||||||
int i = 0;
|
|
||||||
for (i = KDOP_START; i < KDOP_END; i++)
|
|
||||||
{
|
|
||||||
// Minimum test.
|
|
||||||
if (bv1[(2 * i)] > bv2[(2 * i) + 1])
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
// Maxiumum test.
|
|
||||||
if (bv2[(2 * i)] > bv1[(2 * i) + 1])
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// bvh_overlap_self - is it possbile for 2 bv's to selfcollide ?
|
|
||||||
DO_INLINE int bvh_overlap_self(CollisionTree * tree1, CollisionTree * tree2)
|
|
||||||
{
|
|
||||||
// printf("overlap: %f, q: %f\n", (saacos(INPR(tree1->normal, tree2->normal)) / 2.0)+MAX2(tree1->alpha, tree2->alpha), saacos(INPR(tree1->normal, tree2->normal)));
|
|
||||||
|
|
||||||
if((saacos(INPR(tree1->normal, tree2->normal)) / 2.0)+MAX2(tree1->alpha, tree2->alpha) > M_PI)
|
|
||||||
{
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* bvh_traverse - traverse two bvh trees looking for potential collisions.
|
|
||||||
*
|
|
||||||
* max collisions are n*n collisions --> every triangle collide with
|
|
||||||
* every other triangle that doesn't require any realloc, but uses
|
|
||||||
* much memory
|
|
||||||
*/
|
|
||||||
int bvh_traverse ( ModifierData * md1, ModifierData * md2, CollisionTree * tree1, CollisionTree * tree2, float step, CM_COLLISION_RESPONSE collision_response, int selfcollision)
|
|
||||||
{
|
|
||||||
int i = 0, ret=0, overlap = 0;
|
|
||||||
|
|
||||||
/*
|
|
||||||
// Shouldn't be possible
|
|
||||||
if(!tree1 || !tree2)
|
|
||||||
{
|
|
||||||
printf("Error: no tree there\n");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
if(selfcollision)
|
|
||||||
overlap = bvh_overlap_self(tree1, tree2);
|
|
||||||
else
|
|
||||||
overlap = bvh_overlap(tree1->bv, tree2->bv);
|
|
||||||
|
|
||||||
if (overlap)
|
|
||||||
{
|
|
||||||
// Check if this node in the first tree is a leaf
|
|
||||||
if (tree1->isleaf)
|
|
||||||
{
|
|
||||||
// Check if this node in the second tree a leaf
|
|
||||||
if (tree2->isleaf)
|
|
||||||
{
|
|
||||||
// Provide the collision response.
|
|
||||||
|
|
||||||
if(collision_response)
|
|
||||||
collision_response (md1, md2, tree1, tree2);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Process the quad tree.
|
|
||||||
for (i = 0; i < 4; i++)
|
|
||||||
{
|
|
||||||
// Only traverse nodes that exist.
|
|
||||||
if (tree2->nodes[i] && bvh_traverse (md1, md2, tree1, tree2->nodes[i], step, collision_response, selfcollision))
|
|
||||||
ret = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Process the quad tree.
|
|
||||||
for (i = 0; i < 4; i++)
|
|
||||||
{
|
|
||||||
// Only traverse nodes that exist.
|
|
||||||
if (tree1->nodes [i] && bvh_traverse (md1, md2, tree1->nodes[i], tree2, step, collision_response, selfcollision))
|
|
||||||
ret = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
// bottom up update of bvh tree:
|
|
||||||
// join the 4 children here
|
|
||||||
void bvh_join(CollisionTree *tree)
|
|
||||||
{
|
|
||||||
int i = 0, j = 0;
|
|
||||||
float max = 0;
|
|
||||||
|
|
||||||
if (!tree)
|
|
||||||
return;
|
|
||||||
|
|
||||||
for (i = 0; i < 4; i++)
|
|
||||||
{
|
|
||||||
if (tree->nodes[i])
|
|
||||||
{
|
|
||||||
for (j = KDOP_START; j < KDOP_END; j++)
|
|
||||||
{
|
|
||||||
// update minimum
|
|
||||||
if ((tree->nodes[i]->bv[(2 * j)] < tree->bv[(2 * j)]) || (i == 0))
|
|
||||||
{
|
|
||||||
tree->bv[(2 * j)] = tree->nodes[i]->bv[(2 * j)];
|
|
||||||
}
|
|
||||||
// update maximum
|
|
||||||
if ((tree->nodes[i]->bv[(2 * j) + 1] > tree->bv[(2 * j) + 1])|| (i == 0))
|
|
||||||
{
|
|
||||||
tree->bv[(2 * j) + 1] = tree->nodes[i]->bv[(2 * j) + 1];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* for selfcollisions */
|
|
||||||
/*
|
|
||||||
if(!i)
|
|
||||||
{
|
|
||||||
tree->alpha = tree->nodes[i]->alpha;
|
|
||||||
VECCOPY(tree->normal, tree->nodes[i]->normal);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
tree->alpha += saacos(INPR(tree->normal, tree->nodes[i]->normal)) / 2.0;
|
|
||||||
VECADD(tree->normal, tree->normal, tree->nodes[i]->normal);
|
|
||||||
VecMulf(tree->normal, 0.5);
|
|
||||||
max = MAX2(max, tree->nodes[i]->alpha);
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
}
|
|
||||||
else
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
tree->alpha += max;
|
|
||||||
}
|
|
||||||
|
|
||||||
// update static bvh
|
|
||||||
/* you have to update the bvh position before calling this function */
|
|
||||||
void bvh_update(BVH * bvh, int moving)
|
|
||||||
{
|
|
||||||
CollisionTree *leaf, *parent;
|
|
||||||
int traversecheck = 1; // if this is zero we don't go further
|
|
||||||
unsigned int j = 0;
|
|
||||||
|
|
||||||
for (leaf = bvh->leaf_root; leaf; leaf = leaf->nextLeaf)
|
|
||||||
{
|
|
||||||
traversecheck = 1;
|
|
||||||
if ((leaf->parent) && (leaf->parent->traversed == leaf->parent->count_nodes))
|
|
||||||
{
|
|
||||||
leaf->parent->traversed = 0;
|
|
||||||
}
|
|
||||||
if(!moving)
|
|
||||||
bvh_calc_DOP_hull_static(bvh, &leaf, 1, leaf->bv, leaf);
|
|
||||||
else
|
|
||||||
bvh_calc_DOP_hull_moving(bvh, &leaf, 1, leaf->bv, leaf);
|
|
||||||
|
|
||||||
// inflate the bv with some epsilon
|
|
||||||
for (j = KDOP_START; j < KDOP_END; j++)
|
|
||||||
{
|
|
||||||
leaf->bv[(2 * j)] -= bvh->epsilon; // minimum
|
|
||||||
leaf->bv[(2 * j) + 1] += bvh->epsilon; // maximum
|
|
||||||
}
|
|
||||||
|
|
||||||
for (parent = leaf->parent; parent; parent = parent->parent)
|
|
||||||
{
|
|
||||||
if (traversecheck)
|
|
||||||
{
|
|
||||||
parent->traversed++; // we tried to go up in hierarchy
|
|
||||||
if (parent->traversed < parent->count_nodes)
|
|
||||||
{
|
|
||||||
traversecheck = 0;
|
|
||||||
|
|
||||||
if (parent->parent)
|
|
||||||
{
|
|
||||||
if (parent->parent->traversed == parent->parent->count_nodes)
|
|
||||||
{
|
|
||||||
parent->parent->traversed = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break; // we do not need to check further
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
bvh_join(parent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -43,6 +43,7 @@
|
|||||||
|
|
||||||
#include "BLI_arithb.h"
|
#include "BLI_arithb.h"
|
||||||
#include "BLI_blenlib.h"
|
#include "BLI_blenlib.h"
|
||||||
|
#include "BLI_kdopbvh.h"
|
||||||
#include "BLI_kdtree.h"
|
#include "BLI_kdtree.h"
|
||||||
#include "BLI_linklist.h"
|
#include "BLI_linklist.h"
|
||||||
#include "BLI_rand.h"
|
#include "BLI_rand.h"
|
||||||
@@ -76,6 +77,7 @@
|
|||||||
#include "BKE_anim.h"
|
#include "BKE_anim.h"
|
||||||
#include "BKE_bad_level_calls.h"
|
#include "BKE_bad_level_calls.h"
|
||||||
#include "BKE_cloth.h"
|
#include "BKE_cloth.h"
|
||||||
|
#include "BKE_collision.h"
|
||||||
#include "BKE_curve.h"
|
#include "BKE_curve.h"
|
||||||
#include "BKE_customdata.h"
|
#include "BKE_customdata.h"
|
||||||
#include "BKE_global.h"
|
#include "BKE_global.h"
|
||||||
@@ -2736,6 +2738,7 @@ static DerivedMesh *bevelModifier_applyModifier(
|
|||||||
{
|
{
|
||||||
DerivedMesh *result;
|
DerivedMesh *result;
|
||||||
BME_Mesh *bm;
|
BME_Mesh *bm;
|
||||||
|
|
||||||
/*bDeformGroup *def;*/
|
/*bDeformGroup *def;*/
|
||||||
int /*i,*/ options, defgrp_index = -1;
|
int /*i,*/ options, defgrp_index = -1;
|
||||||
BevelModifierData *bmd = (BevelModifierData*) md;
|
BevelModifierData *bmd = (BevelModifierData*) md;
|
||||||
@@ -2754,8 +2757,7 @@ static DerivedMesh *bevelModifier_applyModifier(
|
|||||||
//~ }
|
//~ }
|
||||||
//~ }
|
//~ }
|
||||||
|
|
||||||
bm = BME_make_mesh();
|
bm = BME_derivedmesh_to_bmesh(derivedData);
|
||||||
bm = BME_derivedmesh_to_bmesh(derivedData, bm);
|
|
||||||
BME_bevel(bm,bmd->value,bmd->res,options,defgrp_index,bmd->bevel_angle,NULL);
|
BME_bevel(bm,bmd->value,bmd->res,options,defgrp_index,bmd->bevel_angle,NULL);
|
||||||
result = BME_bmesh_to_derivedmesh(bm,derivedData);
|
result = BME_bmesh_to_derivedmesh(bm,derivedData);
|
||||||
BME_free_mesh(bm);
|
BME_free_mesh(bm);
|
||||||
@@ -5193,7 +5195,7 @@ static void collisionModifier_initData(ModifierData *md)
|
|||||||
collmd->current_v = NULL;
|
collmd->current_v = NULL;
|
||||||
collmd->time = -1;
|
collmd->time = -1;
|
||||||
collmd->numverts = 0;
|
collmd->numverts = 0;
|
||||||
collmd->bvh = NULL;
|
collmd->bvhtree = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void collisionModifier_freeData(ModifierData *md)
|
static void collisionModifier_freeData(ModifierData *md)
|
||||||
@@ -5202,8 +5204,8 @@ static void collisionModifier_freeData(ModifierData *md)
|
|||||||
|
|
||||||
if (collmd)
|
if (collmd)
|
||||||
{
|
{
|
||||||
if(collmd->bvh)
|
if(collmd->bvhtree)
|
||||||
bvh_free(collmd->bvh);
|
BLI_bvhtree_free(collmd->bvhtree);
|
||||||
if(collmd->x)
|
if(collmd->x)
|
||||||
MEM_freeN(collmd->x);
|
MEM_freeN(collmd->x);
|
||||||
if(collmd->xnew)
|
if(collmd->xnew)
|
||||||
@@ -5214,7 +5216,6 @@ static void collisionModifier_freeData(ModifierData *md)
|
|||||||
MEM_freeN(collmd->current_xnew);
|
MEM_freeN(collmd->current_xnew);
|
||||||
if(collmd->current_v)
|
if(collmd->current_v)
|
||||||
MEM_freeN(collmd->current_v);
|
MEM_freeN(collmd->current_v);
|
||||||
|
|
||||||
if(collmd->mfaces)
|
if(collmd->mfaces)
|
||||||
MEM_freeN(collmd->mfaces);
|
MEM_freeN(collmd->mfaces);
|
||||||
|
|
||||||
@@ -5225,7 +5226,7 @@ static void collisionModifier_freeData(ModifierData *md)
|
|||||||
collmd->current_v = NULL;
|
collmd->current_v = NULL;
|
||||||
collmd->time = -1;
|
collmd->time = -1;
|
||||||
collmd->numverts = 0;
|
collmd->numverts = 0;
|
||||||
collmd->bvh = NULL;
|
collmd->bvhtree = NULL;
|
||||||
collmd->mfaces = NULL;
|
collmd->mfaces = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -5293,9 +5294,8 @@ static void collisionModifier_deformVerts(
|
|||||||
collmd->mfaces = dm->dupFaceArray(dm);
|
collmd->mfaces = dm->dupFaceArray(dm);
|
||||||
collmd->numfaces = dm->getNumFaces(dm);
|
collmd->numfaces = dm->getNumFaces(dm);
|
||||||
|
|
||||||
// TODO: epsilon
|
|
||||||
// create bounding box hierarchy
|
// create bounding box hierarchy
|
||||||
collmd->bvh = bvh_build_from_mvert(collmd->mfaces, collmd->numfaces, collmd->x, numverts, ob->pd->pdef_sboft);
|
collmd->bvhtree = bvhtree_build_from_mvert(collmd->mfaces, collmd->numfaces, collmd->x, numverts, ob->pd->pdef_sboft);
|
||||||
|
|
||||||
collmd->time = current_time;
|
collmd->time = current_time;
|
||||||
}
|
}
|
||||||
@@ -5318,25 +5318,25 @@ static void collisionModifier_deformVerts(
|
|||||||
memcpy(collmd->current_x, collmd->x, numverts*sizeof(MVert));
|
memcpy(collmd->current_x, collmd->x, numverts*sizeof(MVert));
|
||||||
|
|
||||||
/* check if GUI setting has changed for bvh */
|
/* check if GUI setting has changed for bvh */
|
||||||
if(collmd->bvh)
|
if(collmd->bvhtree)
|
||||||
{
|
{
|
||||||
if(ob->pd->pdef_sboft != collmd->bvh->epsilon)
|
if(ob->pd->pdef_sboft != BLI_bvhtree_getepsilon(collmd->bvhtree))
|
||||||
{
|
{
|
||||||
bvh_free(collmd->bvh);
|
BLI_bvhtree_free(collmd->bvhtree);
|
||||||
collmd->bvh = bvh_build_from_mvert(collmd->mfaces, collmd->numfaces, collmd->current_x, numverts, ob->pd->pdef_sboft);
|
collmd->bvhtree = bvhtree_build_from_mvert(collmd->mfaces, collmd->numfaces, collmd->current_x, numverts, ob->pd->pdef_sboft);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* happens on file load (ONLY when i decomment changes in readfile.c */
|
/* happens on file load (ONLY when i decomment changes in readfile.c) */
|
||||||
if(!collmd->bvh)
|
if(!collmd->bvhtree)
|
||||||
{
|
{
|
||||||
collmd->bvh = bvh_build_from_mvert(collmd->mfaces, collmd->numfaces, collmd->current_x, numverts, ob->pd->pdef_sboft);
|
collmd->bvhtree = bvhtree_build_from_mvert(collmd->mfaces, collmd->numfaces, collmd->current_x, numverts, ob->pd->pdef_sboft);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// recalc static bounding boxes
|
// recalc static bounding boxes
|
||||||
bvh_update_from_mvert(collmd->bvh, collmd->current_x, numverts, NULL, 0);
|
bvhtree_update_from_mvert ( collmd->bvhtree, collmd->mfaces, collmd->numfaces, collmd->current_x, collmd->current_xnew, collmd->numverts, 1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
collmd->time = current_time;
|
collmd->time = current_time;
|
||||||
@@ -6483,12 +6483,14 @@ static DerivedMesh * explodeModifier_explodeMesh(ExplodeModifierData *emd,
|
|||||||
MFace *mf=0;
|
MFace *mf=0;
|
||||||
MVert *dupvert=0;
|
MVert *dupvert=0;
|
||||||
ParticleSettings *part=psmd->psys->part;
|
ParticleSettings *part=psmd->psys->part;
|
||||||
ParticleData *pa, *pars=psmd->psys->particles;
|
ParticleData *pa=NULL, *pars=psmd->psys->particles;
|
||||||
ParticleKey state;
|
ParticleKey state;
|
||||||
|
EdgeHash *vertpahash;
|
||||||
|
EdgeHashIterator *ehi;
|
||||||
float *vertco=0, imat[4][4];
|
float *vertco=0, imat[4][4];
|
||||||
float loc0[3], nor[3];
|
float loc0[3], nor[3];
|
||||||
float timestep, cfra;
|
float timestep, cfra;
|
||||||
int *facepa=emd->facepa, *vertpa=0;
|
int *facepa=emd->facepa;
|
||||||
int totdup=0,totvert=0,totface=0,totpart=0;
|
int totdup=0,totvert=0,totface=0,totpart=0;
|
||||||
int i, j, v, mindex=0;
|
int i, j, v, mindex=0;
|
||||||
|
|
||||||
@@ -6503,32 +6505,34 @@ static DerivedMesh * explodeModifier_explodeMesh(ExplodeModifierData *emd,
|
|||||||
else
|
else
|
||||||
cfra=bsystem_time(ob,(float)G.scene->r.cfra,0.0);
|
cfra=bsystem_time(ob,(float)G.scene->r.cfra,0.0);
|
||||||
|
|
||||||
/* table for vertice <-> particle relations (row totpart+1 is for yet unexploded verts) */
|
/* hash table for vertice <-> particle relations */
|
||||||
vertpa = MEM_callocN(sizeof(int)*(totpart+1)*totvert, "explode_vertpatab");
|
vertpahash= BLI_edgehash_new();
|
||||||
for(i=0; i<(totpart+1)*totvert; i++)
|
|
||||||
vertpa[i] = -1;
|
|
||||||
|
|
||||||
for (i=0; i<totface; i++) {
|
for (i=0; i<totface; i++) {
|
||||||
|
/* do mindex + totvert to ensure the vertex index to be the first
|
||||||
|
* with BLI_edgehashIterator_getKey */
|
||||||
if(facepa[i]==totpart || cfra <= (pars+facepa[i])->time)
|
if(facepa[i]==totpart || cfra <= (pars+facepa[i])->time)
|
||||||
mindex = totpart*totvert;
|
mindex = totvert+totpart;
|
||||||
else
|
else
|
||||||
mindex = facepa[i]*totvert;
|
mindex = totvert+facepa[i];
|
||||||
|
|
||||||
mf=CDDM_get_face(dm,i);
|
mf=CDDM_get_face(dm,i);
|
||||||
|
|
||||||
/* set face vertices to exist in particle group */
|
/* set face vertices to exist in particle group */
|
||||||
vertpa[mindex+mf->v1] = 1;
|
BLI_edgehash_insert(vertpahash, mf->v1, mindex, NULL);
|
||||||
vertpa[mindex+mf->v2] = 1;
|
BLI_edgehash_insert(vertpahash, mf->v2, mindex, NULL);
|
||||||
vertpa[mindex+mf->v3] = 1;
|
BLI_edgehash_insert(vertpahash, mf->v3, mindex, NULL);
|
||||||
if(mf->v4)
|
if(mf->v4)
|
||||||
vertpa[mindex+mf->v4] = 1;
|
BLI_edgehash_insert(vertpahash, mf->v4, mindex, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* make new vertice indexes & count total vertices after duplication */
|
/* make new vertice indexes & count total vertices after duplication */
|
||||||
for(i=0; i<(totpart+1)*totvert; i++){
|
ehi= BLI_edgehashIterator_new(vertpahash);
|
||||||
if(vertpa[i] != -1)
|
for(; !BLI_edgehashIterator_isDone(ehi); BLI_edgehashIterator_step(ehi)) {
|
||||||
vertpa[i] = totdup++;
|
BLI_edgehashIterator_setValue(ehi, SET_INT_IN_POINTER(totdup));
|
||||||
|
totdup++;
|
||||||
}
|
}
|
||||||
|
BLI_edgehashIterator_free(ehi);
|
||||||
|
|
||||||
/* the final duplicated vertices */
|
/* the final duplicated vertices */
|
||||||
explode= CDDM_from_template(dm, totdup, 0,totface);
|
explode= CDDM_from_template(dm, totdup, 0,totface);
|
||||||
@@ -6540,21 +6544,16 @@ static DerivedMesh * explodeModifier_explodeMesh(ExplodeModifierData *emd,
|
|||||||
psmd->psys->lattice = psys_get_lattice(ob, psmd->psys);
|
psmd->psys->lattice = psys_get_lattice(ob, psmd->psys);
|
||||||
|
|
||||||
/* duplicate & displace vertices */
|
/* duplicate & displace vertices */
|
||||||
for(i=0, pa=pars; i<=totpart; i++, pa++){
|
ehi= BLI_edgehashIterator_new(vertpahash);
|
||||||
if(i!=totpart){
|
for(; !BLI_edgehashIterator_isDone(ehi); BLI_edgehashIterator_step(ehi)) {
|
||||||
psys_particle_on_emitter(ob, psmd,part->from,pa->num,-1,pa->fuv,pa->foffset,loc0,nor,0,0,0,0);
|
|
||||||
Mat4MulVecfl(ob->obmat,loc0);
|
|
||||||
|
|
||||||
state.time=cfra;
|
|
||||||
psys_get_particle_state(ob,psmd->psys,i,&state,1);
|
|
||||||
}
|
|
||||||
|
|
||||||
for(j=0; j<totvert; j++){
|
|
||||||
v=vertpa[i*totvert+j];
|
|
||||||
if(v != -1) {
|
|
||||||
MVert source;
|
MVert source;
|
||||||
MVert *dest;
|
MVert *dest;
|
||||||
|
|
||||||
|
/* get particle + vertex from hash */
|
||||||
|
BLI_edgehashIterator_getKey(ehi, &j, &i);
|
||||||
|
i -= totvert;
|
||||||
|
v= GET_INT_FROM_POINTER(BLI_edgehashIterator_getValue(ehi));
|
||||||
|
|
||||||
dm->getVert(dm, j, &source);
|
dm->getVert(dm, j, &source);
|
||||||
dest = CDDM_get_vert(explode,v);
|
dest = CDDM_get_vert(explode,v);
|
||||||
|
|
||||||
@@ -6562,6 +6561,16 @@ static DerivedMesh * explodeModifier_explodeMesh(ExplodeModifierData *emd,
|
|||||||
*dest = source;
|
*dest = source;
|
||||||
|
|
||||||
if(i!=totpart) {
|
if(i!=totpart) {
|
||||||
|
/* get particle */
|
||||||
|
pa= pars+i;
|
||||||
|
|
||||||
|
/* get particle state */
|
||||||
|
psys_particle_on_emitter(ob, psmd,part->from,pa->num,-1,pa->fuv,pa->foffset,loc0,nor,0,0,0,0);
|
||||||
|
Mat4MulVecfl(ob->obmat,loc0);
|
||||||
|
|
||||||
|
state.time=cfra;
|
||||||
|
psys_get_particle_state(ob,psmd->psys,i,&state,1);
|
||||||
|
|
||||||
vertco=CDDM_get_vert(explode,v)->co;
|
vertco=CDDM_get_vert(explode,v)->co;
|
||||||
|
|
||||||
Mat4MulVecfl(ob->obmat,vertco);
|
Mat4MulVecfl(ob->obmat,vertco);
|
||||||
@@ -6576,8 +6585,7 @@ static DerivedMesh * explodeModifier_explodeMesh(ExplodeModifierData *emd,
|
|||||||
Mat4MulVecfl(imat,vertco);
|
Mat4MulVecfl(imat,vertco);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
BLI_edgehashIterator_free(ehi);
|
||||||
}
|
|
||||||
|
|
||||||
/*map new vertices to faces*/
|
/*map new vertices to faces*/
|
||||||
for (i=0; i<totface; i++) {
|
for (i=0; i<totface; i++) {
|
||||||
@@ -6599,15 +6607,15 @@ static DerivedMesh * explodeModifier_explodeMesh(ExplodeModifierData *emd,
|
|||||||
orig_v4 = source.v4;
|
orig_v4 = source.v4;
|
||||||
|
|
||||||
if(facepa[i]!=totpart && cfra <= pa->time)
|
if(facepa[i]!=totpart && cfra <= pa->time)
|
||||||
mindex = totpart*totvert;
|
mindex = totvert+totpart;
|
||||||
else
|
else
|
||||||
mindex = facepa[i]*totvert;
|
mindex = totvert+facepa[i];
|
||||||
|
|
||||||
source.v1 = vertpa[mindex+source.v1];
|
source.v1 = edgesplit_get(vertpahash, source.v1, mindex);
|
||||||
source.v2 = vertpa[mindex+source.v2];
|
source.v2 = edgesplit_get(vertpahash, source.v2, mindex);
|
||||||
source.v3 = vertpa[mindex+source.v3];
|
source.v3 = edgesplit_get(vertpahash, source.v3, mindex);
|
||||||
if(source.v4)
|
if(source.v4)
|
||||||
source.v4 = vertpa[mindex+source.v4];
|
source.v4 = edgesplit_get(vertpahash, source.v4, mindex);
|
||||||
|
|
||||||
DM_copy_face_data(dm,explode,i,i,1);
|
DM_copy_face_data(dm,explode,i,i,1);
|
||||||
|
|
||||||
@@ -6616,9 +6624,10 @@ static DerivedMesh * explodeModifier_explodeMesh(ExplodeModifierData *emd,
|
|||||||
test_index_face(mf, &explode->faceData, i, (mf->v4 ? 4 : 3));
|
test_index_face(mf, &explode->faceData, i, (mf->v4 ? 4 : 3));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MEM_printmemlist_stats();
|
||||||
|
|
||||||
/* cleanup */
|
/* cleanup */
|
||||||
if(vertpa) MEM_freeN(vertpa);
|
BLI_edgehash_free(vertpahash, NULL);
|
||||||
|
|
||||||
/* finalization */
|
/* finalization */
|
||||||
CDDM_calc_edges(explode);
|
CDDM_calc_edges(explode);
|
||||||
|
|||||||
@@ -1612,7 +1612,7 @@ static void give_parvert(Object *par, int nr, float *vec)
|
|||||||
|
|
||||||
for(eve= em->verts.first; eve; eve= eve->next) {
|
for(eve= em->verts.first; eve; eve= eve->next) {
|
||||||
if(eve->keyindex==nr) {
|
if(eve->keyindex==nr) {
|
||||||
memcpy(vec, eve->co, 12);
|
memcpy(vec, eve->co, sizeof(float)*3);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1650,18 +1650,20 @@ static void give_parvert(Object *par, int nr, float *vec)
|
|||||||
Curve *cu;
|
Curve *cu;
|
||||||
BPoint *bp;
|
BPoint *bp;
|
||||||
BezTriple *bezt;
|
BezTriple *bezt;
|
||||||
|
int found= 0;
|
||||||
|
|
||||||
cu= par->data;
|
cu= par->data;
|
||||||
nu= cu->nurb.first;
|
nu= cu->nurb.first;
|
||||||
if(par==G.obedit) nu= editNurb.first;
|
if(par==G.obedit) nu= editNurb.first;
|
||||||
|
|
||||||
count= 0;
|
count= 0;
|
||||||
while(nu) {
|
while(nu && !found) {
|
||||||
if((nu->type & 7)==CU_BEZIER) {
|
if((nu->type & 7)==CU_BEZIER) {
|
||||||
bezt= nu->bezt;
|
bezt= nu->bezt;
|
||||||
a= nu->pntsu;
|
a= nu->pntsu;
|
||||||
while(a--) {
|
while(a--) {
|
||||||
if(count==nr) {
|
if(count==nr) {
|
||||||
|
found= 1;
|
||||||
VECCOPY(vec, bezt->vec[1]);
|
VECCOPY(vec, bezt->vec[1]);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -1674,7 +1676,8 @@ static void give_parvert(Object *par, int nr, float *vec)
|
|||||||
a= nu->pntsu*nu->pntsv;
|
a= nu->pntsu*nu->pntsv;
|
||||||
while(a--) {
|
while(a--) {
|
||||||
if(count==nr) {
|
if(count==nr) {
|
||||||
memcpy(vec, bp->vec, 12);
|
found= 1;
|
||||||
|
memcpy(vec, bp->vec, sizeof(float)*3);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
count++;
|
count++;
|
||||||
|
|||||||
@@ -185,8 +185,8 @@ static int ptcache_path(PTCacheID *pid, char *filename)
|
|||||||
file[i-6] = '\0';
|
file[i-6] = '\0';
|
||||||
|
|
||||||
sprintf(filename, "//"PTCACHE_PATH"%s", file); /* add blend file name to pointcache dir */
|
sprintf(filename, "//"PTCACHE_PATH"%s", file); /* add blend file name to pointcache dir */
|
||||||
BLI_add_slash(filename);
|
|
||||||
BLI_convertstringcode(filename, blendfilename);
|
BLI_convertstringcode(filename, blendfilename);
|
||||||
|
BLI_add_slash(filename);
|
||||||
return strlen(filename);
|
return strlen(filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2075,14 +2075,14 @@ void txt_backspace_char (Text *text)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
else if (text->curc==0) { /* Appending two lines */
|
else if (text->curc==0) { /* Appending two lines */
|
||||||
if (text->curl->prev) {
|
if (!text->curl->prev) return;
|
||||||
|
|
||||||
text->curl= text->curl->prev;
|
text->curl= text->curl->prev;
|
||||||
text->curc= text->curl->len;
|
text->curc= text->curl->len;
|
||||||
|
|
||||||
txt_combine_lines(text, text->curl, text->curl->next);
|
txt_combine_lines(text, text->curl, text->curl->next);
|
||||||
txt_pop_sel(text);
|
txt_pop_sel(text);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else { /* Just backspacing a char */
|
else { /* Just backspacing a char */
|
||||||
int i= text->curc-1;
|
int i= text->curc-1;
|
||||||
|
|
||||||
|
|||||||
@@ -867,7 +867,7 @@ void end_ffmpeg(void)
|
|||||||
|
|
||||||
fprintf(stderr, "Closing ffmpeg...\n");
|
fprintf(stderr, "Closing ffmpeg...\n");
|
||||||
|
|
||||||
if (audio_stream) {
|
if (audio_stream && video_stream) {
|
||||||
write_audio_frames();
|
write_audio_frames();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
60
source/blender/blenlib/BLI_kdopbvh.h
Normal file
60
source/blender/blenlib/BLI_kdopbvh.h
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||||||
|
*
|
||||||
|
* The Original Code is Copyright (C) 2006 by NaN Holding BV.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* The Original Code is: all of this file.
|
||||||
|
*
|
||||||
|
* Contributor(s): Daniel Genrich, Andre Pinto
|
||||||
|
*
|
||||||
|
* ***** END GPL LICENSE BLOCK *****
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef BLI_KDOPBVH_H
|
||||||
|
#define BLI_KDOPBVH_H
|
||||||
|
|
||||||
|
#include <float.h>
|
||||||
|
|
||||||
|
struct BVHTree;
|
||||||
|
typedef struct BVHTree BVHTree;
|
||||||
|
|
||||||
|
typedef struct BVHTreeOverlap {
|
||||||
|
int indexA;
|
||||||
|
int indexB;
|
||||||
|
} BVHTreeOverlap;
|
||||||
|
|
||||||
|
BVHTree *BLI_bvhtree_new(int maxsize, float epsilon, char tree_type, char axis);
|
||||||
|
void BLI_bvhtree_free(BVHTree *tree);
|
||||||
|
|
||||||
|
/* construct: first insert points, then call balance */
|
||||||
|
int BLI_bvhtree_insert(BVHTree *tree, int index, float *co, int numpoints);
|
||||||
|
void BLI_bvhtree_balance(BVHTree *tree);
|
||||||
|
|
||||||
|
/* update: first update points/nodes, then call update_tree to refit the bounding volumes */
|
||||||
|
int BLI_bvhtree_update_node(BVHTree *tree, int index, float *co, float *co_moving, int numpoints);
|
||||||
|
void BLI_bvhtree_update_tree(BVHTree *tree);
|
||||||
|
|
||||||
|
/* collision/overlap: check two trees if they overlap, alloc's *overlap with length of the int return value */
|
||||||
|
BVHTreeOverlap *BLI_bvhtree_overlap(BVHTree *tree1, BVHTree *tree2, int *result);
|
||||||
|
|
||||||
|
float BLI_bvhtree_getepsilon(BVHTree *tree);
|
||||||
|
|
||||||
|
#endif // BLI_KDOPBVH_H
|
||||||
|
|
||||||
810
source/blender/blenlib/intern/BLI_kdopbvh.c
Normal file
810
source/blender/blenlib/intern/BLI_kdopbvh.c
Normal file
@@ -0,0 +1,810 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||||||
|
*
|
||||||
|
* The Original Code is Copyright (C) 2006 by NaN Holding BV.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* The Original Code is: all of this file.
|
||||||
|
*
|
||||||
|
* Contributor(s): Daniel Genrich, Andre Pinto
|
||||||
|
*
|
||||||
|
* ***** END GPL LICENSE BLOCK *****
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "math.h"
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "MEM_guardedalloc.h"
|
||||||
|
|
||||||
|
#include "BKE_utildefines.h"
|
||||||
|
|
||||||
|
#include "BLI_kdopbvh.h"
|
||||||
|
#include "BLI_arithb.h"
|
||||||
|
|
||||||
|
#ifdef _OPENMP
|
||||||
|
#include <omp.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef struct BVHNode
|
||||||
|
{
|
||||||
|
struct BVHNode **children; // max 8 children
|
||||||
|
struct BVHNode *parent; // needed for bottom - top update
|
||||||
|
float *bv; // Bounding volume of all nodes, max 13 axis
|
||||||
|
int index; /* face, edge, vertex index */
|
||||||
|
char totnode; // how many nodes are used, used for speedup
|
||||||
|
char traversed; // how many nodes already traversed until this level?
|
||||||
|
char main_axis;
|
||||||
|
} BVHNode;
|
||||||
|
|
||||||
|
struct BVHTree
|
||||||
|
{
|
||||||
|
BVHNode **nodes;
|
||||||
|
BVHNode *nodearray; /* pre-alloc branch nodes */
|
||||||
|
BVHNode **nodechild; // pre-alloc childs for nodes
|
||||||
|
float *nodebv; // pre-alloc bounding-volumes for nodes
|
||||||
|
float epsilon; /* epslion is used for inflation of the k-dop */
|
||||||
|
int totleaf; // leafs
|
||||||
|
int totbranch;
|
||||||
|
char tree_type; // type of tree (4 => quadtree)
|
||||||
|
char axis; // kdop type (6 => OBB, 7 => AABB, ...)
|
||||||
|
char start_axis, stop_axis; // KDOP_AXES array indices according to axis
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct BVHOverlapData
|
||||||
|
{
|
||||||
|
BVHTree *tree1, *tree2;
|
||||||
|
BVHTreeOverlap *overlap;
|
||||||
|
int i, max_overlap; /* i is number of overlaps */
|
||||||
|
} BVHOverlapData;
|
||||||
|
////////////////////////////////////////
|
||||||
|
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////
|
||||||
|
// Bounding Volume Hierarchy Definition
|
||||||
|
//
|
||||||
|
// Notes: From OBB until 26-DOP --> all bounding volumes possible, just choose type below
|
||||||
|
// Notes: You have to choose the type at compile time ITM
|
||||||
|
// Notes: You can choose the tree type --> binary, quad, octree, choose below
|
||||||
|
////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
static float KDOP_AXES[13][3] =
|
||||||
|
{ {1.0, 0, 0}, {0, 1.0, 0}, {0, 0, 1.0}, {1.0, 1.0, 1.0}, {1.0, -1.0, 1.0}, {1.0, 1.0, -1.0},
|
||||||
|
{1.0, -1.0, -1.0}, {1.0, 1.0, 0}, {1.0, 0, 1.0}, {0, 1.0, 1.0}, {1.0, -1.0, 0}, {1.0, 0, -1.0},
|
||||||
|
{0, 1.0, -1.0}
|
||||||
|
};
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Introsort
|
||||||
|
// with permission deriven from the following Java code:
|
||||||
|
// http://ralphunden.net/content/tutorials/a-guide-to-introsort/
|
||||||
|
// and he derived it from the SUN STL
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
static int size_threshold = 16;
|
||||||
|
/*
|
||||||
|
* Common methods for all algorithms
|
||||||
|
*/
|
||||||
|
static int floor_lg(int a)
|
||||||
|
{
|
||||||
|
return (int)(floor(log(a)/log(2)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Insertion sort algorithm
|
||||||
|
*/
|
||||||
|
static void bvh_insertionsort(BVHNode **a, int lo, int hi, int axis)
|
||||||
|
{
|
||||||
|
int i,j;
|
||||||
|
BVHNode *t;
|
||||||
|
for (i=lo; i < hi; i++)
|
||||||
|
{
|
||||||
|
j=i;
|
||||||
|
t = a[i];
|
||||||
|
while((j!=lo) && (t->bv[axis] < (a[j-1])->bv[axis]))
|
||||||
|
{
|
||||||
|
a[j] = a[j-1];
|
||||||
|
j--;
|
||||||
|
}
|
||||||
|
a[j] = t;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int bvh_partition(BVHNode **a, int lo, int hi, BVHNode * x, int axis)
|
||||||
|
{
|
||||||
|
int i=lo, j=hi;
|
||||||
|
while (1)
|
||||||
|
{
|
||||||
|
while ((a[i])->bv[axis] < x->bv[axis]) i++;
|
||||||
|
j--;
|
||||||
|
while (x->bv[axis] < (a[j])->bv[axis]) j--;
|
||||||
|
if(!(i < j))
|
||||||
|
return i;
|
||||||
|
SWAP( BVHNode* , a[i], a[j]);
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Heapsort algorithm
|
||||||
|
*/
|
||||||
|
static void bvh_downheap(BVHNode **a, int i, int n, int lo, int axis)
|
||||||
|
{
|
||||||
|
BVHNode * d = a[lo+i-1];
|
||||||
|
int child;
|
||||||
|
while (i<=n/2)
|
||||||
|
{
|
||||||
|
child = 2*i;
|
||||||
|
if ((child < n) && ((a[lo+child-1])->bv[axis] < (a[lo+child])->bv[axis]))
|
||||||
|
{
|
||||||
|
child++;
|
||||||
|
}
|
||||||
|
if (!(d->bv[axis] < (a[lo+child-1])->bv[axis])) break;
|
||||||
|
a[lo+i-1] = a[lo+child-1];
|
||||||
|
i = child;
|
||||||
|
}
|
||||||
|
a[lo+i-1] = d;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void bvh_heapsort(BVHNode **a, int lo, int hi, int axis)
|
||||||
|
{
|
||||||
|
int n = hi-lo, i;
|
||||||
|
for (i=n/2; i>=1; i=i-1)
|
||||||
|
{
|
||||||
|
bvh_downheap(a, i,n,lo, axis);
|
||||||
|
}
|
||||||
|
for (i=n; i>1; i=i-1)
|
||||||
|
{
|
||||||
|
SWAP(BVHNode*, a[lo],a[lo+i-1]);
|
||||||
|
bvh_downheap(a, 1,i-1,lo, axis);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static BVHNode *bvh_medianof3(BVHNode **a, int lo, int mid, int hi, int axis) // returns Sortable
|
||||||
|
{
|
||||||
|
if ((a[mid])->bv[axis] < (a[lo])->bv[axis])
|
||||||
|
{
|
||||||
|
if ((a[hi])->bv[axis] < (a[mid])->bv[axis])
|
||||||
|
return a[mid];
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if ((a[hi])->bv[axis] < (a[lo])->bv[axis])
|
||||||
|
return a[hi];
|
||||||
|
else
|
||||||
|
return a[lo];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if ((a[hi])->bv[axis] < (a[mid])->bv[axis])
|
||||||
|
{
|
||||||
|
if ((a[hi])->bv[axis] < (a[lo])->bv[axis])
|
||||||
|
return a[lo];
|
||||||
|
else
|
||||||
|
return a[hi];
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return a[mid];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
* Quicksort algorithm modified for Introsort
|
||||||
|
*/
|
||||||
|
static void bvh_introsort_loop (BVHNode **a, int lo, int hi, int depth_limit, int axis)
|
||||||
|
{
|
||||||
|
int p;
|
||||||
|
|
||||||
|
while (hi-lo > size_threshold)
|
||||||
|
{
|
||||||
|
if (depth_limit == 0)
|
||||||
|
{
|
||||||
|
bvh_heapsort(a, lo, hi, axis);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
depth_limit=depth_limit-1;
|
||||||
|
p=bvh_partition(a, lo, hi, bvh_medianof3(a, lo, lo+((hi-lo)/2)+1, hi-1, axis), axis);
|
||||||
|
bvh_introsort_loop(a, p, hi, depth_limit, axis);
|
||||||
|
hi=p;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void sort(BVHNode **a0, int begin, int end, int axis)
|
||||||
|
{
|
||||||
|
if (begin < end)
|
||||||
|
{
|
||||||
|
BVHNode **a=a0;
|
||||||
|
bvh_introsort_loop(a, begin, end, 2*floor_lg(end-begin), axis);
|
||||||
|
bvh_insertionsort(a, begin, end, axis);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void sort_along_axis(BVHTree *tree, int start, int end, int axis)
|
||||||
|
{
|
||||||
|
sort(tree->nodes, start, end, axis);
|
||||||
|
}
|
||||||
|
|
||||||
|
//after a call to this function you can expect one of:
|
||||||
|
// every node to left of a[n] are smaller or equal to it
|
||||||
|
// every node to the right of a[n] are greater or equal to it
|
||||||
|
int partition_nth_element(BVHNode **a, int _begin, int _end, int n, int axis){
|
||||||
|
int begin = _begin, end = _end, cut;
|
||||||
|
while(end-begin > 3)
|
||||||
|
{
|
||||||
|
cut = bvh_partition(a, begin, end, bvh_medianof3(a, begin, (begin+end)/2, end-1, axis), axis );
|
||||||
|
if(cut <= n)
|
||||||
|
begin = cut;
|
||||||
|
else
|
||||||
|
end = cut;
|
||||||
|
}
|
||||||
|
bvh_insertionsort(a, begin, end, axis);
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void BLI_bvhtree_free(BVHTree *tree)
|
||||||
|
{
|
||||||
|
if(tree)
|
||||||
|
{
|
||||||
|
MEM_freeN(tree->nodes);
|
||||||
|
MEM_freeN(tree->nodearray);
|
||||||
|
MEM_freeN(tree->nodebv);
|
||||||
|
MEM_freeN(tree->nodechild);
|
||||||
|
MEM_freeN(tree);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
BVHTree *BLI_bvhtree_new(int maxsize, float epsilon, char tree_type, char axis)
|
||||||
|
{
|
||||||
|
BVHTree *tree;
|
||||||
|
int numbranches=0, i;
|
||||||
|
|
||||||
|
// only support up to octree
|
||||||
|
if(tree_type > 8)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
tree = (BVHTree *)MEM_callocN(sizeof(BVHTree), "BVHTree");
|
||||||
|
|
||||||
|
if(tree)
|
||||||
|
{
|
||||||
|
tree->epsilon = epsilon;
|
||||||
|
tree->tree_type = tree_type;
|
||||||
|
tree->axis = axis;
|
||||||
|
|
||||||
|
if(axis == 26)
|
||||||
|
{
|
||||||
|
tree->start_axis = 0;
|
||||||
|
tree->stop_axis = 13;
|
||||||
|
}
|
||||||
|
else if(axis == 18)
|
||||||
|
{
|
||||||
|
tree->start_axis = 7;
|
||||||
|
tree->stop_axis = 13;
|
||||||
|
}
|
||||||
|
else if(axis == 14)
|
||||||
|
{
|
||||||
|
tree->start_axis = 0;
|
||||||
|
tree->stop_axis = 7;
|
||||||
|
}
|
||||||
|
else if(axis == 8) // AABB
|
||||||
|
{
|
||||||
|
tree->start_axis = 0;
|
||||||
|
tree->stop_axis = 4;
|
||||||
|
}
|
||||||
|
else if(axis == 6) // OBB
|
||||||
|
{
|
||||||
|
tree->start_axis = 0;
|
||||||
|
tree->stop_axis = 3;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
MEM_freeN(tree);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// calculate max number of branches, our bvh kdop is "almost perfect"
|
||||||
|
for(i = 1; i <= (int)ceil((float)((float)log(maxsize)/(float)log(tree_type))); i++)
|
||||||
|
numbranches += (pow(tree_type, i) / tree_type);
|
||||||
|
|
||||||
|
tree->nodes = (BVHNode **)MEM_callocN(sizeof(BVHNode *)*(numbranches+maxsize + tree_type), "BVHNodes");
|
||||||
|
|
||||||
|
if(!tree->nodes)
|
||||||
|
{
|
||||||
|
MEM_freeN(tree);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
tree->nodebv = (float*)MEM_callocN(sizeof(float)* axis * (numbranches+maxsize + tree_type), "BVHNodeBV");
|
||||||
|
if(!tree->nodebv)
|
||||||
|
{
|
||||||
|
MEM_freeN(tree->nodes);
|
||||||
|
MEM_freeN(tree);
|
||||||
|
}
|
||||||
|
|
||||||
|
tree->nodechild = (BVHNode**)MEM_callocN(sizeof(BVHNode*) * tree_type * (numbranches+maxsize + tree_type), "BVHNodeBV");
|
||||||
|
if(!tree->nodechild)
|
||||||
|
{
|
||||||
|
MEM_freeN(tree->nodebv);
|
||||||
|
MEM_freeN(tree->nodes);
|
||||||
|
MEM_freeN(tree);
|
||||||
|
}
|
||||||
|
|
||||||
|
tree->nodearray = (BVHNode *)MEM_callocN(sizeof(BVHNode)*(numbranches+maxsize + tree_type), "BVHNodeArray");
|
||||||
|
|
||||||
|
if(!tree->nodearray)
|
||||||
|
{
|
||||||
|
MEM_freeN(tree->nodechild);
|
||||||
|
MEM_freeN(tree->nodebv);
|
||||||
|
MEM_freeN(tree->nodes);
|
||||||
|
MEM_freeN(tree);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
//link the dynamic bv and child links
|
||||||
|
for(i=0; i< numbranches+maxsize + tree_type; i++)
|
||||||
|
{
|
||||||
|
tree->nodearray[i].bv = tree->nodebv + i * axis;
|
||||||
|
tree->nodearray[i].children = tree->nodechild + i * tree_type;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return tree;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void create_kdop_hull(BVHTree *tree, BVHNode *node, float *co, int numpoints, int moving)
|
||||||
|
{
|
||||||
|
float newminmax;
|
||||||
|
int i, k;
|
||||||
|
|
||||||
|
// don't init boudings for the moving case
|
||||||
|
if(!moving)
|
||||||
|
{
|
||||||
|
for (i = tree->start_axis; i < tree->stop_axis; i++)
|
||||||
|
{
|
||||||
|
node->bv[2*i] = FLT_MAX;
|
||||||
|
node->bv[2*i + 1] = -FLT_MAX;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for(k = 0; k < numpoints; k++)
|
||||||
|
{
|
||||||
|
// for all Axes.
|
||||||
|
for (i = tree->start_axis; i < tree->stop_axis; i++)
|
||||||
|
{
|
||||||
|
newminmax = INPR(&co[k * 3], KDOP_AXES[i]);
|
||||||
|
if (newminmax < node->bv[2 * i])
|
||||||
|
node->bv[2 * i] = newminmax;
|
||||||
|
if (newminmax > node->bv[(2 * i) + 1])
|
||||||
|
node->bv[(2 * i) + 1] = newminmax;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// depends on the fact that the BVH's for each face is already build
|
||||||
|
static void refit_kdop_hull(BVHTree *tree, BVHNode *node, int start, int end)
|
||||||
|
{
|
||||||
|
float newmin,newmax;
|
||||||
|
int i, j;
|
||||||
|
float *bv = node->bv;
|
||||||
|
|
||||||
|
for (i = tree->start_axis; i < tree->stop_axis; i++)
|
||||||
|
{
|
||||||
|
bv[2*i] = FLT_MAX;
|
||||||
|
bv[2*i + 1] = -FLT_MAX;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (j = start; j < end; j++)
|
||||||
|
{
|
||||||
|
// for all Axes.
|
||||||
|
for (i = tree->start_axis; i < tree->stop_axis; i++)
|
||||||
|
{
|
||||||
|
newmin = tree->nodes[j]->bv[(2 * i)];
|
||||||
|
if ((newmin < bv[(2 * i)]))
|
||||||
|
bv[(2 * i)] = newmin;
|
||||||
|
|
||||||
|
newmax = tree->nodes[j]->bv[(2 * i) + 1];
|
||||||
|
if ((newmax > bv[(2 * i) + 1]))
|
||||||
|
bv[(2 * i) + 1] = newmax;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int BLI_bvhtree_insert(BVHTree *tree, int index, float *co, int numpoints)
|
||||||
|
{
|
||||||
|
BVHNode *node= NULL;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
// insert should only possible as long as tree->totbranch is 0
|
||||||
|
if(tree->totbranch > 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if(tree->totleaf+1 >= MEM_allocN_len(tree->nodes))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// TODO check if have enough nodes in array
|
||||||
|
|
||||||
|
node = tree->nodes[tree->totleaf] = &(tree->nodearray[tree->totleaf]);
|
||||||
|
tree->totleaf++;
|
||||||
|
|
||||||
|
create_kdop_hull(tree, node, co, numpoints, 0);
|
||||||
|
|
||||||
|
// inflate the bv with some epsilon
|
||||||
|
for (i = tree->start_axis; i < tree->stop_axis; i++)
|
||||||
|
{
|
||||||
|
node->bv[(2 * i)] -= tree->epsilon; // minimum
|
||||||
|
node->bv[(2 * i) + 1] += tree->epsilon; // maximum
|
||||||
|
}
|
||||||
|
|
||||||
|
node->index= index;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// only supports x,y,z axis in the moment
|
||||||
|
// but we should use a plain and simple function here for speed sake
|
||||||
|
static char get_largest_axis(float *bv)
|
||||||
|
{
|
||||||
|
float middle_point[3];
|
||||||
|
|
||||||
|
middle_point[0] = (bv[1]) - (bv[0]); // x axis
|
||||||
|
middle_point[1] = (bv[3]) - (bv[2]); // y axis
|
||||||
|
middle_point[2] = (bv[5]) - (bv[4]); // z axis
|
||||||
|
if (middle_point[0] > middle_point[1])
|
||||||
|
{
|
||||||
|
if (middle_point[0] > middle_point[2])
|
||||||
|
return 1; // max x axis
|
||||||
|
else
|
||||||
|
return 5; // max z axis
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (middle_point[1] > middle_point[2])
|
||||||
|
return 3; // max y axis
|
||||||
|
else
|
||||||
|
return 5; // max z axis
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void bvh_div_nodes(BVHTree *tree, BVHNode *node, int start, int end, char lastaxis)
|
||||||
|
{
|
||||||
|
char laxis;
|
||||||
|
int i, tend;
|
||||||
|
BVHNode *tnode;
|
||||||
|
int slice = (end-start+tree->tree_type-1)/tree->tree_type; //division rounded up
|
||||||
|
|
||||||
|
// Determine which axis to split along
|
||||||
|
laxis = get_largest_axis(node->bv);
|
||||||
|
|
||||||
|
// split nodes along longest axis
|
||||||
|
for (i=0; start < end; start += slice, i++) //i counts the current child
|
||||||
|
{
|
||||||
|
tend = start + slice;
|
||||||
|
|
||||||
|
if(tend > end) tend = end;
|
||||||
|
|
||||||
|
if(tend-start == 1) // ok, we have 1 left for this node
|
||||||
|
{
|
||||||
|
node->children[i] = tree->nodes[start];
|
||||||
|
node->children[i]->parent = node;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
tnode = node->children[i] = tree->nodes[tree->totleaf + tree->totbranch] = &(tree->nodearray[tree->totbranch + tree->totleaf]);
|
||||||
|
tree->totbranch++;
|
||||||
|
tnode->parent = node;
|
||||||
|
|
||||||
|
if(tend != end)
|
||||||
|
partition_nth_element(tree->nodes, start, end, tend, laxis);
|
||||||
|
refit_kdop_hull(tree, tnode, start, tend);
|
||||||
|
bvh_div_nodes(tree, tnode, start, tend, laxis);
|
||||||
|
}
|
||||||
|
node->totnode++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void verify_tree(BVHTree *tree)
|
||||||
|
{
|
||||||
|
int i, j, check = 0;
|
||||||
|
|
||||||
|
// check the pointer list
|
||||||
|
for(i = 0; i < tree->totleaf; i++)
|
||||||
|
{
|
||||||
|
if(tree->nodes[i]->parent == NULL)
|
||||||
|
printf("Leaf has no parent: %d\n", i);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for(j = 0; j < tree->tree_type; j++)
|
||||||
|
{
|
||||||
|
if(tree->nodes[i]->parent->children[j] == tree->nodes[i])
|
||||||
|
check = 1;
|
||||||
|
}
|
||||||
|
if(!check)
|
||||||
|
{
|
||||||
|
printf("Parent child relationship doesn't match: %d\n", i);
|
||||||
|
}
|
||||||
|
check = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// check the leaf list
|
||||||
|
for(i = 0; i < tree->totleaf; i++)
|
||||||
|
{
|
||||||
|
if(tree->nodearray[i].parent == NULL)
|
||||||
|
printf("Leaf has no parent: %d\n", i);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for(j = 0; j < tree->tree_type; j++)
|
||||||
|
{
|
||||||
|
if(tree->nodearray[i].parent->children[j] == &tree->nodearray[i])
|
||||||
|
check = 1;
|
||||||
|
}
|
||||||
|
if(!check)
|
||||||
|
{
|
||||||
|
printf("Parent child relationship doesn't match: %d\n", i);
|
||||||
|
}
|
||||||
|
check = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("branches: %d, leafs: %d, total: %d\n", tree->totbranch, tree->totleaf, tree->totbranch + tree->totleaf);
|
||||||
|
}
|
||||||
|
|
||||||
|
void BLI_bvhtree_balance(BVHTree *tree)
|
||||||
|
{
|
||||||
|
BVHNode *node;
|
||||||
|
|
||||||
|
if(tree->totleaf == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// create root node
|
||||||
|
node = tree->nodes[tree->totleaf] = &(tree->nodearray[tree->totleaf]);
|
||||||
|
tree->totbranch++;
|
||||||
|
|
||||||
|
// refit root bvh node
|
||||||
|
refit_kdop_hull(tree, tree->nodes[tree->totleaf], 0, tree->totleaf);
|
||||||
|
// create + balance tree
|
||||||
|
bvh_div_nodes(tree, tree->nodes[tree->totleaf], 0, tree->totleaf, 0);
|
||||||
|
|
||||||
|
// verify_tree(tree);
|
||||||
|
}
|
||||||
|
|
||||||
|
// overlap - is it possbile for 2 bv's to collide ?
|
||||||
|
static int tree_overlap(float *bv1, float *bv2, int start_axis, int stop_axis)
|
||||||
|
{
|
||||||
|
float *bv1_end = bv1 + (stop_axis<<1);
|
||||||
|
|
||||||
|
bv1 += start_axis<<1;
|
||||||
|
bv2 += start_axis<<1;
|
||||||
|
|
||||||
|
// test all axis if min + max overlap
|
||||||
|
for (; bv1 != bv1_end; bv1+=2, bv2+=2)
|
||||||
|
{
|
||||||
|
if ((*(bv1) > *(bv2 + 1)) || (*(bv2) > *(bv1 + 1)))
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void traverse(BVHOverlapData *data, BVHNode *node1, BVHNode *node2)
|
||||||
|
{
|
||||||
|
int j;
|
||||||
|
|
||||||
|
if(tree_overlap(node1->bv, node2->bv, MIN2(data->tree1->start_axis, data->tree2->start_axis), MIN2(data->tree1->stop_axis, data->tree2->stop_axis)))
|
||||||
|
{
|
||||||
|
// check if node1 is a leaf
|
||||||
|
if(!node1->totnode)
|
||||||
|
{
|
||||||
|
// check if node2 is a leaf
|
||||||
|
if(!node2->totnode)
|
||||||
|
{
|
||||||
|
|
||||||
|
if(node1 == node2)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(data->i >= data->max_overlap)
|
||||||
|
{
|
||||||
|
// try to make alloc'ed memory bigger
|
||||||
|
data->overlap = realloc(data->overlap, sizeof(BVHTreeOverlap)*data->max_overlap*2);
|
||||||
|
|
||||||
|
if(!data->overlap)
|
||||||
|
{
|
||||||
|
printf("Out of Memory in traverse\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
data->max_overlap *= 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
// both leafs, insert overlap!
|
||||||
|
data->overlap[data->i].indexA = node1->index;
|
||||||
|
data->overlap[data->i].indexB = node2->index;
|
||||||
|
|
||||||
|
data->i++;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for(j = 0; j < data->tree2->tree_type; j++)
|
||||||
|
{
|
||||||
|
if(node2->children[j])
|
||||||
|
traverse(data, node1, node2->children[j]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
|
||||||
|
for(j = 0; j < data->tree2->tree_type; j++)
|
||||||
|
{
|
||||||
|
if(node1->children[j])
|
||||||
|
traverse(data, node1->children[j], node2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
BVHTreeOverlap *BLI_bvhtree_overlap(BVHTree *tree1, BVHTree *tree2, int *result)
|
||||||
|
{
|
||||||
|
int j, total = 0;
|
||||||
|
BVHTreeOverlap *overlap = NULL, *to = NULL;
|
||||||
|
BVHOverlapData **data;
|
||||||
|
|
||||||
|
// check for compatibility of both trees (can't compare 14-DOP with 18-DOP)
|
||||||
|
if((tree1->axis != tree2->axis) && ((tree1->axis == 14) || tree2->axis == 14))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// fast check root nodes for collision before doing big splitting + traversal
|
||||||
|
if(!tree_overlap(tree1->nodes[tree1->totleaf]->bv, tree2->nodes[tree2->totleaf]->bv, MIN2(tree1->start_axis, tree2->start_axis), MIN2(tree1->stop_axis, tree2->stop_axis)))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
data = MEM_callocN(sizeof(BVHOverlapData *)* tree1->tree_type, "BVHOverlapData_star");
|
||||||
|
|
||||||
|
for(j = 0; j < tree1->tree_type; j++)
|
||||||
|
{
|
||||||
|
data[j] = (BVHOverlapData *)MEM_callocN(sizeof(BVHOverlapData), "BVHOverlapData");
|
||||||
|
|
||||||
|
// init BVHOverlapData
|
||||||
|
data[j]->overlap = (BVHTreeOverlap *)malloc(sizeof(BVHTreeOverlap)*MAX2(tree1->totleaf, tree2->totleaf));
|
||||||
|
data[j]->tree1 = tree1;
|
||||||
|
data[j]->tree2 = tree2;
|
||||||
|
data[j]->max_overlap = MAX2(tree1->totleaf, tree2->totleaf);
|
||||||
|
data[j]->i = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#pragma omp parallel for private(j) schedule(static)
|
||||||
|
for(j = 0; j < MIN2(tree1->tree_type, tree1->nodes[tree1->totleaf]->totnode); j++)
|
||||||
|
{
|
||||||
|
traverse(data[j], tree1->nodes[tree1->totleaf]->children[j], tree2->nodes[tree2->totleaf]);
|
||||||
|
}
|
||||||
|
|
||||||
|
for(j = 0; j < tree1->tree_type; j++)
|
||||||
|
total += data[j]->i;
|
||||||
|
|
||||||
|
to = overlap = (BVHTreeOverlap *)MEM_callocN(sizeof(BVHTreeOverlap)*total, "BVHTreeOverlap");
|
||||||
|
|
||||||
|
for(j = 0; j < tree1->tree_type; j++)
|
||||||
|
{
|
||||||
|
memcpy(to, data[j]->overlap, data[j]->i*sizeof(BVHTreeOverlap));
|
||||||
|
to+=data[j]->i;
|
||||||
|
}
|
||||||
|
|
||||||
|
for(j = 0; j < tree1->tree_type; j++)
|
||||||
|
{
|
||||||
|
free(data[j]->overlap);
|
||||||
|
MEM_freeN(data[j]);
|
||||||
|
}
|
||||||
|
MEM_freeN(data);
|
||||||
|
|
||||||
|
(*result) = total;
|
||||||
|
return overlap;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// bottom up update of bvh tree:
|
||||||
|
// join the 4 children here
|
||||||
|
static void node_join(BVHTree *tree, BVHNode *node)
|
||||||
|
{
|
||||||
|
int i, j;
|
||||||
|
|
||||||
|
for (i = tree->start_axis; i < tree->stop_axis; i++)
|
||||||
|
{
|
||||||
|
node->bv[2*i] = FLT_MAX;
|
||||||
|
node->bv[2*i + 1] = -FLT_MAX;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < tree->tree_type; i++)
|
||||||
|
{
|
||||||
|
if (node->children[i])
|
||||||
|
{
|
||||||
|
for (j = tree->start_axis; j < tree->stop_axis; j++)
|
||||||
|
{
|
||||||
|
// update minimum
|
||||||
|
if (node->children[i]->bv[(2 * j)] < node->bv[(2 * j)])
|
||||||
|
node->bv[(2 * j)] = node->children[i]->bv[(2 * j)];
|
||||||
|
|
||||||
|
// update maximum
|
||||||
|
if (node->children[i]->bv[(2 * j) + 1] > node->bv[(2 * j) + 1])
|
||||||
|
node->bv[(2 * j) + 1] = node->children[i]->bv[(2 * j) + 1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// call before BLI_bvhtree_update_tree()
|
||||||
|
int BLI_bvhtree_update_node(BVHTree *tree, int index, float *co, float *co_moving, int numpoints)
|
||||||
|
{
|
||||||
|
BVHNode *node= NULL;
|
||||||
|
int i = 0;
|
||||||
|
|
||||||
|
// check if index exists
|
||||||
|
if(index > tree->totleaf)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
node = tree->nodearray + index;
|
||||||
|
|
||||||
|
create_kdop_hull(tree, node, co, numpoints, 0);
|
||||||
|
|
||||||
|
if(co_moving)
|
||||||
|
create_kdop_hull(tree, node, co_moving, numpoints, 1);
|
||||||
|
|
||||||
|
// inflate the bv with some epsilon
|
||||||
|
for (i = tree->start_axis; i < tree->stop_axis; i++)
|
||||||
|
{
|
||||||
|
node->bv[(2 * i)] -= tree->epsilon; // minimum
|
||||||
|
node->bv[(2 * i) + 1] += tree->epsilon; // maximum
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// call BLI_bvhtree_update_node() first for every node/point/triangle
|
||||||
|
void BLI_bvhtree_update_tree(BVHTree *tree)
|
||||||
|
{
|
||||||
|
BVHNode *leaf, *parent;
|
||||||
|
|
||||||
|
// reset tree traversing flag
|
||||||
|
for (leaf = tree->nodearray + tree->totleaf; leaf != tree->nodearray + tree->totleaf + tree->totbranch; leaf++)
|
||||||
|
leaf->traversed = 0;
|
||||||
|
|
||||||
|
for (leaf = tree->nodearray; leaf != tree->nodearray + tree->totleaf; leaf++)
|
||||||
|
{
|
||||||
|
for (parent = leaf->parent; parent; parent = parent->parent)
|
||||||
|
{
|
||||||
|
parent->traversed++; // we tried to go up in hierarchy
|
||||||
|
if (parent->traversed < parent->totnode)
|
||||||
|
break; // we do not need to check further
|
||||||
|
else
|
||||||
|
node_join(tree, parent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
float BLI_bvhtree_getepsilon(BVHTree *tree)
|
||||||
|
{
|
||||||
|
return tree->epsilon;
|
||||||
|
}
|
||||||
@@ -456,13 +456,10 @@ void checkMissingFiles( char *txtname ) {
|
|||||||
|
|
||||||
/* be sure there is low chance of the path being too short */
|
/* be sure there is low chance of the path being too short */
|
||||||
char filepath_expanded[FILE_MAXDIR*2];
|
char filepath_expanded[FILE_MAXDIR*2];
|
||||||
char *libpath;
|
|
||||||
int files_missing = 0;
|
int files_missing = 0;
|
||||||
|
|
||||||
BLI_bpathIterator_init(&bpi);
|
BLI_bpathIterator_init(&bpi);
|
||||||
while (!BLI_bpathIterator_isDone(&bpi)) {
|
while (!BLI_bpathIterator_isDone(&bpi)) {
|
||||||
libpath = BLI_bpathIterator_getLib(&bpi);
|
|
||||||
|
|
||||||
BLI_bpathIterator_getPathExpanded( &bpi, filepath_expanded );
|
BLI_bpathIterator_getPathExpanded( &bpi, filepath_expanded );
|
||||||
|
|
||||||
if (!BLI_exists(filepath_expanded)) {
|
if (!BLI_exists(filepath_expanded)) {
|
||||||
|
|||||||
@@ -150,7 +150,7 @@ void freetypechar_to_vchar(FT_Face face, FT_ULong charcode, VFontData *vfd)
|
|||||||
nu->type= CU_BEZIER+CU_2D;
|
nu->type= CU_BEZIER+CU_2D;
|
||||||
nu->pntsu = onpoints[j];
|
nu->pntsu = onpoints[j];
|
||||||
nu->resolu= 8;
|
nu->resolu= 8;
|
||||||
nu->flagu= 1;
|
nu->flagu= CU_CYCLIC;
|
||||||
nu->bezt = bezt;
|
nu->bezt = bezt;
|
||||||
|
|
||||||
//individual curve loop, start-end
|
//individual curve loop, start-end
|
||||||
|
|||||||
@@ -2017,7 +2017,7 @@ static VFontData *objfnt_to_vfontdata(objfnt *fnt)
|
|||||||
nu->type= CU_BEZIER+CU_2D;
|
nu->type= CU_BEZIER+CU_2D;
|
||||||
nu->pntsu = count;
|
nu->pntsu = count;
|
||||||
nu->resolu= 8;
|
nu->resolu= 8;
|
||||||
nu->flagu= 1;
|
nu->flagu= CU_CYCLIC;
|
||||||
nu->bezt = bezt;
|
nu->bezt = bezt;
|
||||||
stop = 0;
|
stop = 0;
|
||||||
|
|
||||||
|
|||||||
@@ -863,11 +863,8 @@ int BLI_strcaseeq(char *a, char *b) {
|
|||||||
void BLI_cleanup_dir(const char *relabase, char *dir)
|
void BLI_cleanup_dir(const char *relabase, char *dir)
|
||||||
{
|
{
|
||||||
BLI_cleanup_file(relabase, dir);
|
BLI_cleanup_file(relabase, dir);
|
||||||
#ifdef WIN32
|
BLI_add_slash(dir);
|
||||||
strcat(dir, "\\");
|
|
||||||
#else
|
|
||||||
strcat(dir, "/");
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void BLI_cleanup_file(const char *relabase, char *dir)
|
void BLI_cleanup_file(const char *relabase, char *dir)
|
||||||
@@ -877,6 +874,13 @@ void BLI_cleanup_file(const char *relabase, char *dir)
|
|||||||
|
|
||||||
if (relabase) {
|
if (relabase) {
|
||||||
BLI_convertstringcode(dir, relabase);
|
BLI_convertstringcode(dir, relabase);
|
||||||
|
} else {
|
||||||
|
if (dir[0]=='/' && dir[1]=='/') {
|
||||||
|
if (dir[2]== '\0') {
|
||||||
|
return; /* path is "//" - cant clean it */
|
||||||
|
}
|
||||||
|
dir = dir+2; /* skip the first // */
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
@@ -1144,23 +1148,32 @@ int BLI_convertstringcode(char *path, const char *basepath)
|
|||||||
BLI_char_switch(tmp, '\\', '/');
|
BLI_char_switch(tmp, '\\', '/');
|
||||||
BLI_char_switch(base, '\\', '/');
|
BLI_char_switch(base, '\\', '/');
|
||||||
|
|
||||||
|
/* Paths starting with // will get the blend file as their base,
|
||||||
|
* this isnt standard in any os but is uesed in blender all over the place */
|
||||||
if (tmp[0] == '/' && tmp[1] == '/') {
|
if (tmp[0] == '/' && tmp[1] == '/') {
|
||||||
char *filepart= BLI_strdup(tmp+2); /* skip code */
|
|
||||||
char *lslash= BLI_last_slash(base);
|
char *lslash= BLI_last_slash(base);
|
||||||
|
|
||||||
if (lslash) {
|
if (lslash) {
|
||||||
int baselen= (int) (lslash-base) + 1;
|
int baselen= (int) (lslash-base) + 1;
|
||||||
|
/* use path for for temp storage here, we copy back over it right away */
|
||||||
|
BLI_strncpy(path, tmp+2, FILE_MAX);
|
||||||
|
|
||||||
memcpy(tmp, base, baselen);
|
memcpy(tmp, base, baselen);
|
||||||
strcpy(tmp+baselen, filepart);
|
strcpy(tmp+baselen, path);
|
||||||
} else {
|
|
||||||
strcpy(tmp, filepart);
|
|
||||||
}
|
|
||||||
|
|
||||||
MEM_freeN(filepart);
|
|
||||||
}
|
|
||||||
|
|
||||||
strcpy(path, tmp);
|
strcpy(path, tmp);
|
||||||
|
} else {
|
||||||
|
strcpy(path, tmp+2);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
strcpy(path, tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path[0]!='\0') {
|
||||||
|
if ( path[strlen(path)-1]=='/') {
|
||||||
|
BLI_cleanup_dir(NULL, path);
|
||||||
|
} else {
|
||||||
|
BLI_cleanup_file(NULL, path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
/* skip first two chars, which in case of
|
/* skip first two chars, which in case of
|
||||||
|
|||||||
@@ -3108,7 +3108,7 @@ static void direct_link_modifiers(FileData *fd, ListBase *lb)
|
|||||||
collmd->current_v = NULL;
|
collmd->current_v = NULL;
|
||||||
collmd->time = -1;
|
collmd->time = -1;
|
||||||
collmd->numverts = 0;
|
collmd->numverts = 0;
|
||||||
collmd->bvh = NULL;
|
collmd->bvhtree = NULL;
|
||||||
collmd->mfaces = NULL;
|
collmd->mfaces = NULL;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,6 +27,14 @@ if env['WITH_BF_FFMPEG'] == 1:
|
|||||||
defs.append('WITH_FFMPEG')
|
defs.append('WITH_FFMPEG')
|
||||||
incs += ' ' + env['BF_FFMPEG_INC']
|
incs += ' ' + env['BF_FFMPEG_INC']
|
||||||
|
|
||||||
|
if env['WITH_BF_OPENJPEG'] == 1:
|
||||||
|
defs.append('WITH_OPENJPEG')
|
||||||
|
incs += ' ' + env['BF_OPENJPEG_INC']
|
||||||
|
|
||||||
|
if env['WITH_BF_REDCODE'] == 1:
|
||||||
|
defs.append('WITH_REDCODE')
|
||||||
|
incs += ' ' + env['BF_REDCODE_INC']
|
||||||
|
|
||||||
if env['WITH_BF_QUICKTIME']==1:
|
if env['WITH_BF_QUICKTIME']==1:
|
||||||
incs += ' ' + env['BF_QUICKTIME_INC']
|
incs += ' ' + env['BF_QUICKTIME_INC']
|
||||||
defs.append('WITH_QUICKTIME')
|
defs.append('WITH_QUICKTIME')
|
||||||
|
|||||||
@@ -81,6 +81,10 @@
|
|||||||
#include <ffmpeg/swscale.h>
|
#include <ffmpeg/swscale.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
#include <redcode/format.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "IMB_imbuf_types.h"
|
#include "IMB_imbuf_types.h"
|
||||||
#include "IMB_imbuf.h"
|
#include "IMB_imbuf.h"
|
||||||
|
|
||||||
@@ -116,6 +120,7 @@
|
|||||||
#define ANIM_AVI (1 << 6)
|
#define ANIM_AVI (1 << 6)
|
||||||
#define ANIM_QTIME (1 << 7)
|
#define ANIM_QTIME (1 << 7)
|
||||||
#define ANIM_FFMPEG (1 << 8)
|
#define ANIM_FFMPEG (1 << 8)
|
||||||
|
#define ANIM_REDCODE (1 << 9)
|
||||||
|
|
||||||
#define ANIM5_MMAP 0
|
#define ANIM5_MMAP 0
|
||||||
#define ANIM5_MALLOC 1
|
#define ANIM5_MALLOC 1
|
||||||
@@ -184,6 +189,9 @@ struct anim {
|
|||||||
int videoStream;
|
int videoStream;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
struct redcode_handle * redcodeCtx;
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -96,6 +96,11 @@
|
|||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
#include <redcode/format.h>
|
||||||
|
#include <redcode/codec.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
/****/
|
/****/
|
||||||
|
|
||||||
#ifdef __sgi
|
#ifdef __sgi
|
||||||
@@ -307,6 +312,9 @@ void IMB_free_anim_ibuf(struct anim * anim) {
|
|||||||
#ifdef WITH_FFMPEG
|
#ifdef WITH_FFMPEG
|
||||||
static void free_anim_ffmpeg(struct anim * anim);
|
static void free_anim_ffmpeg(struct anim * anim);
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
static void free_anim_redcode(struct anim * anim);
|
||||||
|
#endif
|
||||||
|
|
||||||
void IMB_free_anim(struct anim * anim) {
|
void IMB_free_anim(struct anim * anim) {
|
||||||
if (anim == NULL) {
|
if (anim == NULL) {
|
||||||
@@ -325,6 +333,9 @@ void IMB_free_anim(struct anim * anim) {
|
|||||||
#ifdef WITH_FFMPEG
|
#ifdef WITH_FFMPEG
|
||||||
free_anim_ffmpeg(anim);
|
free_anim_ffmpeg(anim);
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
free_anim_redcode(anim);
|
||||||
|
#endif
|
||||||
|
|
||||||
free(anim);
|
free(anim);
|
||||||
}
|
}
|
||||||
@@ -830,6 +841,58 @@ static void free_anim_ffmpeg(struct anim * anim) {
|
|||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
|
||||||
|
static int startredcode(struct anim * anim) {
|
||||||
|
anim->redcodeCtx = redcode_open(anim->name);
|
||||||
|
if (!anim->redcodeCtx) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
anim->duration = redcode_get_length(anim->redcodeCtx);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ImBuf * redcode_fetchibuf(struct anim * anim, int position) {
|
||||||
|
struct ImBuf * ibuf;
|
||||||
|
struct redcode_frame * frame;
|
||||||
|
struct redcode_frame_raw * raw_frame;
|
||||||
|
|
||||||
|
if (!anim->redcodeCtx) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
frame = redcode_read_video_frame(anim->redcodeCtx, position);
|
||||||
|
|
||||||
|
if (!frame) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
raw_frame = redcode_decode_video_raw(frame, 1);
|
||||||
|
|
||||||
|
redcode_free_frame(frame);
|
||||||
|
|
||||||
|
if (!raw_frame) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
ibuf = IMB_allocImBuf(raw_frame->width * 2,
|
||||||
|
raw_frame->height * 2, 32, IB_rectfloat, 0);
|
||||||
|
|
||||||
|
redcode_decode_video_float(raw_frame, ibuf->rect_float, 1);
|
||||||
|
|
||||||
|
return ibuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void free_anim_redcode(struct anim * anim) {
|
||||||
|
if (anim->redcodeCtx) {
|
||||||
|
redcode_close(anim->redcodeCtx);
|
||||||
|
anim->redcodeCtx = 0;
|
||||||
|
}
|
||||||
|
anim->duration = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
/* probeer volgende plaatje te lezen */
|
/* probeer volgende plaatje te lezen */
|
||||||
/* Geen plaatje, probeer dan volgende animatie te openen */
|
/* Geen plaatje, probeer dan volgende animatie te openen */
|
||||||
@@ -849,6 +912,10 @@ static struct ImBuf * anim_getnew(struct anim * anim) {
|
|||||||
#ifdef WITH_FFMPEG
|
#ifdef WITH_FFMPEG
|
||||||
free_anim_ffmpeg(anim);
|
free_anim_ffmpeg(anim);
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
free_anim_redcode(anim);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
if (anim->curtype != 0) return (0);
|
if (anim->curtype != 0) return (0);
|
||||||
anim->curtype = imb_get_anim_type(anim->name);
|
anim->curtype = imb_get_anim_type(anim->name);
|
||||||
@@ -887,9 +954,14 @@ static struct ImBuf * anim_getnew(struct anim * anim) {
|
|||||||
if (startffmpeg(anim)) return (0);
|
if (startffmpeg(anim)) return (0);
|
||||||
ibuf = IMB_allocImBuf (anim->x, anim->y, 24, 0, 0);
|
ibuf = IMB_allocImBuf (anim->x, anim->y, 24, 0, 0);
|
||||||
break;
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
case ANIM_REDCODE:
|
||||||
|
if (startredcode(anim)) return (0);
|
||||||
|
ibuf = IMB_allocImBuf (8, 8, 32, 0, 0);
|
||||||
|
break;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
return(ibuf);
|
return(ibuf);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -969,6 +1041,12 @@ struct ImBuf * IMB_anim_absolute(struct anim * anim, int position) {
|
|||||||
ibuf = ffmpeg_fetchibuf(anim, position);
|
ibuf = ffmpeg_fetchibuf(anim, position);
|
||||||
if (ibuf) anim->curposition = position;
|
if (ibuf) anim->curposition = position;
|
||||||
break;
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
case ANIM_REDCODE:
|
||||||
|
ibuf = redcode_fetchibuf(anim, position);
|
||||||
|
if (ibuf) anim->curposition = position;
|
||||||
|
break;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -321,6 +321,19 @@ static int isffmpeg (char *filename) {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
static int isredcode(char * filename)
|
||||||
|
{
|
||||||
|
struct redcode_handle * h = redcode_open(filename);
|
||||||
|
if (!h) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
redcode_close(h);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
int imb_get_anim_type(char * name) {
|
int imb_get_anim_type(char * name) {
|
||||||
int type;
|
int type;
|
||||||
struct stat st;
|
struct stat st;
|
||||||
@@ -354,6 +367,9 @@ int imb_get_anim_type(char * name) {
|
|||||||
# ifdef WITH_FFMPEG
|
# ifdef WITH_FFMPEG
|
||||||
if (isffmpeg(name)) return (ANIM_FFMPEG);
|
if (isffmpeg(name)) return (ANIM_FFMPEG);
|
||||||
# endif
|
# endif
|
||||||
|
#endif
|
||||||
|
#ifdef WITH_REDCODE
|
||||||
|
if (isredcode(name)) return (ANIM_REDCODE);
|
||||||
#endif
|
#endif
|
||||||
type = IMB_ispic(name);
|
type = IMB_ispic(name);
|
||||||
if (type == ANIM) return (ANIM_ANIM5);
|
if (type == ANIM) return (ANIM_ANIM5);
|
||||||
@@ -369,6 +385,7 @@ int IMB_isanim(char *filename) {
|
|||||||
if( BLI_testextensie(filename, ".avi")
|
if( BLI_testextensie(filename, ".avi")
|
||||||
|| BLI_testextensie(filename, ".flc")
|
|| BLI_testextensie(filename, ".flc")
|
||||||
|| BLI_testextensie(filename, ".dv")
|
|| BLI_testextensie(filename, ".dv")
|
||||||
|
|| BLI_testextensie(filename, ".r3d")
|
||||||
|| BLI_testextensie(filename, ".mov")
|
|| BLI_testextensie(filename, ".mov")
|
||||||
|| BLI_testextensie(filename, ".movie")
|
|| BLI_testextensie(filename, ".movie")
|
||||||
|| BLI_testextensie(filename, ".mv")) {
|
|| BLI_testextensie(filename, ".mv")) {
|
||||||
@@ -379,6 +396,7 @@ int IMB_isanim(char *filename) {
|
|||||||
} else { // no quicktime
|
} else { // no quicktime
|
||||||
if( BLI_testextensie(filename, ".avi")
|
if( BLI_testextensie(filename, ".avi")
|
||||||
|| BLI_testextensie(filename, ".dv")
|
|| BLI_testextensie(filename, ".dv")
|
||||||
|
|| BLI_testextensie(filename, ".r3d")
|
||||||
|| BLI_testextensie(filename, ".mv")) {
|
|| BLI_testextensie(filename, ".mv")) {
|
||||||
type = imb_get_anim_type(filename);
|
type = imb_get_anim_type(filename);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,6 +37,9 @@ struct BezTriple;
|
|||||||
struct BPoint;
|
struct BPoint;
|
||||||
struct BezTripleNurb;
|
struct BezTripleNurb;
|
||||||
|
|
||||||
|
void set_actNurb(struct Nurb *nu);
|
||||||
|
struct Nurb * get_actNurb( void );
|
||||||
|
|
||||||
short isNurbsel(struct Nurb *nu);
|
short isNurbsel(struct Nurb *nu);
|
||||||
int isNurbsel_count(struct Nurb *nu);
|
int isNurbsel_count(struct Nurb *nu);
|
||||||
void printknots(void);
|
void printknots(void);
|
||||||
|
|||||||
@@ -92,8 +92,8 @@ start and end are from the start and fixed length of the sequence.
|
|||||||
int seq_tx_get_start(struct Sequence *seq);
|
int seq_tx_get_start(struct Sequence *seq);
|
||||||
int seq_tx_get_end(struct Sequence *seq);
|
int seq_tx_get_end(struct Sequence *seq);
|
||||||
|
|
||||||
int seq_tx_get_final_left(struct Sequence *seq);
|
int seq_tx_get_final_left(struct Sequence *seq, int metaclip);
|
||||||
int seq_tx_get_final_right(struct Sequence *seq);
|
int seq_tx_get_final_right(struct Sequence *seq, int metaclip);
|
||||||
|
|
||||||
void seq_tx_set_final_left(struct Sequence *seq, int i);
|
void seq_tx_set_final_left(struct Sequence *seq, int i);
|
||||||
void seq_tx_set_final_right(struct Sequence *seq, int i);
|
void seq_tx_set_final_right(struct Sequence *seq, int i);
|
||||||
|
|||||||
@@ -63,7 +63,9 @@ void window_to_3d(float *vec, short mx, short my);
|
|||||||
void project_short(float *vec, short *adr);
|
void project_short(float *vec, short *adr);
|
||||||
void project_short_noclip(float *vec, short *adr);
|
void project_short_noclip(float *vec, short *adr);
|
||||||
void project_int(float *vec, int *adr);
|
void project_int(float *vec, int *adr);
|
||||||
|
void project_int_noclip(float *vec, int *adr);
|
||||||
void project_float(float *vec, float *adr);
|
void project_float(float *vec, float *adr);
|
||||||
|
void project_float_noclip(float *vec, float *adr);
|
||||||
|
|
||||||
int boundbox_clip(float obmat[][4], struct BoundBox *bb);
|
int boundbox_clip(float obmat[][4], struct BoundBox *bb);
|
||||||
void fdrawline(float x1, float y1, float x2, float y2);
|
void fdrawline(float x1, float y1, float x2, float y2);
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user