Error handling for small output buffers/damaged compressed data. Small performance improvements, and a litle refactoring.

This commit is contained in:
jdv_cp 2009-03-09 13:32:44 -07:00
parent 068d887135
commit 105136964c
11 changed files with 242 additions and 262 deletions

View File

@ -40,9 +40,6 @@ struct CContextRunMode
}
void UpdateVariables(int Errval, UINT EMErrval)
{
if (Errval < 0)

View File

@ -35,13 +35,28 @@ public:
}
inlinehint void Skip(int length)
{
_cbitValid -= length;
_valcurrent = _valcurrent << length;
}
void OnLineBegin(void* /*ptypeCur*/, void* /*ptypeLine*/, int /*cpixel*/) {}
template <class T>
void OnLineEnd(T* ptypeCur, T* ptypeLine, int cpixel)
{
#ifdef _DEBUG
for (int i = 0; i < cpixel; ++i)
{
//CheckedAssign(ptypeLine[i], ptypeCur[i]);
ptypeLine[i] = ptypeCur[i];
}
#else
memcpy(ptypeLine, ptypeCur, cpixel * sizeof(T));
#endif
}
void MakeValid()

View File

@ -24,6 +24,16 @@ public:
virtual ~EncoderStrategy()
{}
template <class T>
void OnLineBegin(T* ptypeCur, T* ptypeLine, int cpixel)
{
memcpy(ptypeCur, ptypeLine, cpixel * sizeof(T));
}
void OnLineEnd(void* /*ptypeCur*/, void* /*ptypeLine*/, int /*cpixel*/) {};
virtual void SetPresets(const JlsCustomParameters& presets) = 0;
virtual int EncodeScan(const void* pvoid, const Size& size, int ccomp, void* pvoidOut, int cbyte, void* pvoidCompare) = 0;

View File

