/* * Copyright (c) 2021 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef BLOCKS_DIFF_H #define BLOCKS_DIFF_H #include #include #include "bzip2_adapter.h" #include "diffpatch.h" #include "pkg_manager.h" #include "securec.h" #include "update_diff.h" namespace UpdatePatch { template class SuffixArray { public: SuffixArray() = default; ~SuffixArray() {} void Init(const BlockBuffer &oldInfo); int64_t Search(const BlockBuffer &newInfo, const BlockBuffer &oldInfo, int64_t start, int64_t end, int64_t &pos) const; private: void InitBuckets(const BlockBuffer &oldInfo, std::vector &buckets, std::vector &suffixArrayTemp); void Split(std::vector &suffixArrayTemp, DataType start, DataType len, DataType h); void SplitForLess(std::vector &suffixArrayTemp, DataType start, DataType len, DataType h); int64_t MatchLength(const BlockBuffer &oldBuffer, const BlockBuffer &newBuffer) const; std::vector suffixArray_; }; class BlocksDiff { public: BlocksDiff() = default; virtual ~BlocksDiff() {} static int32_t MakePatch(const std::string &oldFileName, const std::string &newFileName, const std::string &patchFileName); static int32_t MakePatch(const BlockBuffer &newInfo, const BlockBuffer &oldInfo, std::vector &patchData, size_t offset, size_t &patchSize); static int32_t MakePatch(const BlockBuffer &newInfo, const BlockBuffer &oldInfo, std::fstream &patchFile, size_t &patchSize); int32_t MakePatch(const BlockBuffer &newInfo, const BlockBuffer &oldInfo, size_t &patchSize); private: virtual std::unique_ptr CreateBZip2Adapter(size_t patchOffset) = 0; virtual int32_t WritePatchHeader(int64_t controlSize, int64_t diffDataSize, int64_t newSize, size_t &headerLen) = 0; int32_t GetCtrlDatas(const BlockBuffer &newInfo, const BlockBuffer &oldInfo, std::vector &controlDatas); int32_t WritePatchData(const std::vector &controlDatas, const BlockBuffer &newInfo, size_t &patchSize); int32_t WriteControlData(const std::vector controlDatas, size_t &patchSize); int32_t WriteDiffData(const std::vector controlDatas, size_t &patchSize); int32_t WriteExtraData(const std::vector controlDatas, size_t &patchSize); void ComputeOldScore(const BlockBuffer &newInfo, const BlockBuffer &oldInfo, int64_t &oldScore, int64_t &matchLen); void ComputeLength(const BlockBuffer &newInfo, const BlockBuffer &oldInfo, int64_t &lengthFront, int64_t &lengthBack); std::unique_ptr> suffixArray_ {nullptr}; std::vector controls_ {}; int64_t matchPos_ { 0 }; int64_t currentOffset_ { 0 }; int64_t lastOffset_ { 0 }; int64_t lastScan_ { 0 }; int64_t lastPos_ { 0 }; }; class BlocksStreamDiff : public BlocksDiff { public: BlocksStreamDiff(std::fstream &stream, size_t offset) : BlocksDiff(), stream_(stream), offset_(offset) {} ~BlocksStreamDiff() override {} private: std::unique_ptr CreateBZip2Adapter(size_t patchOffset) override; int32_t WritePatchHeader(int64_t controlSize, int64_t diffDataSize, int64_t newSize, size_t &headerLen) override; std::fstream &stream_; size_t offset_ { 0 }; }; class BlocksBufferDiff : public BlocksDiff { public: BlocksBufferDiff(std::vector &patchData, size_t offset) : BlocksDiff(), patchData_(patchData), offset_(offset) {} ~BlocksBufferDiff() override {} private: std::unique_ptr CreateBZip2Adapter(size_t patchOffset) override; int32_t WritePatchHeader(int64_t controlSize, int64_t diffDataSize, int64_t newSize, size_t &headerLen) override; std::vector &patchData_; size_t offset_ { 0 }; }; } // namespace UpdatePatch #endif // BLOCKS_DIFF_H