Index: third_party/lzma_sdk/C/XzEnc.c |
=================================================================== |
--- third_party/lzma_sdk/C/XzEnc.c (revision 0) |
+++ third_party/lzma_sdk/C/XzEnc.c (revision 0) |
@@ -0,0 +1,497 @@ |
+/* XzEnc.c -- Xz Encode |
+2009-06-04 : Igor Pavlov : Public domain */ |
+ |
+#include <stdlib.h> |
+#include <string.h> |
+ |
+#include "7zCrc.h" |
+#include "Alloc.h" |
+#include "Bra.h" |
+#include "CpuArch.h" |
+#ifdef USE_SUBBLOCK |
+#include "SbEnc.h" |
+#endif |
+ |
+#include "XzEnc.h" |
+ |
+static void *SzBigAlloc(void *p, size_t size) { p = p; return BigAlloc(size); } |
+static void SzBigFree(void *p, void *address) { p = p; BigFree(address); } |
+static ISzAlloc g_BigAlloc = { SzBigAlloc, SzBigFree }; |
+ |
+static void *SzAlloc(void *p, size_t size) { p = p; return MyAlloc(size); } |
+static void SzFree(void *p, void *address) { p = p; MyFree(address); } |
+static ISzAlloc g_Alloc = { SzAlloc, SzFree }; |
+ |
+#define XzBlock_ClearFlags(p) (p)->flags = 0; |
+#define XzBlock_SetNumFilters(p, n) (p)->flags |= ((n) - 1); |
+#define XzBlock_SetHasPackSize(p) (p)->flags |= XZ_BF_PACK_SIZE; |
+#define XzBlock_SetHasUnpackSize(p) (p)->flags |= XZ_BF_UNPACK_SIZE; |
+ |
+static SRes WriteBytes(ISeqOutStream *s, const void *buf, UInt32 size) |
+{ |
+ return (s->Write(s, buf, size) == size) ? SZ_OK : SZ_ERROR_WRITE; |
+} |
+ |
+static SRes WriteBytesAndCrc(ISeqOutStream *s, const void *buf, UInt32 size, UInt32 *crc) |
+{ |
+ *crc = CrcUpdate(*crc, buf, size); |
+ return WriteBytes(s, buf, size); |
+} |
+ |
+SRes Xz_WriteHeader(CXzStreamFlags f, ISeqOutStream *s) |
+{ |
+ UInt32 crc; |
+ Byte header[XZ_STREAM_HEADER_SIZE]; |
+ memcpy(header, XZ_SIG, XZ_SIG_SIZE); |
+ header[XZ_SIG_SIZE] = (Byte)(f >> 8); |
+ header[XZ_SIG_SIZE + 1] = (Byte)(f & 0xFF); |
+ crc = CrcCalc(header + XZ_SIG_SIZE, XZ_STREAM_FLAGS_SIZE); |
+ SetUi32(header + XZ_SIG_SIZE + XZ_STREAM_FLAGS_SIZE, crc); |
+ return WriteBytes(s, header, XZ_STREAM_HEADER_SIZE); |
+} |
+ |
+SRes XzBlock_WriteHeader(const CXzBlock *p, ISeqOutStream *s) |
+{ |
+ Byte header[XZ_BLOCK_HEADER_SIZE_MAX]; |
+ |
+ unsigned pos = 1; |
+ int numFilters, i; |
+ header[pos++] = p->flags; |
+ |
+ if (XzBlock_HasPackSize(p)) pos += Xz_WriteVarInt(header + pos, p->packSize); |
+ if (XzBlock_HasUnpackSize(p)) pos += Xz_WriteVarInt(header + pos, p->unpackSize); |
+ numFilters = XzBlock_GetNumFilters(p); |
+ for (i = 0; i < numFilters; i++) |
+ { |
+ const CXzFilter *f = &p->filters[i]; |
+ pos += Xz_WriteVarInt(header + pos, f->id); |
+ pos += Xz_WriteVarInt(header + pos, f->propsSize); |
+ memcpy(header + pos, f->props, f->propsSize); |
+ pos += f->propsSize; |
+ } |
+ while((pos & 3) != 0) |
+ header[pos++] = 0; |
+ header[0] = (Byte)(pos >> 2); |
+ SetUi32(header + pos, CrcCalc(header, pos)); |
+ return WriteBytes(s, header, pos + 4); |
+} |
+ |
+SRes Xz_WriteFooter(CXzStream *p, ISeqOutStream *s) |
+{ |
+ Byte buf[32]; |
+ UInt64 globalPos; |
+ { |
+ UInt32 crc = CRC_INIT_VAL; |
+ unsigned pos = 1 + Xz_WriteVarInt(buf + 1, p->numBlocks); |
+ size_t i; |
+ |
+ globalPos = pos; |
+ buf[0] = 0; |
+ RINOK(WriteBytesAndCrc(s, buf, pos, &crc)); |
+ for (i = 0; i < p->numBlocks; i++) |
+ { |
+ const CXzBlockSizes *block = &p->blocks[i]; |
+ pos = Xz_WriteVarInt(buf, block->totalSize); |
+ pos += Xz_WriteVarInt(buf + pos, block->unpackSize); |
+ globalPos += pos; |
+ RINOK(WriteBytesAndCrc(s, buf, pos, &crc)); |
+ } |
+ pos = ((unsigned)globalPos & 3); |
+ if (pos != 0) |
+ { |
+ buf[0] = buf[1] = buf[2] = 0; |
+ RINOK(WriteBytesAndCrc(s, buf, 4 - pos, &crc)); |
+ globalPos += 4 - pos; |
+ } |
+ { |
+ SetUi32(buf, CRC_GET_DIGEST(crc)); |
+ RINOK(WriteBytes(s, buf, 4)); |
+ globalPos += 4; |
+ } |
+ } |
+ |
+ { |
+ UInt32 indexSize = (UInt32)((globalPos >> 2) - 1); |
+ SetUi32(buf + 4, indexSize); |
+ buf[8] = (Byte)(p->flags >> 8); |
+ buf[9] = (Byte)(p->flags & 0xFF); |
+ SetUi32(buf, CrcCalc(buf + 4, 6)); |
+ memcpy(buf + 10, XZ_FOOTER_SIG, XZ_FOOTER_SIG_SIZE); |
+ return WriteBytes(s, buf, 12); |
+ } |
+} |
+ |
+SRes Xz_AddIndexRecord(CXzStream *p, UInt64 unpackSize, UInt64 totalSize, ISzAlloc *alloc) |
+{ |
+ if (p->blocks == 0 || p->numBlocksAllocated == p->numBlocks) |
+ { |
+ size_t num = (p->numBlocks + 1) * 2; |
+ size_t newSize = sizeof(CXzBlockSizes) * num; |
+ CXzBlockSizes *blocks; |
+ if (newSize / sizeof(CXzBlockSizes) != num) |
+ return SZ_ERROR_MEM; |
+ blocks = alloc->Alloc(alloc, newSize); |
+ if (blocks == 0) |
+ return SZ_ERROR_MEM; |
+ if (p->numBlocks != 0) |
+ { |
+ memcpy(blocks, p->blocks, p->numBlocks * sizeof(CXzBlockSizes)); |
+ Xz_Free(p, alloc); |
+ } |
+ p->blocks = blocks; |
+ p->numBlocksAllocated = num; |
+ } |
+ { |
+ CXzBlockSizes *block = &p->blocks[p->numBlocks++]; |
+ block->totalSize = totalSize; |
+ block->unpackSize = unpackSize; |
+ } |
+ return SZ_OK; |
+} |
+ |
+/* ---------- CSeqCheckInStream ---------- */ |
+ |
+typedef struct |
+{ |
+ ISeqInStream p; |
+ ISeqInStream *realStream; |
+ UInt64 processed; |
+ CXzCheck check; |
+} CSeqCheckInStream; |
+ |
+void SeqCheckInStream_Init(CSeqCheckInStream *p, int mode) |
+{ |
+ p->processed = 0; |
+ XzCheck_Init(&p->check, mode); |
+} |
+ |
+void SeqCheckInStream_GetDigest(CSeqCheckInStream *p, Byte *digest) |
+{ |
+ XzCheck_Final(&p->check, digest); |
+} |
+ |
+static SRes SeqCheckInStream_Read(void *pp, void *data, size_t *size) |
+{ |
+ CSeqCheckInStream *p = (CSeqCheckInStream *)pp; |
+ SRes res = p->realStream->Read(p->realStream, data, size); |
+ XzCheck_Update(&p->check, data, *size); |
+ p->processed += *size; |
+ return res; |
+} |
+ |
+/* ---------- CSeqSizeOutStream ---------- */ |
+ |
+typedef struct |
+{ |
+ ISeqOutStream p; |
+ ISeqOutStream *realStream; |
+ UInt64 processed; |
+} CSeqSizeOutStream; |
+ |
+static size_t MyWrite(void *pp, const void *data, size_t size) |
+{ |
+ CSeqSizeOutStream *p = (CSeqSizeOutStream *)pp; |
+ size = p->realStream->Write(p->realStream, data, size); |
+ p->processed += size; |
+ return size; |
+} |
+ |
+/* ---------- CSeqInFilter ---------- */ |
+ |
+/* |
+typedef struct _IFilter |
+{ |
+ void *p; |
+ void (*Free)(void *p, ISzAlloc *alloc); |
+ SRes (*SetProps)(void *p, const Byte *props, size_t propSize, ISzAlloc *alloc); |
+ void (*Init)(void *p); |
+ size_t (*Filter)(void *p, Byte *data, SizeT destLen); |
+} IFilter; |
+ |
+#define FILT_BUF_SIZE (1 << 19) |
+ |
+typedef struct |
+{ |
+ ISeqInStream p; |
+ ISeqInStream *realStream; |
+ UInt32 x86State; |
+ UInt32 ip; |
+ UInt64 processed; |
+ CXzCheck check; |
+ Byte buf[FILT_BUF_SIZE]; |
+ UInt32 bufferPos; |
+ UInt32 convertedPosBegin; |
+ UInt32 convertedPosEnd; |
+ IFilter *filter; |
+} CSeqInFilter; |
+ |
+static SRes SeqInFilter_Read(void *pp, void *data, size_t *size) |
+{ |
+ CSeqInFilter *p = (CSeqInFilter *)pp; |
+ size_t remSize = *size; |
+ *size = 0; |
+ |
+ while (remSize > 0) |
+ { |
+ int i; |
+ if (p->convertedPosBegin != p->convertedPosEnd) |
+ { |
+ UInt32 sizeTemp = p->convertedPosEnd - p->convertedPosBegin; |
+ if (remSize < sizeTemp) |
+ sizeTemp = (UInt32)remSize; |
+ memmove(data, p->buf + p->convertedPosBegin, sizeTemp); |
+ p->convertedPosBegin += sizeTemp; |
+ data = (void *)((Byte *)data + sizeTemp); |
+ remSize -= sizeTemp; |
+ *size += sizeTemp; |
+ break; |
+ } |
+ for (i = 0; p->convertedPosEnd + i < p->bufferPos; i++) |
+ p->buf[i] = p->buf[i + p->convertedPosEnd]; |
+ p->bufferPos = i; |
+ p->convertedPosBegin = p->convertedPosEnd = 0; |
+ { |
+ size_t processedSizeTemp = FILT_BUF_SIZE - p->bufferPos; |
+ RINOK(p->realStream->Read(p->realStream, p->buf + p->bufferPos, &processedSizeTemp)); |
+ p->bufferPos = p->bufferPos + (UInt32)processedSizeTemp; |
+ } |
+ p->convertedPosEnd = (UInt32)p->filter->Filter(p->filter->p, p->buf, p->bufferPos); |
+ if (p->convertedPosEnd == 0) |
+ { |
+ if (p->bufferPos == 0) |
+ break; |
+ else |
+ { |
+ p->convertedPosEnd = p->bufferPos; |
+ continue; |
+ } |
+ } |
+ if (p->convertedPosEnd > p->bufferPos) |
+ { |
+ for (; p->bufferPos < p->convertedPosEnd; p->bufferPos++) |
+ p->buf[p->bufferPos] = 0; |
+ p->convertedPosEnd = (UInt32)p->filter->Filter(p->filter->p, p->buf, p->bufferPos); |
+ } |
+ } |
+ return SZ_OK; |
+} |
+*/ |
+ |
+/* |
+typedef struct |
+{ |
+ ISeqInStream p; |
+ ISeqInStream *realStream; |
+ CMixCoder mixCoder; |
+ Byte buf[FILT_BUF_SIZE]; |
+ UInt32 bufPos; |
+ UInt32 bufSize; |
+} CMixCoderSeqInStream; |
+ |
+static SRes CMixCoderSeqInStream_Read(void *pp, void *data, size_t *size) |
+{ |
+ CMixCoderSeqInStream *p = (CMixCoderSeqInStream *)pp; |
+ SRes res = SZ_OK; |
+ size_t remSize = *size; |
+ *size = 0; |
+ while (remSize > 0) |
+ { |
+ if (p->bufPos == p->bufSize) |
+ { |
+ size_t curSize; |
+ p->bufPos = p->bufSize = 0; |
+ if (*size != 0) |
+ break; |
+ curSize = FILT_BUF_SIZE; |
+ RINOK(p->realStream->Read(p->realStream, p->buf, &curSize)); |
+ p->bufSize = (UInt32)curSize; |
+ } |
+ { |
+ SizeT destLen = remSize; |
+ SizeT srcLen = p->bufSize - p->bufPos; |
+ res = MixCoder_Code(&p->mixCoder, data, &destLen, p->buf + p->bufPos, &srcLen, 0); |
+ data = (void *)((Byte *)data + destLen); |
+ remSize -= destLen; |
+ *size += destLen; |
+ p->bufPos += srcLen; |
+ } |
+ } |
+ return res; |
+} |
+*/ |
+ |
+#ifdef USE_SUBBLOCK |
+typedef struct |
+{ |
+ ISeqInStream p; |
+ CSubblockEnc sb; |
+ UInt64 processed; |
+} CSbEncInStream; |
+ |
+void SbEncInStream_Init(CSbEncInStream *p) |
+{ |
+ p->processed = 0; |
+ SubblockEnc_Init(&p->sb); |
+} |
+ |
+static SRes SbEncInStream_Read(void *pp, void *data, size_t *size) |
+{ |
+ CSbEncInStream *p = (CSbEncInStream *)pp; |
+ SRes res = SubblockEnc_Read(&p->sb, data, size); |
+ p->processed += *size; |
+ return res; |
+} |
+#endif |
+ |
+typedef struct |
+{ |
+ /* CMixCoderSeqInStream inStream; */ |
+ CLzma2EncHandle lzma2; |
+ #ifdef USE_SUBBLOCK |
+ CSbEncInStream sb; |
+ #endif |
+ ISzAlloc *alloc; |
+ ISzAlloc *bigAlloc; |
+} CLzma2WithFilters; |
+ |
+ |
+static void Lzma2WithFilters_Construct(CLzma2WithFilters *p, ISzAlloc *alloc, ISzAlloc *bigAlloc) |
+{ |
+ p->alloc = alloc; |
+ p->bigAlloc = bigAlloc; |
+ p->lzma2 = NULL; |
+ #ifdef USE_SUBBLOCK |
+ p->sb.p.Read = SbEncInStream_Read; |
+ SubblockEnc_Construct(&p->sb.sb, p->alloc); |
+ #endif |
+} |
+ |
+static SRes Lzma2WithFilters_Create(CLzma2WithFilters *p) |
+{ |
+ p->lzma2 = Lzma2Enc_Create(p->alloc, p->bigAlloc); |
+ if (p->lzma2 == 0) |
+ return SZ_ERROR_MEM; |
+ return SZ_OK; |
+} |
+ |
+static void Lzma2WithFilters_Free(CLzma2WithFilters *p) |
+{ |
+ #ifdef USE_SUBBLOCK |
+ SubblockEnc_Free(&p->sb.sb); |
+ #endif |
+ if (p->lzma2) |
+ { |
+ Lzma2Enc_Destroy(p->lzma2); |
+ p->lzma2 = NULL; |
+ } |
+} |
+ |
+static SRes Xz_Compress(CXzStream *xz, |
+ CLzma2WithFilters *lzmaf, |
+ ISeqOutStream *outStream, |
+ ISeqInStream *inStream, |
+ const CLzma2EncProps *lzma2Props, |
+ Bool useSubblock, |
+ ICompressProgress *progress) |
+{ |
+ xz->flags = XZ_CHECK_CRC32; |
+ |
+ RINOK(Lzma2Enc_SetProps(lzmaf->lzma2, lzma2Props)); |
+ RINOK(Xz_WriteHeader(xz->flags, outStream)); |
+ |
+ { |
+ CSeqCheckInStream checkInStream; |
+ CSeqSizeOutStream seqSizeOutStream; |
+ CXzBlock block; |
+ int filterIndex = 0; |
+ |
+ XzBlock_ClearFlags(&block); |
+ XzBlock_SetNumFilters(&block, 1 + (useSubblock ? 1 : 0)); |
+ |
+ if (useSubblock) |
+ { |
+ CXzFilter *f = &block.filters[filterIndex++]; |
+ f->id = XZ_ID_Subblock; |
+ f->propsSize = 0; |
+ } |
+ |
+ { |
+ CXzFilter *f = &block.filters[filterIndex++]; |
+ f->id = XZ_ID_LZMA2; |
+ f->propsSize = 1; |
+ f->props[0] = Lzma2Enc_WriteProperties(lzmaf->lzma2); |
+ } |
+ |
+ seqSizeOutStream.p.Write = MyWrite; |
+ seqSizeOutStream.realStream = outStream; |
+ seqSizeOutStream.processed = 0; |
+ |
+ RINOK(XzBlock_WriteHeader(&block, &seqSizeOutStream.p)); |
+ |
+ checkInStream.p.Read = SeqCheckInStream_Read; |
+ checkInStream.realStream = inStream; |
+ SeqCheckInStream_Init(&checkInStream, XzFlags_GetCheckType(xz->flags)); |
+ |
+ #ifdef USE_SUBBLOCK |
+ if (useSubblock) |
+ { |
+ lzmaf->sb.sb.inStream = &checkInStream.p; |
+ SubblockEnc_Init(&lzmaf->sb.sb); |
+ } |
+ #endif |
+ |
+ { |
+ UInt64 packPos = seqSizeOutStream.processed; |
+ SRes res = Lzma2Enc_Encode(lzmaf->lzma2, &seqSizeOutStream.p, |
+ #ifdef USE_SUBBLOCK |
+ useSubblock ? &lzmaf->sb.p: |
+ #endif |
+ &checkInStream.p, |
+ progress); |
+ RINOK(res); |
+ block.unpackSize = checkInStream.processed; |
+ block.packSize = seqSizeOutStream.processed - packPos; |
+ } |
+ |
+ { |
+ unsigned padSize = 0; |
+ Byte buf[128]; |
+ while((((unsigned)block.packSize + padSize) & 3) != 0) |
+ buf[padSize++] = 0; |
+ SeqCheckInStream_GetDigest(&checkInStream, buf + padSize); |
+ RINOK(WriteBytes(&seqSizeOutStream.p, buf, padSize + XzFlags_GetCheckSize(xz->flags))); |
+ RINOK(Xz_AddIndexRecord(xz, block.unpackSize, seqSizeOutStream.processed - padSize, &g_Alloc)); |
+ } |
+ } |
+ return Xz_WriteFooter(xz, outStream); |
+} |
+ |
+SRes Xz_Encode(ISeqOutStream *outStream, ISeqInStream *inStream, |
+ const CLzma2EncProps *lzma2Props, Bool useSubblock, |
+ ICompressProgress *progress) |
+{ |
+ SRes res; |
+ CXzStream xz; |
+ CLzma2WithFilters lzmaf; |
+ Xz_Construct(&xz); |
+ Lzma2WithFilters_Construct(&lzmaf, &g_Alloc, &g_BigAlloc); |
+ res = Lzma2WithFilters_Create(&lzmaf); |
+ if (res == SZ_OK) |
+ res = Xz_Compress(&xz, &lzmaf, outStream, inStream, |
+ lzma2Props, useSubblock, progress); |
+ Lzma2WithFilters_Free(&lzmaf); |
+ Xz_Free(&xz, &g_Alloc); |
+ return res; |
+} |
+ |
+SRes Xz_EncodeEmpty(ISeqOutStream *outStream) |
+{ |
+ SRes res; |
+ CXzStream xz; |
+ Xz_Construct(&xz); |
+ res = Xz_WriteHeader(xz.flags, outStream); |
+ if (res == SZ_OK) |
+ res = Xz_WriteFooter(&xz, outStream); |
+ Xz_Free(&xz, &g_Alloc); |
+ return res; |
+} |
Property changes on: third_party\lzma_sdk\C\XzEnc.c |
___________________________________________________________________ |
Added: svn:executable |
+ * |
Added: svn:eol-style |
+ LF |