Refactorings for better color support in the future. (BGR format and 16/12 bits per sample color).

This commit is contained in:
jdv_cp 2009-04-28 14:42:45 -07:00
parent 3aaa171d3e
commit 24a29b1345
10 changed files with 208 additions and 170 deletions

View File

@ -4,18 +4,30 @@
#ifndef CHARLS_COLORTRANSFORM
#define CHARLS_COLORTRANSFORM
struct TransformNone
struct TransformNoneImpl
{
inlinehint static Triplet Apply(int v1, int v2, int v3)
{ return Triplet(v1, v2, v3); }
inlinehint static Triplet<BYTE> Apply(int v1, int v2, int v3)
{ return Triplet<BYTE>(v1, v2, v3); }
};
struct TransformRgbToHp1
struct TransformNone : public TransformNoneImpl
{
inlinehint static Triplet Apply(int R, int G, int B)
typedef struct TransformNoneImpl INVERSE;
};
struct TransformHp1ToRgb
{
inlinehint static Triplet<BYTE> Apply(int v1, int v2, int v3)
{ return Triplet<BYTE>(v1 + v2 - 0x80, v2, v3 + v2 - 0x80); }
};
struct TransformHp1
{
typedef struct TransformHp1ToRgb INVERSE;
inlinehint static Triplet<BYTE> Apply(int R, int G, int B)
{
Triplet hp1;
Triplet<BYTE> hp1;
hp1.v2 = BYTE(G);
hp1.v1 = BYTE(R - G + 0x80);
hp1.v3 = BYTE(B - G + 0x80);
@ -23,32 +35,26 @@ struct TransformRgbToHp1
}
};
struct TransformHp1ToRgb
struct TransformHp2
{
inlinehint static Triplet Apply(int v1, int v2, int v3)
{ return Triplet(v1 + v2 - 0x80, v2, v3 + v2 - 0x80); }
};
struct TransformRgbToHp2
{
inlinehint static Triplet Apply(int R, int G, int B)
typedef struct TransformHp2ToRgb INVERSE;
inlinehint static Triplet<BYTE> Apply(int R, int G, int B)
{
//Triplet hp2;
//Triplet<BYTE> hp2;
//hp2.v1 = BYTE(R - G + 0x80);
//hp2.v2 = BYTE(G);
//hp2.v3 = BYTE(B - ((R+G )>>1) - 0x80);
//return hp2;
return Triplet(R - G + 0x80, G, B - ((R+G )>>1) - 0x80);
return Triplet<BYTE>(R - G + 0x80, G, B - ((R+G )>>1) - 0x80);
}
};
struct TransformHp2ToRgb
{
inlinehint static Triplet Apply(int v1, int v2, int v3)
inlinehint static Triplet<BYTE> Apply(int v1, int v2, int v3)
{
Triplet rgb;
Triplet<BYTE> rgb;
rgb.R = BYTE(v1 + v2 - 0x80); // new R
rgb.G = BYTE(v2); // new G
rgb.B = BYTE(v3 + ((rgb.R + rgb.G) >> 1) - 0x80); // new B
@ -56,15 +62,17 @@ struct TransformHp2ToRgb
}
};
struct TransformRgbToHp3
struct TransformHp3
{
inlinehint static Triplet Apply(int R, int G, int B)
typedef struct TransformHp3ToRgb INVERSE;
inlinehint static Triplet<BYTE> Apply(int R, int G, int B)
{
Triplet hp3;
Triplet<BYTE> hp3;
hp3.v2 = BYTE(B - G + 0x80);
hp3.v3 = BYTE(R - G + 0x80);
hp3.v1 = BYTE(G + ((hp3.v2 + hp3.v3 )>>2) - 0x40);
hp3.v1 = BYTE(G + ((hp3.v2 + hp3.v3)>>2)) - 0x40;
return hp3;
}
};
@ -72,10 +80,10 @@ struct TransformRgbToHp3
struct TransformHp3ToRgb
{
inlinehint static Triplet Apply(int v1, int v2, int v3)
inlinehint static Triplet<BYTE> Apply(int v1, int v2, int v3)
{
int G = v1 - ((v3 + v2)>>2)+0x40;
Triplet rgb;
int G = v1 - ((v3 + v2)>>2) + 0x40;
Triplet<BYTE> rgb;
rgb.R = BYTE(v3 + G - 0x80); // new R
rgb.G = BYTE(G); // new G
rgb.B = BYTE(v2 + G - 0x80); // new B
@ -85,7 +93,7 @@ struct TransformHp3ToRgb
template<class TRANSFORM>
void TransformLine(Triplet* pDest, const Triplet* pSrc, int pixelCount, const TRANSFORM&)
void TransformLine(Triplet<BYTE>* pDest, const Triplet<BYTE>* pSrc, int pixelCount, const TRANSFORM&)
{
for (int i = 0; i < pixelCount; ++i)
{
@ -98,7 +106,7 @@ template<class TRANSFORM>
void TransformLineToTriplet(const BYTE* ptypeInput, LONG pixelStrideIn, BYTE* pbyteBuffer, LONG pixelStride, const TRANSFORM&)
{
int cpixel = MIN(pixelStride, pixelStrideIn);
Triplet* ptypeBuffer = (Triplet*)pbyteBuffer;
Triplet<BYTE>* ptypeBuffer = (Triplet<BYTE>*)pbyteBuffer;
for (int x = 0; x < cpixel; ++x)
{
@ -110,12 +118,12 @@ template<class TRANSFORM>
void TransformTripletToLine(const BYTE* pbyteInput, LONG pixelStrideIn, BYTE* ptypeBuffer, LONG pixelStride, const TRANSFORM&)
{
int cpixel = MIN(pixelStride, pixelStrideIn);
const Triplet* ptypeBufferIn = (Triplet*)pbyteInput;
const Triplet<BYTE>* ptypeBufferIn = (Triplet<BYTE>*)pbyteInput;
for (int x = 0; x < cpixel; ++x)
{
Triplet color = ptypeBufferIn[x];
Triplet colorTranformed = TRANSFORM::Apply(color.v1, color.v2, color.v3);
Triplet<BYTE> color = ptypeBufferIn[x];
Triplet<BYTE> colorTranformed = TRANSFORM::Apply(color.v1, color.v2, color.v3);
ptypeBuffer[x] = colorTranformed.v1;
ptypeBuffer[x + pixelStride] = colorTranformed.v2;
@ -124,4 +132,86 @@ void TransformTripletToLine(const BYTE* pbyteInput, LONG pixelStrideIn, BYTE* pt
}
class PostProcessLine
{
public:
virtual ~PostProcessLine() {}
virtual void NewLineDecoded(const void* pSrc, int pixelCount, int byteStride) = 0;
virtual void NewLineRequested(void* pSrc, int pixelCount, int byteStride) = 0;
};
template<class TRANSFORM>
class PostProcessTransformed : public PostProcessLine
{
PostProcessTransformed(const PostProcessTransformed&) {}
public:
PostProcessTransformed(void* pbyteOutput, const JlsParamaters& info) :
_pbyteOutput((BYTE*)pbyteOutput),
_info(info)
{
ASSERT(_info.components == 3);
}
void NewLineRequested(void* pDst, int pixelCount, int byteStride)
{
if (_info.ilv == ILV_SAMPLE)
{
TransformLine((Triplet<BYTE>*)pDst, (const Triplet<BYTE>*)_pbyteOutput, pixelCount, TRANSFORM());
}
else
{
TransformTripletToLine((const BYTE*)_pbyteOutput, pixelCount, (BYTE*)pDst, byteStride, TRANSFORM());
}
_pbyteOutput += 3*pixelCount;
}
void NewLineDecoded(const void* pSrc, int pixelCount, int byteStride)
{
if (_info.ilv == ILV_SAMPLE)
{
TransformLine((Triplet<BYTE>*)_pbyteOutput, (const Triplet<BYTE>*)pSrc, pixelCount, TRANSFORM::INVERSE());
}
else
{
TransformLineToTriplet((const BYTE*)pSrc, byteStride, _pbyteOutput, pixelCount, TRANSFORM::INVERSE());
}
_pbyteOutput += 3* pixelCount;
}
private:
BYTE* _pbyteOutput;
const JlsParamaters& _info;
};
class PostProcesSingleComponent : public PostProcessLine
{
public:
PostProcesSingleComponent(void* pbyteOutput, const JlsParamaters& info, int bytesPerPixel) :
_pbyteOutput((BYTE*)pbyteOutput),
_bytesPerPixel(bytesPerPixel),
_info(info)
{
}
void NewLineRequested(void* pDst, int pixelCount, int byteStride)
{
::memcpy(pDst, _pbyteOutput, pixelCount * _bytesPerPixel);
_pbyteOutput += pixelCount * _bytesPerPixel;
}
void NewLineDecoded(const void* pSrc, int pixelCount, int byteStride)
{
::memcpy(_pbyteOutput, pSrc, pixelCount * _bytesPerPixel);
_pbyteOutput += pixelCount * _bytesPerPixel;
}
private:
BYTE* _pbyteOutput;
int _bytesPerPixel;
const JlsParamaters& _info;
};
#endif

View File

@ -16,18 +16,22 @@ public:
_readCache(0),
_validBits(0),
_pbyteCompressed(0),
_info(info)
_info(info),
_postProcessLine(0)
{
if (_info.ilv != ILV_LINE)
{
_info.components = 1;
}
}
enum { IsDecoding = 1};
virtual ~DecoderStrategy()
{}
{
delete _postProcessLine;
}
virtual void SetPresets(const JlsCustomParameters& presets) = 0;
virtual size_t DecodeScan(void* pvoidOut, const Size& size, const void* pvoidIn, size_t cbyte, bool bCheck) = 0;
@ -49,63 +53,13 @@ public:
}
void OnLineBegin(int iline, LONG cpixel, void* ptypeBuffer, LONG pixelStride)
void OnLineBegin(LONG cpixel, void* ptypeBuffer, LONG pixelStride)
{}
void OnLineEnd(int iline, LONG cpixel, const Triplet* ptypeBuffer, LONG pixelStride)
void OnLineEnd(LONG cpixel, const void* ptypeBuffer, LONG pixelStride)
{
Triplet* ptypeUnc = ((Triplet*)_ptypeUncompressed) + iline * cpixel;
switch(_info.colorTransform)
{
case COLORXFORM_NONE : return TransformLine(ptypeUnc, ptypeBuffer, cpixel, TransformNone());
case COLORXFORM_HP1 : return TransformLine(ptypeUnc, ptypeBuffer, cpixel, TransformHp1ToRgb());
case COLORXFORM_HP2 : return TransformLine(ptypeUnc, ptypeBuffer, cpixel, TransformHp2ToRgb());
case COLORXFORM_HP3 : return TransformLine(ptypeUnc, ptypeBuffer, cpixel, TransformHp3ToRgb());
}
}
template <class T>
void OnLineEnd(int iline, LONG cpixel, const T* ptypeBuffer, LONG pixelStride)
{
for (int icomponent = 0; icomponent < _info.components; ++icomponent)
{
T* ptypeUnc = ((T*)_ptypeUncompressed) + (iline *_info.components + icomponent)* cpixel;
#ifdef _DEBUG
for (LONG i = 0; i < cpixel; ++i)
{
//CheckedAssign(ptypeLine[i], ptypeCur[i]);
ptypeUnc[i] = ptypeBuffer[i + icomponent*pixelStride];
}
#else
memcpy(ptypeUnc, ptypeBuffer+ icomponent*pixelStride, cpixel * sizeof(T) );
#endif
}
}
void OnLineEnd(int iline, LONG cpixel, const BYTE* ptypeBuffer, LONG pixelStride)
{
BYTE* ptypeUnc = ((BYTE*)(_ptypeUncompressed));
if (_info.components == 1)
{
memcpy(ptypeUnc + cpixel * iline, ptypeBuffer, cpixel * sizeof(BYTE));
return;
}
ptypeUnc += iline * _info.components * cpixel;
switch(_info.colorTransform)
{
case COLORXFORM_NONE : return TransformLineToTriplet(ptypeBuffer, pixelStride, ptypeUnc, cpixel, TransformNone());
case COLORXFORM_HP1 : return TransformLineToTriplet(ptypeBuffer, pixelStride, ptypeUnc, cpixel, TransformHp1ToRgb());
case COLORXFORM_HP2 : return TransformLineToTriplet(ptypeBuffer, pixelStride, ptypeUnc, cpixel, TransformHp2ToRgb());
case COLORXFORM_HP3 : return TransformLineToTriplet(ptypeBuffer, pixelStride, ptypeUnc, cpixel, TransformHp3ToRgb());
}
_postProcessLine->NewLineDecoded(ptypeBuffer, cpixel, pixelStride);
}
typedef size_t bufType;
@ -142,8 +96,8 @@ public:
return;
}
bufType valnew = *_pbyteCompressed;
_readCache |= bufType(_pbyteCompressed[0]) << (bufferbits - 8 - _validBits);
bufType valnew = _pbyteCompressed[0];
_readCache |= valnew << (bufferbits - 8 - _validBits);
_pbyteCompressed += 1;
_validBits += 8;
@ -155,6 +109,7 @@ public:
while (_validBits < bufferbits - 8);
_pbyteNextFF = FindNextFF();
return;
}
@ -163,9 +118,12 @@ public:
{
BYTE* pbyteNextFF =_pbyteCompressed;
while (pbyteNextFF < _pbyteCompressedEnd && *pbyteNextFF != 0xFF)
while (pbyteNextFF < _pbyteCompressedEnd)
{
pbyteNextFF++;
if (*pbyteNextFF == 0xFF)
break;
pbyteNextFF++;
}
return pbyteNextFF - (sizeof(bufType)-1);
@ -279,6 +237,7 @@ public:
protected:
JlsParamaters _info;
void* _ptypeUncompressed;
PostProcessLine* _postProcessLine;
private:
// decoding

View File

@ -61,7 +61,7 @@ public:
inlinehint bool IsNear(LONG lhs, LONG rhs) const
{ return abs(lhs-rhs) <=NEAR; }
bool IsNear(Triplet lhs, Triplet rhs) const
bool IsNear(Triplet<BYTE> lhs, Triplet<BYTE> rhs) const
{
return abs(lhs.v1-rhs.v1) <=NEAR && abs(lhs.v2-rhs.v2) <=NEAR && abs(lhs.v3-rhs.v3) <=NEAR;
}

View File

@ -18,61 +18,28 @@ public:
bitpos(0),
_bFFWritten(false),
_cbyteWritten(0),
_info(info)
_info(info),
_postProcessLine(0)
{
if (_info.ilv != ILV_LINE)
{
_info.components = 1;
}
};
};
virtual ~EncoderStrategy()
{}
{
delete _postProcessLine;
}
LONG PeekByte();
void OnLineBegin(int iline, LONG cpixel, Triplet* ptypeBuffer, LONG /*pixelStride*/)
void OnLineBegin(LONG cpixel, void* ptypeBuffer, LONG pixelStride)
{
Triplet* ptypeInput = ((Triplet*)(_ptypeUncompressed));
switch(_info.colorTransform)
{
case COLORXFORM_NONE : return TransformLine(ptypeBuffer, ptypeInput + cpixel*iline, cpixel, TransformNone());
case COLORXFORM_HP1 : return TransformLine(ptypeBuffer, ptypeInput + cpixel*iline, cpixel, TransformRgbToHp1());
case COLORXFORM_HP2 : return TransformLine(ptypeBuffer, ptypeInput + cpixel*iline, cpixel, TransformRgbToHp2());
case COLORXFORM_HP3 : return TransformLine(ptypeBuffer, ptypeInput + cpixel*iline, cpixel, TransformRgbToHp3());
}
_postProcessLine->NewLineRequested(ptypeBuffer, cpixel, pixelStride);
}
void OnLineBegin(int iline, LONG cpixel, USHORT* ptypeBuffer, LONG /*pixelStride*/)
{
USHORT* ptypeInput = ((USHORT*)(_ptypeUncompressed));
memcpy(ptypeBuffer, ptypeInput + cpixel*iline, cpixel * sizeof(USHORT));
}
void OnLineBegin(int iline, LONG cpixel, BYTE* ptypeBuffer, LONG pixelStride)
{
BYTE* ptypeUnc = ((BYTE*)(_ptypeUncompressed));
if (_info.components == 1)
{
memcpy(ptypeBuffer, ptypeUnc + cpixel * iline, cpixel * sizeof(BYTE));
return;
}
ptypeUnc += iline * _info.components * cpixel;
switch(_info.colorTransform)
{
case COLORXFORM_NONE : return TransformTripletToLine(ptypeUnc, cpixel, ptypeBuffer, pixelStride, TransformNone());
case COLORXFORM_HP1 : return TransformTripletToLine(ptypeUnc, cpixel, ptypeBuffer, pixelStride, TransformRgbToHp1());
case COLORXFORM_HP2 : return TransformTripletToLine(ptypeUnc, cpixel, ptypeBuffer, pixelStride, TransformRgbToHp2());
case COLORXFORM_HP3 : return TransformTripletToLine(ptypeUnc, cpixel, ptypeBuffer, pixelStride, TransformRgbToHp3());
}
}
void OnLineEnd(int iline, LONG cpixel, void* ptypeBuffer, LONG pixelStride) {};
void OnLineEnd(LONG cpixel, void* ptypeBuffer, LONG pixelStride) {};
virtual void SetPresets(const JlsCustomParameters& presets) = 0;
@ -167,7 +134,7 @@ protected:
protected:
JlsParamaters _info;
const void* _ptypeUncompressed;
PostProcessLine* _postProcessLine;
private:
unsigned int valcurrent;

View File

@ -106,10 +106,10 @@ STRATEGY* JlsCodecFactory<STRATEGY>::GetCodecImpl(const JlsParamaters& _info)
if (_info.components == 3 && _info.bitspersample == 8)
{
if (_info.allowedlossyerror == 0)
return new JlsCodec<LosslessTraitsT<Triplet,8>, STRATEGY>(LosslessTraitsT<Triplet,8>(), _info);
return new JlsCodec<LosslessTraitsT<Triplet<BYTE>,8>, STRATEGY>(LosslessTraitsT<Triplet<BYTE>,8>(), _info);
DefaultTraitsT<BYTE,Triplet> traits((1 << _info.bitspersample) - 1, _info.allowedlossyerror);
return new JlsCodec<DefaultTraitsT<BYTE,Triplet>, STRATEGY>(traits, _info);
DefaultTraitsT<BYTE,Triplet<BYTE>> traits((1 << _info.bitspersample) - 1, _info.allowedlossyerror);
return new JlsCodec<DefaultTraitsT<BYTE,Triplet<BYTE>>, STRATEGY>(traits, _info);
}
return NULL;

View File

@ -100,10 +100,10 @@ struct LosslessTraitsT<USHORT,16> : public LosslessTraitsImplT<USHORT,16>
template<>
struct LosslessTraitsT<Triplet,8> : public LosslessTraitsImplT<BYTE,8>
struct LosslessTraitsT<Triplet<BYTE>,8> : public LosslessTraitsImplT<BYTE,8>
{
// enum { ccomponent = 3 };
typedef Triplet PIXEL;
typedef Triplet<BYTE> PIXEL;
static inlinehint bool IsNear(LONG lhs, LONG rhs)
{ return lhs == rhs; }

63
scan.h
View File

@ -160,7 +160,7 @@ public:
}
void InitQuantizationLUT();
LONG DecodeValue(LONG k, LONG limit, LONG qbpp);
inlinehint void EncodeMappedValue(LONG k, LONG mappederval, LONG limit);
@ -170,14 +170,14 @@ public:
{ RUNindex = MAX(0,RUNindex - 1); }
LONG DecodeRIError(CContextRunMode& ctx);
Triplet DecodeRIPixel(Triplet Ra, Triplet Rb);
PIXEL DecodeRIPixel(LONG Ra, LONG Rb);
Triplet<SAMPLE> DecodeRIPixel(Triplet<SAMPLE> Ra, Triplet<SAMPLE> Rb);
SAMPLE DecodeRIPixel(LONG Ra, LONG Rb);
LONG DecodeRunPixels(PIXEL Ra, PIXEL* ptype, LONG cpixelMac);
LONG DoRunMode(LONG ipixel, DecoderStrategy*);
void EncodeRIError(CContextRunMode& ctx, LONG Errval);
SAMPLE EncodeRIPixel(LONG x, LONG Ra, LONG Rb);
Triplet EncodeRIPixel(Triplet x, Triplet Ra, Triplet Rb);
Triplet<SAMPLE> EncodeRIPixel(Triplet<SAMPLE> x, Triplet<SAMPLE> Ra, Triplet<SAMPLE> Rb);
void EncodeRunPixels(LONG runLength, bool bEndofline);
LONG DoRunMode(LONG ipixel, EncoderStrategy*);
@ -185,11 +185,12 @@ public:
inlinehint SAMPLE DoRegular(LONG Qs, LONG x, LONG pred, EncoderStrategy*);
void DoLine(SAMPLE* pdummy);
void DoLine(Triplet* pdummy);
void DoLine(Triplet<SAMPLE>* pdummy);
void DoScan(BYTE* pbyteCompressed, size_t cbyteCompressed);
public:
void InitDefault();
void InitPostProcess(void* pvoidOut);
void InitDefault();
void InitParams(LONG t1, LONG t2, LONG t3, LONG nReset);
size_t EncodeScan(const void* pvoid, const Size& size, void* pvoidOut, size_t cbyte, void* pvoidCompare);
@ -500,15 +501,14 @@ void JlsCodec<TRAITS,STRATEGY>::EncodeRIError(CContextRunMode& ctx, LONG Errval)
}
template<class TRAITS, class STRATEGY>
Triplet JlsCodec<TRAITS,STRATEGY>::DecodeRIPixel(Triplet Ra, Triplet Rb)
Triplet<typename TRAITS::SAMPLE> JlsCodec<TRAITS,STRATEGY>::DecodeRIPixel(Triplet<SAMPLE> Ra, Triplet<SAMPLE> Rb)
{
LONG Errval1 = DecodeRIError(_contextRunmode[0]);
LONG Errval2 = DecodeRIError(_contextRunmode[0]);
LONG Errval3 = DecodeRIError(_contextRunmode[0]);
return Triplet(traits.ComputeReconstructedSample(Rb.v1, Errval1 * Sign(Rb.v1 - Ra.v1)),
return Triplet<SAMPLE>(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)));
}
@ -516,7 +516,7 @@ Triplet JlsCodec<TRAITS,STRATEGY>::DecodeRIPixel(Triplet Ra, Triplet Rb)
template<class TRAITS, class STRATEGY>
Triplet JlsCodec<TRAITS,STRATEGY>::EncodeRIPixel(Triplet x, Triplet Ra, Triplet Rb)
Triplet<typename TRAITS::SAMPLE> JlsCodec<TRAITS,STRATEGY>::EncodeRIPixel(Triplet<SAMPLE> x, Triplet<SAMPLE> Ra, Triplet<SAMPLE> Rb)
{
const LONG RItype = 0;
@ -530,7 +530,7 @@ Triplet JlsCodec<TRAITS,STRATEGY>::EncodeRIPixel(Triplet x, Triplet Ra, Triplet
EncodeRIError(_contextRunmode[RItype], errval3);
return Triplet(traits.ComputeReconstructedSample(Rb.v1, errval1 * Sign(Rb.v1 - Ra.v1)),
return Triplet<SAMPLE>(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)));
}
@ -538,7 +538,7 @@ Triplet JlsCodec<TRAITS,STRATEGY>::EncodeRIPixel(Triplet x, Triplet Ra, Triplet
template<class TRAITS, class STRATEGY>
typename TRAITS::PIXEL JlsCodec<TRAITS,STRATEGY>::DecodeRIPixel(LONG Ra, LONG Rb)
typename TRAITS::SAMPLE JlsCodec<TRAITS,STRATEGY>::DecodeRIPixel(LONG Ra, LONG Rb)
{
if (abs(Ra - Rb) <= traits.NEAR)
{
@ -621,7 +621,7 @@ LONG JlsCodec<TRAITS,STRATEGY>::DoRunMode(LONG ipixelStart, DecoderStrategy*)
// run interruption
PIXEL Rb = ptypePrev[ipixelEnd];
ptypeCur[ipixelEnd] = DecodeRIPixel(Ra, Rb);
ptypeCur[ipixelEnd] = DecodeRIPixel(Ra, Rb);
DecrementRunIndex();
return ipixelEnd - ipixelStart + 1;
}
@ -661,15 +661,15 @@ void JlsCodec<TRAITS,STRATEGY>::DoLine(SAMPLE*)
template<class TRAITS, class STRATEGY>
void JlsCodec<TRAITS,STRATEGY>::DoLine(Triplet*)
void JlsCodec<TRAITS,STRATEGY>::DoLine(Triplet<SAMPLE>*)
{
LONG ipixel = 0;
while(ipixel < _size.cx)
{
Triplet Ra = ptypeCur[ipixel -1];
Triplet Rc = ptypePrev[ipixel-1];
Triplet Rb = ptypePrev[ipixel];
Triplet Rd = ptypePrev[ipixel + 1];
Triplet<SAMPLE> Ra = ptypeCur[ipixel -1];
Triplet<SAMPLE> Rc = ptypePrev[ipixel-1];
Triplet<SAMPLE> Rb = ptypePrev[ipixel];
Triplet<SAMPLE> Rd = ptypePrev[ipixel + 1];
LONG Qs1 = ComputeContextID(QuantizeGratient(Rd.v1 - Rb.v1), QuantizeGratient(Rb.v1 - Rc.v1), QuantizeGratient(Rc.v1 - Ra.v1));
LONG Qs2 = ComputeContextID(QuantizeGratient(Rd.v2 - Rb.v2), QuantizeGratient(Rb.v2 - Rc.v2), QuantizeGratient(Rc.v2 - Ra.v2));
@ -681,7 +681,7 @@ void JlsCodec<TRAITS,STRATEGY>::DoLine(Triplet*)
}
else
{
Triplet Rx;
Triplet<SAMPLE> 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));
@ -717,7 +717,7 @@ void JlsCodec<TRAITS,STRATEGY>::DoScan(BYTE* pbyteCompressed, size_t cbyteCompre
std::swap(ptypePrev, ptypeCur);
}
STRATEGY::OnLineBegin(iline, _size.cx, ptypeCur, pixelstride);
STRATEGY::OnLineBegin(_size.cx, ptypeCur, pixelstride);
for (int component = 0; component < components; ++component)
{
@ -734,7 +734,7 @@ void JlsCodec<TRAITS,STRATEGY>::DoScan(BYTE* pbyteCompressed, size_t cbyteCompre
ptypeCur += pixelstride;
}
STRATEGY::OnLineEnd(iline, _size.cx, ptypeCur - (components * pixelstride), pixelstride);
STRATEGY::OnLineEnd(_size.cx, ptypeCur - (components * pixelstride), pixelstride);
}
}
@ -743,6 +743,7 @@ void JlsCodec<TRAITS,STRATEGY>::DoScan(BYTE* pbyteCompressed, size_t cbyteCompre
template<class TRAITS, class STRATEGY>
size_t JlsCodec<TRAITS,STRATEGY>::EncodeScan(const void* pvoid, const Size& size, void* pvoidOut, size_t cbyte, void* pvoidCompare)
{
InitPostProcess(const_cast<void*>(pvoid));
_size = size;
BYTE* pbyteCompressed = static_cast<BYTE*>(pvoidOut);
@ -765,11 +766,31 @@ size_t JlsCodec<TRAITS,STRATEGY>::EncodeScan(const void* pvoid, const Size& size
}
template<class TRAITS, class STRATEGY>
void JlsCodec<TRAITS,STRATEGY>::InitPostProcess(void* pvoidOut)
{
if (STRATEGY::_info.components == 1)
{
STRATEGY::_postProcessLine = new PostProcesSingleComponent(pvoidOut, STRATEGY::_info, sizeof(TRAITS::PIXEL));
}
else
{
switch(STRATEGY::_info.colorTransform)
{
case COLORXFORM_NONE: STRATEGY::_postProcessLine = new PostProcessTransformed<TransformNone>(pvoidOut, STRATEGY::_info); break;
case COLORXFORM_HP1 : STRATEGY::_postProcessLine = new PostProcessTransformed<TransformHp1>(pvoidOut, STRATEGY::_info); break;
case COLORXFORM_HP2 : STRATEGY::_postProcessLine = new PostProcessTransformed<TransformHp2>(pvoidOut, STRATEGY::_info); break;
case COLORXFORM_HP3 : STRATEGY::_postProcessLine = new PostProcessTransformed<TransformHp3>(pvoidOut, STRATEGY::_info); break;
}
}
}
template<class TRAITS, class STRATEGY>
size_t JlsCodec<TRAITS,STRATEGY>::DecodeScan(void* pvoidOut, const Size& size, const void* pvoidIn, size_t cbyte, bool bCompare)
{
InitPostProcess(pvoidOut);
PIXEL* ptypeOut = static_cast<PIXEL*>(pvoidOut);
BYTE* pbyteCompressed = const_cast<BYTE*>(static_cast<const BYTE*>(pvoidIn));
_bCompare = bCompare;

View File

@ -9,6 +9,7 @@
#if defined(WIN32)
#define CHARLS_IMEXPORT __declspec(dllexport)
#pragma warning (disable:4100)
#pragma warning (disable:4512)
#endif
#include "util.h"

View File

@ -9,7 +9,6 @@
#include <vector>
#include "util.h"
class JpegSegment;

13
util.h
View File

@ -104,7 +104,8 @@ inline LONG BitWiseSign(LONG i)
#pragma pack(push, 1)
struct Triplet
template<class SAMPLE>
struct Triplet
{
Triplet() :
v1(0),
@ -113,9 +114,9 @@ struct Triplet
{};
Triplet(LONG x1, LONG x2, LONG x3) :
v1((BYTE)x1),
v2((BYTE)x2),
v3((BYTE)x3)
v1((SAMPLE)x1),
v2((SAMPLE)x2),
v3((SAMPLE)x3)
{};
union
@ -141,10 +142,10 @@ struct Triplet
#include "interface.h"
inline bool operator==(const Triplet& lhs, const Triplet& rhs)
inline bool operator==(const Triplet<BYTE>& lhs, const Triplet<BYTE>& rhs)
{ return lhs.v1 == rhs.v1 && lhs.v2 == rhs.v2 && lhs.v3 == rhs.v3; }
inline bool operator!=(const Triplet& lhs, const Triplet& rhs)
inline bool operator!=(const Triplet<BYTE>& lhs, const Triplet<BYTE>& rhs)
{ return !(lhs == rhs); }
#endif