@ -110,11 +110,11 @@ JLSOutputStream::JLSOutputStream() :
//
JLSOutputStream::~JLSOutputStream()
{
for (UINT i = 0; i < _rgsegment.size(); ++i)
for (UINT i = 0; i < _segments.size(); ++i)
{
delete _rgsegment[i];
delete _segments[i];
}
_rgsegment.empty();
_segments.empty();
}
@ -125,7 +125,7 @@ JLSOutputStream::~JLSOutputStream()
//
void JLSOutputStream::Init(Size size, int cbpp, int ccomp)
{
_rgsegment.push_back(CreateMarkerStartOfFrame(size, cbpp, ccomp));
_segments.push_back(CreateMarkerStartOfFrame(size, cbpp, ccomp));
}
@ -142,9 +142,9 @@ int JLSOutputStream::Write(BYTE* pdata, int cbyteLength)
WriteByte(JPEG_SOI);
for (UINT i = 0; i < _rgsegment.size(); ++i)
for (UINT i = 0; i < _segments.size(); ++i)
{
_rgsegment[i]->Write(this);
_segments[i]->Write(this);
}
//_bCompare = false;
@ -161,7 +161,8 @@ JLSInputStream::JLSInputStream(const BYTE* pdata, int cbyteLength) :
_pdata(pdata),
_cbyteOffset(0),
_cbyteLength(cbyteLength),
_bCompare(false)
_bCompare(false),
_info()
{
}
@ -183,23 +184,24 @@ bool JLSInputStream::Read(void* pvoid, int cbyteAvailable)
//
bool JLSInputStream::ReadPixels(void* pvoid, int cbyteAvailable)
{
int cbytePlane = _info.size.cx * _info.size.cy * ((_info.cbit + 7)/8);
int cbytePlane = _info.width * _info.height * ((_info.bitspersample + 7)/8);
if (cbyteAvailable < cbytePlane * _info.ccomp)
return false;
if (cbyteAvailable < cbytePlane * _info.components)
throw JlsException(UncompressedBufferTooSmall);
if (_info.ilv == ILV_NONE)
{
BYTE* pbyte = (BYTE*)pvoid;
for (int icomp = 0; icomp < _info.ccomp; ++icomp)
for (int icomp = 0; icomp < _info.components; ++icomp)
{
ReadScan(pbyte);
ReadScan(pbyte);
pbyte += cbytePlane;
}
}
else
{
ReadScan(pvoid);
ReadScan(pvoid);
}
return true;
@ -306,11 +308,11 @@ void JLSInputStream::ReadPresetParameters()
{
case 1:
{
_presets.MAXVAL = ReadWord();
_presets.T1 = ReadWord();
_presets.T2 = ReadWord();
_presets.T3 = ReadWord();
_presets.RESET = ReadWord();
_info.custom.MAXVAL = ReadWord();
_info.custom.T1 = ReadWord();
_info.custom.T2 = ReadWord();
_info.custom.T3 = ReadWord();
_info.custom.RESET = ReadWord();
return;
}
}
@ -330,7 +332,7 @@ void JLSInputStream::ReadStartOfScan()
ReadByte();
ReadByte();
}
_info.nnear = ReadByte();
_info.allowedlossyerror = ReadByte();
_info.ilv = interleavemode(ReadByte());
}
@ -347,11 +349,12 @@ void JLSInputStream::ReadComment()
//
void JLSInputStream::ReadStartOfFrame()
{
_info.cbit = ReadByte();
_info.bitspersample = ReadByte();
int cline = ReadWord();
int ccol = ReadWord();
_info.size = Size(ccol, cline);
_info.ccomp = ReadByte();
_info.width = ccol;
_info.height = cline;
_info.components= ReadByte();
}
@ -375,9 +378,10 @@ int JLSInputStream::ReadWord()
void JLSInputStream::ReadScan(void* pvout)
{
std::auto_ptr<DecoderStrategy> qcodec(JlsCodecFactory<DecoderStrategy>().GetCodec(_info, _presets));
int cline = _info.ilv == ILV_LINE ? _info.ccomp : 1;
_cbyteOffset += qcodec->DecodeScan(pvout, _info.size, cline, _pdata + _cbyteOffset, _cbyteLength - _cbyteOffset, _bCompare);
std::auto_ptr<DecoderStrategy> qcodec(JlsCodecFactory<DecoderStrategy>().GetCodec(_info, _info.custom));
int cline = _info.ilv == ILV_LINE ? _info.components : 1;
Size size = Size(_info.width,_info.height);
_cbyteOffset += qcodec->DecodeScan(pvout, size, cline, _pdata + _cbyteOffset, _cbyteLength - _cbyteOffset, _bCompare);
};
@ -400,11 +404,11 @@ public:
void Write(JLSOutputStream* pstream)
{
ScanInfo info;
info.cbit = _cbit;
info.ccomp = _ccompScan;
JlsParamaters info;
info.bitspersample = _cbit;
info.components = _ccompScan;
info.ilv = _ilv;
info.nnear = _nnear;
info.allowedlossyerror = _nnear;
int ccompInterleaved = _ilv == ILV_LINE ? _ccompScan : 1;
@ -432,15 +436,15 @@ void JLSOutputStream::AddScan(const void* pbyteComp, const JlsParamaters* pparam
{
if (!IsDefault(&pparams->custom))
{
_rgsegment.push_back(CreateLSE(&pparams->custom));
_segments.push_back(CreateLSE(&pparams->custom));
}
_icompLast += 1;
_rgsegment.push_back(EncodeStartOfScan(pparams,pparams->ilv == ILV_NONE ? _icompLast : -1));
_segments.push_back(EncodeStartOfScan(pparams,pparams->ilv == ILV_NONE ? _icompLast : -1));
Size size = Size(pparams->width, pparams->height);
int ccomp = pparams->ilv == ILV_NONE ? 1 : pparams->components;
_rgsegment.push_back(new JpegImageDataSegment(pbyteComp, size, pparams->bitspersample, _icompLast, ccomp, pparams->ilv, pparams->allowedlossyerror, pparams->custom));
_segments.push_back(new JpegImageDataSegment(pbyteComp, size, pparams->bitspersample, _icompLast, ccomp, pparams->ilv, pparams->allowedlossyerror, pparams->custom));
}

View File

@ -24,9 +24,9 @@ template<class STRATEGY>
class JlsCodecFactory
{
public:
STRATEGY* GetCodec(const ScanInfo& info, const JlsCustomParameters&);
STRATEGY* GetCodec(const JlsParamaters& info, const JlsCustomParameters&);
private:
STRATEGY* GetCodecImpl(const ScanInfo& info);
STRATEGY* GetCodecImpl(const JlsParamaters& info);
};

View File

@ -87,11 +87,19 @@ __declspec(dllexport) JLS_ERROR JpegLsDecode(void* pdataUncompressed, int cbyteU
{
JLSInputStream reader((BYTE*)pdataCompressed, cbyteCompressed);
if (!reader.Read(pdataUncompressed, cbyteUncompressed))
return InvalidCompressedData;
try
{
if (!reader.Read(pdataUncompressed, cbyteUncompressed))
return InvalidCompressedData;
reader.GetBytesRead();
return OK;
reader.GetBytesRead();
return OK;
}
catch (JlsException e)
{
return e._error;
}
}
@ -145,14 +153,8 @@ __declspec(dllexport) JLS_ERROR JpegLsReadHeader(const void* pdataCompressed, in
{
JLSInputStream reader((BYTE*)pdataCompressed, cbyteCompressed);
reader.ReadHeader();
ScanInfo info = reader.GetMetadata();
pparams->width = info.size.cx;
pparams->height = info.size.cy;
pparams->components= info.ccomp;
pparams->bitspersample = info.cbit;
pparams->allowedlossyerror = info.nnear;
pparams->ilv = info.ilv;
pparams->custom = reader.GetCustomPreset();
JlsParamaters info = reader.GetMetadata();
*pparams = info;
return OK;
}
}

View File

@ -12,6 +12,15 @@ enum JLS_ERROR
InvalidCompressedData
};
class JlsException
{
public:
JlsException(JLS_ERROR error) : _error(error)
{ }
JLS_ERROR _error;
};
enum interleavemode
{
ILV_NONE = 0,

View File

@ -48,6 +48,7 @@ signed char QuantizeGratientOrg(const Presets& preset, int NEAR, int Di)
}
std::vector<signed char> CreateQLutLossless(int cbit)
{
Presets preset = ComputeDefault((1 << cbit) - 1, 0);
@ -75,12 +76,12 @@ std::vector<signed char> rgquant16Ll = CreateQLutLossless(16);
template<class STRATEGY>
STRATEGY* JlsCodecFactory<STRATEGY>::GetCodec(const ScanInfo& _info, const JlsCustomParameters& presets)
STRATEGY* JlsCodecFactory<STRATEGY>::GetCodec(const JlsParamaters& _info, const JlsCustomParameters& presets)
{
STRATEGY* pstrategy = NULL;
if (presets.RESET != 0 && presets.RESET != BASIC_RESET)
{
typename DefaultTraitsT<BYTE,BYTE> traits((1 << _info.cbit) - 1, _info.nnear);
typename DefaultTraitsT<BYTE,BYTE> traits((1 << _info.bitspersample) - 1, _info.allowedlossyerror);
traits.MAXVAL = presets.MAXVAL;
traits.RESET = presets.RESET;
pstrategy = new JlsCodec<DefaultTraitsT<BYTE, BYTE>, STRATEGY>(traits);
@ -98,16 +99,16 @@ STRATEGY* JlsCodecFactory<STRATEGY>::GetCodec(const ScanInfo& _info, const JlsCu
}
template<class STRATEGY>
STRATEGY* JlsCodecFactory<STRATEGY>::GetCodecImpl(const ScanInfo& _info)
STRATEGY* JlsCodecFactory<STRATEGY>::GetCodecImpl(const JlsParamaters& _info)
{
if (_info.ccomp != 1 && _info.ilv == ILV_SAMPLE)
if (_info.components != 1 && _info.ilv == ILV_SAMPLE)
{
if (_info.ccomp == 3 && _info.cbit == 8)
if (_info.components == 3 && _info.bitspersample == 8)
{
if (_info.nnear == 0)
if (_info.allowedlossyerror == 0)
return new JlsCodec<LosslessTraitsT<Triplet,8>, STRATEGY>();
typename DefaultTraitsT<BYTE,Triplet> traits((1 << _info.cbit) - 1, _info.nnear);
typename DefaultTraitsT<BYTE,Triplet> traits((1 << _info.bitspersample) - 1, _info.allowedlossyerror);
return new JlsCodec<DefaultTraitsT<BYTE,Triplet>, STRATEGY>(traits);
}
@ -115,9 +116,9 @@ STRATEGY* JlsCodecFactory<STRATEGY>::GetCodecImpl(const ScanInfo& _info)
}
// optimized lossless versions common monochrome formats (lossless)
if (_info.nnear == 0)
if (_info.allowedlossyerror == 0)
{
switch (_info.cbit)
switch (_info.bitspersample)
{
case 8: return new JlsCodec<LosslessTraitsT<BYTE, 8>, STRATEGY>();
case 10: return new JlsCodec<LosslessTraitsT<USHORT,10>, STRATEGY>();
@ -127,15 +128,15 @@ STRATEGY* JlsCodecFactory<STRATEGY>::GetCodecImpl(const ScanInfo& _info)
}
if (_info.cbit <= 8)
if (_info.bitspersample <= 8)
{
typename DefaultTraitsT<BYTE, BYTE> traits((1 << _info.cbit) - 1, _info.nnear);
typename DefaultTraitsT<BYTE, BYTE> traits((1 << _info.bitspersample) - 1, _info.allowedlossyerror);
return new JlsCodec<DefaultTraitsT<BYTE, BYTE>, STRATEGY>(traits);
}
if (_info.cbit <= 16)
if (_info.bitspersample <= 16)
{
typename DefaultTraitsT<USHORT, USHORT> traits((1 << _info.cbit) - 1, _info.nnear);
typename DefaultTraitsT<USHORT, USHORT> traits((1 << _info.bitspersample) - 1, _info.allowedlossyerror);
return new JlsCodec<DefaultTraitsT<USHORT, USHORT>, STRATEGY>(traits);
}
return NULL;

255
scan.h
View File

@ -8,6 +8,10 @@
#include "lookuptable.h"
//
// Apply
//
@ -124,7 +128,6 @@ public:
_pquant(0),
_size(0,0)
{
memset(_rghistogramK, 0, sizeof(_rghistogramK));
}
JlsCodec(const TRAITS& inTraits) :
@ -137,7 +140,6 @@ public:
_pquant(0),
_size(0,0)
{
memset(_rghistogramK, 0, sizeof(_rghistogramK));
}
@ -174,32 +176,20 @@ public:
int DecodeRIError(CContextRunMode& ctx);
Triplet DecodeRIPixel(Triplet Ra, Triplet Rb);
PIXEL DecodeRIPixel(int Ra, int Rb);
int DecodeRunPixels(PIXEL Ra, PIXEL* ptype, int ipixel, int cpixelMac);
int DoRunMode(PIXEL* ptype, const PIXEL* ptypePrev, int ipixel, int cpixelMac, DecoderStrategy*);
int DecodeRunPixels(PIXEL Ra, PIXEL* ptype, int cpixelMac);
int DoRunMode(int ipixel, DecoderStrategy*);
void EncodeRIError(CContextRunMode& ctx, int Errval);
SAMPLE EncodeRIPixel(int x, int Ra, int Rb);
Triplet EncodeRIPixel(Triplet x, Triplet Ra, Triplet Rb);
void EncodeRunPixels(int RUNcnt, bool bEndofline);
int DoRunMode(PIXEL* ptype, const PIXEL* ptypePrev, int ipixel, int ctypeRem, EncoderStrategy*);
void EncodeRunPixels(int runLength, bool bEndofline);
int DoRunMode(int ipixel, EncoderStrategy*);
inlinehint SAMPLE DoRegular(int Qs, int, int pred, DecoderStrategy*);
inlinehint SAMPLE DoRegular(int Qs, int x, int pred, EncoderStrategy*);
inlinehint void CheckedAssign(SAMPLE& typeDst, SAMPLE type)
{
ASSERT(!_bCompare || traits.IsNear(typeDst, type));
typeDst = type;
}
inlinehint void CheckedAssign(Triplet& typeDst, Triplet type)
{
ASSERT(!_bCompare || traits.IsNear(typeDst, type));
typeDst = type;
}
void DoLine(SAMPLE* ptype, const SAMPLE* ptypePrev);
void DoLine(Triplet* ptype, const Triplet* ptypePrev);
void DoLine(SAMPLE* pdummy);
void DoLine(Triplet* pdummy);
void DoScan(PIXEL* ptype, BYTE* pbyteCompressed, int cbyteCompressed);
public:
@ -210,21 +200,26 @@ public:
int DecodeScan(void* pvoidOut, const Size& size, int components, const void* pvoidIn, int cbyte, bool bCompare);
protected:
TRAITS traits;
signed char* _pquant;
std::vector<signed char> _rgquant;
JlsContext _rgcontext[365];
// compression context
JlsContext _contexts[365];
CContextRunMode _contextRunmode[2];
int RUNindex;
Size _size;
int _components; // for line interleaved mode
PIXEL* ptypePrev; // previous line ptr
PIXEL* ptypeCur; // current line ptr
// codec parameters
TRAITS traits;
Size _size;
int _components; // only set for line interleaved mode
int T3;
int T2;
int T1;
// quantization lookup table
signed char* _pquant;
std::vector<signed char> _rgquant;
// debugging
int _rghistogramK[16];
bool _bCompare;
};
@ -234,7 +229,7 @@ template<class TRAITS, class STRATEGY>
typename TRAITS::SAMPLE JlsCodec<TRAITS,STRATEGY>::DoRegular(int Qs, int, int pred, DecoderStrategy*)
{
int sign = BitWiseSign(Qs);
JlsContext& ctx = _rgcontext[ApplySign(Qs, sign)];
JlsContext& ctx = _contexts[ApplySign(Qs, sign)];
int k = ctx.GetGolomb();
int Px = traits.CorrectPrediction(pred + ApplySign(ctx.C, sign));
@ -249,7 +244,8 @@ typename TRAITS::SAMPLE JlsCodec<TRAITS,STRATEGY>::DoRegular(int Qs, int, int pr
else
{
ErrVal = UnMapErrVal(DecodeValue(k, traits.LIMIT, traits.qbpp));
ASSERT(abs(ErrVal) < 65535);
if (abs(ErrVal) > 65535)
throw JlsException(InvalidCompressedData);
}
ErrVal = ErrVal ^ ((traits.NEAR == 0) ? ctx.GetErrorCorrection(k) : 0);
ctx.UpdateVariables(ErrVal, traits.NEAR, traits.RESET);
@ -261,9 +257,10 @@ template<class TRAITS, class STRATEGY>
typename TRAITS::SAMPLE JlsCodec<TRAITS,STRATEGY>::DoRegular(int Qs, int x, int pred, EncoderStrategy*)
{
int sign = BitWiseSign(Qs);
JlsContext& ctx = _rgcontext[ApplySign(Qs, sign)];
JlsContext& ctx = _contexts[ApplySign(Qs, sign)];
int k = ctx.GetGolomb();
int Px = traits.CorrectPrediction(pred + ApplySign(ctx.C, sign));
int ErrVal = traits.ComputeErrVal(ApplySign(x - Px, sign));
EncodeMappedValue(k, GetMappedErrVal(ctx.GetErrorCorrection(k | traits.NEAR) ^ ErrVal), traits.LIMIT);
@ -366,6 +363,7 @@ inlinehint void JlsCodec<TRAITS,STRATEGY>::EncodeMappedValue(int k, UINT mappede
template<class TRAITS, class STRATEGY>
void JlsCodec<TRAITS,STRATEGY>::InitQuantizationLUT()
{
// for lossless mode with default paramaters, we have precomputed te luts for bitcounts 8,10,12 and 16
if (traits.NEAR == 0 && traits.MAXVAL == (1 << traits.bpp) - 1)
{
Presets presets = ComputeDefault(traits.MAXVAL, traits.NEAR);
@ -426,74 +424,39 @@ signed char JlsCodec<TRAITS,STRATEGY>::QuantizeGratientOrg(int Di)
template<class TRAITS, class STRATEGY>
void JlsCodec<TRAITS,STRATEGY>::EncodeRunPixels(int RUNcnt, bool bEndofline)
void JlsCodec<TRAITS,STRATEGY>::EncodeRunPixels(int runLength, bool bEndofline)
{
while (RUNcnt >= (1 << J[RUNindex]))
while (runLength >= (1 << J[RUNindex]))
{
AppendOnesToBitStream(1);
RUNcnt = RUNcnt - (1 << J[RUNindex]);
runLength = runLength - (1 << J[RUNindex]);
IncrementRunIndex();
}
if (bEndofline)
{
if (RUNcnt != 0)
if (runLength != 0)
{
AppendOnesToBitStream(1);
}
}
else
{
AppendToBitStream(RUNcnt, J[RUNindex] + 1); // leading 0 + actual remaining length
AppendToBitStream(runLength, J[RUNindex] + 1); // leading 0 + actual remaining length
}
}
/*
template<class TRAITS, class STRATEGY>
int JlsCodec<TRAITS,STRATEGY>::DecodeRunPixels(PIXEL Ra, PIXEL* ptype, int ipixel, int cpixelMac)
{
int cpixelRun = 0;
while (ReadBit())
{
int cpixel = min(1 << J[RUNindex], cpixelMac - ipixel);
cpixelRun += cpixel;
if (cpixel == (1 << J[RUNindex]))
{
IncrementRunIndex();
}
}
if (ipixel + cpixelRun < cpixelMac)
{
// incomplete run
cpixelRun += (J[RUNindex] > 0) ? ReadValue(J[RUNindex]) : 0;
}
for (int i = 0; i < cpixelRun; ++i)
{
ptype[ipixel + i] = Ra;
}
return ipixel + cpixelRun;
}
*/
template<class TRAITS, class STRATEGY>
int JlsCodec<TRAITS,STRATEGY>::DecodeRunPixels(PIXEL Ra, PIXEL* ptype, int ipixel, int cpixelMac)
int JlsCodec<TRAITS,STRATEGY>::DecodeRunPixels(PIXEL Ra, PIXEL* ptype, int cpixelMac)
{
int ipixel = 0;
while (ReadBit())
{
int cpixel = min(1 << J[RUNindex], cpixelMac - ipixel);
for (int i = 0; i < cpixel; ++i)
{
ptype[ipixel] = Ra;
ipixel++;
ASSERT(ipixel <= cpixelMac);
}
ipixel += cpixel;
ASSERT(ipixel <= cpixelMac);
if (cpixel == (1 << J[RUNindex]))
{
@ -501,18 +464,21 @@ int JlsCodec<TRAITS,STRATEGY>::DecodeRunPixels(PIXEL Ra, PIXEL* ptype, int ipixe
}
if (ipixel == cpixelMac)
return ipixel;
break;
}
// incomplete run
UINT cpixelRun = (J[RUNindex] > 0) ? ReadValue(J[RUNindex]) : 0;
for (UINT i = 0; i < cpixelRun; ++i)
if (ipixel != cpixelMac)
{
ptype[ipixel] = Ra;
ipixel++;
// incomplete run
ipixel += (J[RUNindex] > 0) ? ReadValue(J[RUNindex]) : 0;
}
for (int i = 0; i < ipixel; ++i)
{
ptype[i] = Ra;
}
return ipixel;
}
@ -550,17 +516,9 @@ Triplet JlsCodec<TRAITS,STRATEGY>::DecodeRIPixel(Triplet Ra, Triplet Rb)
int Errval2 = DecodeRIError(_contextRunmode[0]);
int Errval3 = DecodeRIError(_contextRunmode[0]);
//*
return Triplet(traits.ComputeReconstructedSample(Rb.v1, Errval1 * Sign(Rb.v1 - Ra.v1)),
traits.ComputeReconstructedSample(Rb.v2, Errval2 * Sign(Rb.v2 - Ra.v2)),
traits.ComputeReconstructedSample(Rb.v3, Errval3 * Sign(Rb.v3 - Ra.v3)));
/*/
return Triplet(Rb.v1 + Errval1 * Sign(Rb.v1 - Ra.v1),
Rb.v2 + Errval2 * Sign(Rb.v2 - Ra.v2),
Rb.v3 + Errval3 * Sign(Rb.v3 - Ra.v3));
//*/
}
@ -625,60 +583,61 @@ typename TRAITS::SAMPLE JlsCodec<TRAITS,STRATEGY>::EncodeRIPixel(int x, int Ra,
template<class TRAITS, class STRATEGY>
int JlsCodec<TRAITS,STRATEGY>::DoRunMode(PIXEL* ptype, const PIXEL* ptypePrev, int ipixel, int ctypeRem, EncoderStrategy*)
int JlsCodec<TRAITS,STRATEGY>::DoRunMode(int ipixel, EncoderStrategy*)
{
ptype += ipixel;
ptypePrev += ipixel;
ctypeRem -= ipixel;
int ctypeRem = _size.cx - ipixel;
PIXEL* ptypeCurX = ptypeCur + ipixel;
PIXEL* ptypePrevX = ptypePrev + ipixel;
PIXEL Ra = ptype[-1];
PIXEL Ra = ptypeCurX[-1];
int RUNcnt = 0;
int runLength = 0;
while (traits.IsNear(ptype[RUNcnt],Ra))
while (traits.IsNear(ptypeCurX[runLength],Ra))
{
ptype[RUNcnt] = Ra;
RUNcnt++;
ptypeCurX[runLength] = Ra;
runLength++;
if (RUNcnt == ctypeRem)
if (runLength == ctypeRem)
break;
}
EncodeRunPixels(RUNcnt, RUNcnt == ctypeRem);
EncodeRunPixels(runLength, runLength == ctypeRem);
if (RUNcnt == ctypeRem)
return RUNcnt;
if (runLength == ctypeRem)
return runLength;
ptype[RUNcnt] = EncodeRIPixel(ptype[RUNcnt], Ra, ptypePrev[RUNcnt]);
ptypeCurX[runLength] = EncodeRIPixel(ptypeCurX[runLength], Ra, ptypePrevX[runLength]);
DecrementRunIndex();
return RUNcnt + 1;
return runLength + 1;
}
template<class TRAITS, class STRATEGY>
int JlsCodec<TRAITS,STRATEGY>::DoRunMode(PIXEL* ptype, const PIXEL* ptypePrev, int ipixel, int cpixelMac, DecoderStrategy*)
int JlsCodec<TRAITS,STRATEGY>::DoRunMode(int ipixelStart, DecoderStrategy*)
{
PIXEL Ra = ptype[ipixel-1];
int ipixelStart = ipixel;
PIXEL Ra = ptypeCur[ipixelStart-1];
int cpixelRun = DecodeRunPixels(Ra, ptypeCur + ipixelStart, _size.cx - ipixelStart);
int ipixelEnd = ipixelStart + cpixelRun;
ipixel = DecodeRunPixels(Ra, ptype, ipixel, cpixelMac);
if (ipixel == cpixelMac)
return ipixel - ipixelStart;
if (ipixelEnd == _size.cx)
return ipixelEnd - ipixelStart;
// run interruption
PIXEL Rb = ptypePrev[ipixel];
ptype[ipixel] = DecodeRIPixel(Ra, Rb);
PIXEL Rb = ptypePrev[ipixelEnd];
ptypeCur[ipixelEnd] = DecodeRIPixel(Ra, Rb);
DecrementRunIndex();
return ipixel - ipixelStart + 1 ;
return ipixelEnd - ipixelStart + 1;
}
template<class TRAITS, class STRATEGY>
void JlsCodec<TRAITS,STRATEGY>::DoLine(typename TRAITS::SAMPLE* ptype, const typename TRAITS::SAMPLE* ptypePrev)
void JlsCodec<TRAITS,STRATEGY>::DoLine(SAMPLE*)
{
int ipixel = 0;
int Rb = ptypePrev[ipixel-1];
@ -686,7 +645,7 @@ void JlsCodec<TRAITS,STRATEGY>::DoLine(typename TRAITS::SAMPLE* ptype, const typ
while(ipixel < _size.cx)
{
int Ra = ptype[ipixel -1];
int Ra = ptypeCur[ipixel -1];
int Rc = Rb;
Rb = Rd;
Rd = ptypePrev[ipixel + 1];
@ -695,13 +654,13 @@ void JlsCodec<TRAITS,STRATEGY>::DoLine(typename TRAITS::SAMPLE* ptype, const typ
if (Qs == 0)
{
ipixel += DoRunMode(ptype, ptypePrev, ipixel, _size.cx, (STRATEGY*)(NULL));
ipixel += DoRunMode(ipixel, (STRATEGY*)(NULL));
Rb = ptypePrev[ipixel-1];
Rd = ptypePrev[ipixel];
}
else
{
ptype[ipixel] = DoRegular(Qs, ptype[ipixel], GetPredictedValue(Ra, Rb, Rc), (STRATEGY*)(NULL));
ptypeCur[ipixel] = DoRegular(Qs, ptypeCur[ipixel], GetPredictedValue(Ra, Rb, Rc), (STRATEGY*)(NULL));
ipixel++;
}
}
@ -710,12 +669,12 @@ void JlsCodec<TRAITS,STRATEGY>::DoLine(typename TRAITS::SAMPLE* ptype, const typ
template<class TRAITS, class STRATEGY>
void JlsCodec<TRAITS,STRATEGY>::DoLine(Triplet* ptype, const Triplet* ptypePrev)
void JlsCodec<TRAITS,STRATEGY>::DoLine(Triplet*)
{
int ipixel = 0;
while(ipixel < _size.cx)
{
Triplet Ra = ptype[ipixel -1];
Triplet Ra = ptypeCur[ipixel -1];
Triplet Rc = ptypePrev[ipixel-1];
Triplet Rb = ptypePrev[ipixel];
Triplet Rd = ptypePrev[ipixel + 1];
@ -726,15 +685,15 @@ void JlsCodec<TRAITS,STRATEGY>::DoLine(Triplet* ptype, const Triplet* ptypePrev)
if (Qs1 == 0 && Qs2 == 0 && Qs3 == 0)
{
ipixel += DoRunMode(ptype, ptypePrev, ipixel, _size.cx, (STRATEGY*)(NULL));
ipixel += DoRunMode(ipixel, (STRATEGY*)(NULL));
}
else
{
Triplet Rx;
Rx.v1 = DoRegular(Qs1, ptype[ipixel].v1, GetPredictedValue(Ra.v1, Rb.v1, Rc.v1), (STRATEGY*)(NULL));
Rx.v2 = DoRegular(Qs2, ptype[ipixel].v2, GetPredictedValue(Ra.v2, Rb.v2, Rc.v2), (STRATEGY*)(NULL));
Rx.v3 = DoRegular(Qs3, ptype[ipixel].v3, GetPredictedValue(Ra.v3, Rb.v3, Rc.v3), (STRATEGY*)(NULL));
ptype[ipixel] = Rx;
Rx.v1 = DoRegular(Qs1, ptypeCur[ipixel].v1, GetPredictedValue(Ra.v1, Rb.v1, Rc.v1), (STRATEGY*)(NULL));
Rx.v2 = DoRegular(Qs2, ptypeCur[ipixel].v2, GetPredictedValue(Ra.v2, Rb.v2, Rc.v2), (STRATEGY*)(NULL));
Rx.v3 = DoRegular(Qs3, ptypeCur[ipixel].v3, GetPredictedValue(Ra.v3, Rb.v3, Rc.v3), (STRATEGY*)(NULL));
ptypeCur[ipixel] = Rx;
ipixel++;
}
@ -743,8 +702,6 @@ void JlsCodec<TRAITS,STRATEGY>::DoLine(Triplet* ptype, const Triplet* ptypePrev)
template<class TRAITS, class STRATEGY>
void JlsCodec<TRAITS,STRATEGY>::DoScan(PIXEL* ptype, BYTE* pbyteCompressed, int cbyteCompressed)
{
@ -756,7 +713,7 @@ void JlsCodec<TRAITS,STRATEGY>::DoScan(PIXEL* ptype, BYTE* pbyteCompressed, int
vectmp.resize((1 + _components) * pixelstride);
std::vector<int> rgRUNindex;
rgRUNindex.resize(3);
rgRUNindex.resize(_components);
for (int iline = 0; iline < _size.cy * _components; ++iline)
{
@ -764,29 +721,19 @@ void JlsCodec<TRAITS,STRATEGY>::DoScan(PIXEL* ptype, BYTE* pbyteCompressed, int
RUNindex = rgRUNindex[icomponent];
int indexPrev = iline % (_components + 1);
int indexCur = (iline + 3) % (_components + 1);
int indexCur = (iline + _components) % (_components + 1);
PIXEL* ptypePrev = &vectmp[1 + indexPrev * pixelstride];
PIXEL* ptypeCur = &vectmp[1 + indexCur * pixelstride];
ptypePrev[_size.cx] = ptypePrev[_size.cx - 1];
ptypeCur[-1] = ptypePrev[0];
PIXEL* ptypeLine = ptype + iline * _size.cx;
if (!STRATEGY::IsDecoding)
{
for (int i = 0; i < _size.cx; ++i)
{
ptypeCur[i] = ptypeLine[i];
}
}
DoLine(ptypeCur, ptypePrev);
if (STRATEGY::IsDecoding)
{
for (int i = 0; i < _size.cx; ++i)
{
CheckedAssign(ptypeLine[i], ptypeCur[i]);
}
}
ptypePrev = &vectmp[1 + indexPrev * pixelstride];
ptypeCur = &vectmp[1 + indexCur * pixelstride];
PIXEL* ptypeLine = ptype + iline * _size.cx;
// initialize edge pixels used for prediction
ptypePrev[_size.cx] = ptypePrev[_size.cx - 1];
ptypeCur[-1] = ptypePrev[0];
OnLineBegin(ptypeCur, ptypeLine, _size.cx);
DoLine((PIXEL*) NULL); // dummy arg for overload resolution
OnLineEnd(ptypeCur, ptypeLine, _size.cx);
rgRUNindex[icomponent] = RUNindex;
}
@ -861,10 +808,10 @@ void JlsCodec<TRAITS,STRATEGY>::InitParams(int t1, int t2, int t3, int nReset)
InitQuantizationLUT();
JlsContext ctxDefault = JlsContext(max(2, (traits.RANGE + 32)/64));
for (UINT Q = 0; Q < sizeof(_rgcontext) / sizeof(_rgcontext[0]); ++Q)
int A = max(2, (traits.RANGE + 32)/64);
for (UINT Q = 0; Q < sizeof(_contexts) / sizeof(_contexts[0]); ++Q)
{
_rgcontext[Q] = ctxDefault;
_contexts[Q] = JlsContext(A);
}
_contextRunmode[0] = CContextRunMode(max(2, (traits.RANGE + 32)/64), 0, nReset);

View File

@ -8,22 +8,6 @@
#include <vector>
#include "util.h"
struct ScanInfo
{
ScanInfo() :
cbit(0),
nnear(0),
ccomp(0),
ilv(ILV_NONE),
size(0,0)
{
}
int cbit;
int nnear;
int ccomp;
interleavemode ilv;
Size size;
};
class JpegSegment;
@ -89,7 +73,7 @@ private:
int _cbyteOffset;
int _cbyteLength;
int _icompLast;
std::vector<JpegSegment*> _rgsegment;
std::vector<JpegSegment*> _segments;
};
@ -119,11 +103,11 @@ public:
int GetBytesRead()
{ return _cbyteOffset; }
const ScanInfo& GetMetadata() const
const JlsParamaters& GetMetadata() const
{ return _info; }
const Presets& GetCustomPreset() const
{ return _presets; }
const JlsCustomParameters& GetCustomPreset() const
{ return _info.custom; }
bool Read(void* pvoid, int cbyteAvailable);
int ReadHeader();
@ -145,8 +129,7 @@ private:
int _cbyteOffset;
int _cbyteLength;
bool _bCompare;
Presets _presets;
ScanInfo _info;
JlsParamaters _info;
};

View File

@ -111,12 +111,6 @@ void TestRoundTrip(const char* strName, const BYTE* rgbyteRaw, Size size, int cb
int cbyteCompressed;
JpegLsEncode(&rgbyteCompressed[0], rgbyteCompressed.size(), &cbyteCompressed, rgbyteRaw, rgbyteOut.size(), &params);
//size_t cbyteCompressed = JpegLsEncode(rgbyteRaw, size, cbit, ccomp, ccomp == 3 ? ILV_SAMPLE : ILV_NONE, &rgbyteCompressed[0], int(rgbyteCompressed.size()), 0);
//CString strDst = strName;
//strDst = strDst + ".jls";
//WriteFile(strDst.GetString(), &rgbyteCompressed[0], cbyteCompressed);
double dwtimeEncodeComplete = getTime();
JpegLsDecode(&rgbyteOut[0], rgbyteOut.size(), &rgbyteCompressed[0], int(cbyteCompressed));
@ -169,27 +163,16 @@ void TestCompliance(const BYTE* pbyteCompressed, int cbyteCompressed, const BYTE
JLS_ERROR error = JpegLsVerifyEncode(&rgbyteRaw[0], cbyteRaw, pbyteCompressed, cbyteCompressed);
ASSERT(error == OK);
// JpegLsEncode(&rgbyteOut[0], rgbyteOut.size(), &cbyteCompressedActual, &rgbyteRaw[0], cbyteRaw, &params);
//BYTE* pbyteOut = &rgbyteOut[0];
//for (UINT i = 0; i < cbyteCompressed; ++i)
//{
// if (pbyteCompressed[i] != pbyteOut[i])
// {
// ASSERT(false);
// break;
// }
//}
}
void TestFile(SZC strName, int ioffs, Size size2, int cbit, int ccomp)
{
std::vector<BYTE> rgbyteNoise;
std::vector<BYTE> rgbyteUncompressed;
ReadFile(strName, &rgbyteNoise, ioffs);
ReadFile(strName, &rgbyteUncompressed, ioffs);
TestRoundTrip(strName, &rgbyteNoise[0], size2, cbit, ccomp);
TestRoundTrip(strName, &rgbyteUncompressed[0], size2, cbit, ccomp);
};
@ -341,6 +324,36 @@ void TestSampleAnnexH3()
// TestJls(vecRaw, size, 8, 1, ILV_NONE, rgbyteComp, sizeof(rgbyteComp), false);
}
void TestSmallBuffer()
{
std::vector<BYTE> rgbyteCompressed;
ReadFile("..\\test\\lena8b.jls", &rgbyteCompressed, 0);
std::vector<BYTE> rgbyteOut;
rgbyteOut.resize(512 * 511);
JLS_ERROR error = JpegLsDecode(&rgbyteOut[0], rgbyteOut.size(), &rgbyteCompressed[0], int(rgbyteCompressed.size()));
ASSERT(error == JLS_ERROR::UncompressedBufferTooSmall);
}
void TestDamagedBitStream()
{
std::vector<BYTE> rgbyteCompressed;
ReadFile("..\\test\\lena8b.jls", &rgbyteCompressed, 0);
rgbyteCompressed.resize(900);
rgbyteCompressed.resize(40000,3);
std::vector<BYTE> rgbyteOut;
rgbyteOut.resize(512 * 512);
JLS_ERROR error = JpegLsDecode(&rgbyteOut[0], rgbyteOut.size(), &rgbyteCompressed[0], int(rgbyteCompressed.size()));
ASSERT(error == JLS_ERROR::InvalidCompressedData);
}
void TestConformance()
{
@ -382,9 +395,6 @@ void TestConformance()
// additional, Lena compressed with other codec (UBC?), vfy with CharLS
DecompressFile("..\\test\\lena8b.jls", "..\\test\\lena8b.raw",0);
}
@ -392,6 +402,8 @@ void TestConformance()
int _tmain(int argc, _TCHAR* argv[])
{
TestDamagedBitStream();
TestSmallBuffer();
TestConformance();
TestSampleAnnexH3();
TestTraits16bit();