// sync_info_make.cpp // sync_make // Created by housisong on 2019-09-17. /* The MIT License (MIT) Copyright (c) 2019-2023 HouSisong Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #include "sync_info_make.h" #include "sync_make_hash_clash.h" // isCanUse32bitRollHash #include "../sync_client/sync_info_client.h" // TNewDataSyncInfo_dir_saveHeadTo using namespace hdiff_private; namespace sync_private{ #define _outBuf(_buf_begin,_buf_end) { \ checksumInfo.append(_buf_begin,_buf_end); \ writeStream(out_stream,outPos,_buf_begin,_buf_end); } #define _flushV(_v) if (!_v.empty()){ _outBuf(_v.data(),_v.data()+_v.size()); _v.clear(); } #define _flushV_end(_v) if (!_v.empty()){ _flushV(_v); swapClear(_v); } static void saveSamePairList(std::vector &buf, const TSameNewBlockPair* samePairList, size_t samePairCount) { uint32_t pre=0; for (size_t i=0;isp.sameIndex); packUInt(buf,(uint32_t)(sp.curIndex-pre)); packUInt(buf,(uint32_t)(sp.curIndex-sp.sameIndex)); pre=sp.curIndex; } } static void saveSavedSizes(std::vector &buf, TNewDataSyncInfo *self) { const uint32_t kBlockCount=(uint32_t)TNewDataSyncInfo_blockCount(self); uint32_t backSize=0; for (uint32_t i=0; isavedSizes[i]; if (savedSize!=0){ if (savedSize>=backSize) packUIntWithTag(buf,(uint32_t)(savedSize-backSize),0,1); else packUIntWithTag(buf,(uint32_t)(backSize-savedSize),1,1); backSize=savedSize; }else{ packUIntWithTag(buf,0,1,1); backSize=self->kSyncBlockSize; } } } static void _compressBuf(std::vector &buf,hsync_TDictCompress* compressPlugin) { std::vector cmbuf; cmbuf.resize((size_t)compressPlugin->maxCompressedSize(buf.size())); hsync_dictCompressHandle compressHandle=compressPlugin->dictCompressOpen(compressPlugin,1,buf.size()); checkv(compressHandle!=0); size_t compressedSize=compressPlugin->dictCompress(compressHandle,0,cmbuf.data(),cmbuf.data()+cmbuf.size(), buf.data(),buf.data()+buf.size()); checkv(compressedSize!=kDictCompressError); if ((compressedSize==kDictCompressCancel)||(compressedSize>=buf.size())) compressedSize=0; //cancel compress compressPlugin->dictCompressClose(compressPlugin,compressHandle); if (compressedSize>0){ cmbuf.resize(compressedSize); buf.swap(cmbuf); } } static void savePartHash(const hpatch_TStreamOutput *out_stream,hpatch_StreamPos_t &outPos, uint32_t kBlockCount,const TByte* partHash,size_t partBits, const TSameNewBlockPair* samePairList,uint32_t samePairCount,CChecksum &checksumInfo) { std::vector buf; uint32_t curPair=0; const size_t partBytes=_bitsToBytes(partBits); size_t bitsValue=0; size_t bitsCount=0; const size_t highBit=(partBits&7)?(partBits&7):8; for (size_t i=0; i=8){ buf.push_back((hpatch_byte)bitsValue); bitsValue>>=8; bitsCount-=8; //assert(bitsValue<((size_t)1<=hpatch_kFileIOBufBetterSize) _flushV(buf); } //assert(bitsCount<8); } } assert((bitsValue>>bitsCount)==0); if (bitsCount>0) buf.push_back((hpatch_byte)bitsValue); _flushV(buf); assert(curPair==samePairCount); } #if (_IS_NEED_DIR_DIFF_PATCH) static void _saveDirHeadTo(const TNewDataSyncInfo_dir* self,std::vector& out_buf){ packUInt(out_buf,self->dir_newExecuteCount); packUInt(out_buf,self->dir_newPathCount); packUInt(out_buf,self->dir_newPathSumCharSize); } void TNewDataSyncInfo_dirWithHead_saveTo(TNewDataSyncInfo_dir* self,std::vector& out_buf){ size_t pos=out_buf.size(); TNewDataSyncInfo_dir_saveTo(self,out_buf); std::vector head; _saveDirHeadTo(self,head); out_buf.insert(out_buf.begin()+pos,head.begin(),head.end()); } #endif void TNewDataSyncInfo_saveTo(TNewDataSyncInfo* self,const hpatch_TStreamOutput* out_stream, hsync_TDictCompress* compressPlugin){ #if (_IS_NEED_DIR_DIFF_PATCH) if (self->isDirSyncInfo) checkv(self->dirInfoSavedSize>0); #else checkv(!self->isDirSyncInfo); #endif const char* kVersionType=self->isDirSyncInfo?"HDirSyni23":"HSyni23"; if (compressPlugin) checkv(0==strcmp(compressPlugin->compressType(),self->compressType)); else checkv(self->compressType==0); hpatch_TChecksum* strongChecksumPlugin=self->_strongChecksumPlugin; checkv(0==strcmp(strongChecksumPlugin->checksumType(),self->strongChecksumType)); const size_t privateExternDataSize=0; //reserved ,now empty const size_t externDataSize=0;//reserved ,now empty const uint8_t isSavedSizes=(self->savedSizes)?1:0; std::vector buf; saveSamePairList(buf,self->samePairList,self->samePairCount); if (isSavedSizes) saveSavedSizes(buf,self); //compress buf size_t uncompressDataSize=buf.size(); size_t compressDataSize=0; if (compressPlugin){ _compressBuf(buf,compressPlugin); if (buf.size() head; {//head pushTypes(head,kVersionType,compressPlugin?compressPlugin->compressType():0,strongChecksumPlugin); packUInt(head,self->dictSize); packUInt(head,self->decompressInfoSize); pushBack(head,self->decompressInfo,self->decompressInfoSize); packUInt(head,self->newSyncDataSize); packUInt(head,self->newSyncDataOffsert); packUInt(head,self->newDataSize); packUInt(head,self->kSyncBlockSize); packUInt(head,self->kStrongChecksumByteSize); packUInt(head,self->savedStrongChecksumBits); packUInt(head,self->savedRollHashBits); packUInt(head,self->samePairCount); pushUInt(head,isSavedSizes); packUInt(head,privateExternDataSize); packUInt(head,externDataSize); packUInt(head,uncompressDataSize); packUInt(head,compressDataSize); pushUInt(head,self->isDirSyncInfo); #if (_IS_NEED_DIR_DIFF_PATCH) if (self->isDirSyncInfo){ packUInt(head,self->dirInfoSavedSize); _saveDirHeadTo(&self->dirInfo,head); } #endif {//newSyncInfoSize const hpatch_StreamPos_t savedBCount=kBlockCount-self->samePairCount; self->newSyncInfoSize = head.size()+privateExternDataSize+externDataSize+buf.size(); self->newSyncInfoSize +=_bitsToBytes(self->savedRollHashBits*savedBCount) +_bitsToBytes(self->savedStrongChecksumBits*savedBCount); self->newSyncInfoSize += self->kStrongChecksumByteSize*3; { hpatch_uint pksize=hpatch_packUInt_size(self->newSyncInfoSize); self->newSyncInfoSize+=pksize; if (hpatch_packUInt_size(self->newSyncInfoSize)>pksize){ assert(hpatch_packUInt_size(self->newSyncInfoSize)==pksize+1); self->newSyncInfoSize++; assert(hpatch_packUInt_size(self->newSyncInfoSize)==pksize+1); } } packUInt(head,self->newSyncInfoSize); } //end head info } CChecksum checksumInfo(strongChecksumPlugin); hpatch_StreamPos_t outPos=0; //out head buf _flushV_end(head); //adding newDataCheckChecksum _outBuf(self->savedNewDataCheckChecksum,self->savedNewDataCheckChecksum+self->kStrongChecksumByteSize); assert(privateExternDataSize==0);//out privateExternData //reserved ,now empty assert(externDataSize==0);//reserved ,now empty _flushV_end(buf); {// out info Checksum checksumInfo.appendEnd(); assert(checksumInfo.checksum.size()==self->kStrongChecksumByteSize); memcpy(self->infoChecksum,checksumInfo.checksum.data(),checksumInfo.checksum.size()); checksumInfo.appendBegin(); _outBuf(self->infoChecksum,self->infoChecksum+self->kStrongChecksumByteSize); self->infoChecksumEndPos=outPos; } savePartHash(out_stream,outPos,(uint32_t)kBlockCount,self->rollHashs,self->savedRollHashBits, self->samePairList,self->samePairCount,checksumInfo); savePartHash(out_stream,outPos,(uint32_t)kBlockCount,self->partChecksums,self->savedStrongChecksumBits, self->samePairList,self->samePairCount,checksumInfo); {// out info partHash Checksum checksumInfo.appendEnd(); assert(checksumInfo.checksum.size()==self->kStrongChecksumByteSize); memcpy(self->infoPartHashChecksum,checksumInfo.checksum.data(),checksumInfo.checksum.size()); writeStream(out_stream,outPos,self->infoPartHashChecksum,checksumInfo.checksum.size()); checkv(outPos==self->newSyncInfoSize); } } CNewDataSyncInfo::CNewDataSyncInfo(hpatch_TChecksum* strongChecksumPlugin,const hsync_TDictCompress* compressPlugin, hpatch_StreamPos_t newDataSize,uint32_t syncBlockSize,size_t kSafeHashClashBit){ TNewDataSyncInfo_init(this); if (compressPlugin){ this->_compressType.assign(compressPlugin->compressType()); this->compressType=this->_compressType.c_str(); } this->_strongChecksumPlugin=strongChecksumPlugin; this->_strongChecksumType.assign(strongChecksumPlugin->checksumType()); this->strongChecksumType=this->_strongChecksumType.c_str(); this->kStrongChecksumByteSize=(uint32_t)strongChecksumPlugin->checksumByteSize(); syncBlockSize=(syncBlockSize<=newDataSize)?syncBlockSize:(uint32_t)newDataSize; syncBlockSize=(syncBlockSize<=_kSyncBlockSize_min_limit)?_kSyncBlockSize_min_limit:syncBlockSize; this->kSyncBlockSize=syncBlockSize; this->newDataSize=newDataSize; getSavedHashBits(kSafeHashClashBit,newDataSize,this->kSyncBlockSize,this->kStrongChecksumByteSize*8, &this->savedRollHashBits,&this->savedStrongChecksumBits); this->savedRollHashByteSize=_bitsToBytes(this->savedRollHashBits); this->savedStrongChecksumByteSize=_bitsToBytes(this->savedStrongChecksumBits); //mem const size_t kBlockCount=this->blockCount(); hpatch_StreamPos_t memSize=kBlockCount*( (hpatch_StreamPos_t)0 +sizeof(TSameNewBlockPair)+this->savedStrongChecksumByteSize +this->savedRollHashByteSize +(compressPlugin?sizeof(uint32_t):0)); memSize+=this->kStrongChecksumByteSize*2+checkChecksumBufByteSize(this->kStrongChecksumByteSize); checkv(memSize==(size_t)memSize); _mem.realloc((size_t)memSize); TByte* curMem=_mem.data(); this->infoChecksum=curMem; curMem+=this->kStrongChecksumByteSize; this->infoPartHashChecksum=curMem; curMem+=this->kStrongChecksumByteSize; this->savedNewDataCheckChecksum=curMem; curMem+=checkChecksumBufByteSize(this->kStrongChecksumByteSize); this->partChecksums=curMem; curMem+=kBlockCount*this->savedStrongChecksumByteSize; this->samePairCount=0; this->samePairList=(TSameNewBlockPair*)curMem; curMem+=kBlockCount*sizeof(TSameNewBlockPair); this->rollHashs=curMem; curMem+=kBlockCount*this->savedRollHashByteSize; if (compressPlugin){ this->savedSizes=(uint32_t*)curMem; curMem+=kBlockCount*sizeof(uint32_t); } assert(curMem==_mem.data_end()); } static hpatch_StreamPos_t _hsynz_write_head(struct hsync_THsynz* zPlugin, const hpatch_TStreamOutput* out_stream,hpatch_StreamPos_t curOutPos, bool isDirSync,hpatch_StreamPos_t newDataSize,uint32_t kSyncBlockSize, hpatch_TChecksum* strongChecksumPlugin,hsync_TDictCompress* compressPlugin){ const char* kVersionType=isDirSync?"HDirSynz23":"HSynz23"; std::vector head; pushTypes(head,kVersionType,compressPlugin?compressPlugin->compressType():0,strongChecksumPlugin); packUInt(head,newDataSize); packUInt(head,kSyncBlockSize); packUInt(head,strongChecksumPlugin->checksumByteSize()); checkv(out_stream->write(out_stream,curOutPos,head.data(),head.data()+head.size())); return curOutPos+head.size(); } static hpatch_StreamPos_t _hsynz_write_foot(struct hsync_THsynz* zPlugin, const hpatch_TStreamOutput* out_stream,hpatch_StreamPos_t curOutPos, const hpatch_byte* newDataCheckChecksum,size_t checksumSize){ checkv(out_stream->write(out_stream,curOutPos,newDataCheckChecksum,newDataCheckChecksum+checksumSize)); return curOutPos+checksumSize; } void getHsynzPluginDefault(hsync_THsynz* out_hsynzPlugin){ checkv(out_hsynzPlugin!=0); out_hsynzPlugin->hsynz_write_head=_hsynz_write_head; out_hsynzPlugin->hsynz_readed_data=0; out_hsynzPlugin->hsynz_write_foot=_hsynz_write_foot; } }//namespace sync_private