/* compress.cpp -- This file is part of the UPX executable compressor. Copyright (C) 1996-2024 Markus Franz Xaver Johannes Oberhumer All Rights Reserved. UPX and the UCL library are free software; you can redistribute them and/or modify them under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; see the file COPYING. If not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. Markus F.X.J. Oberhumer */ #include "../conf.h" #include "compress.h" #include "../util/membuffer.h" /************************************************************************* // **************************************************************************/ unsigned upx_adler32(const void *buf, unsigned len, unsigned adler) { if (len == 0) return adler; assert(buf != nullptr); #if 1 return upx_ucl_adler32(buf, len, adler); #else return upx_zlib_adler32(buf, len, adler); #endif } #if 0 // UNUSED unsigned upx_crc32(const void *buf, unsigned len, unsigned crc) { if (len == 0) return crc; assert(buf != nullptr); #if 1 return upx_ucl_crc32(buf, len, crc); #else return upx_zlib_crc32(buf, len, crc); #endif } #endif // UNUSED /************************************************************************* // **************************************************************************/ int upx_compress(const upx_bytep src, unsigned src_len, upx_bytep dst, unsigned *dst_len, upx_callback_t *cb, int method, int level, const upx_compress_config_t *cconf, upx_compress_result_t *cresult) { int r = UPX_E_ERROR; upx_compress_result_t cresult_buffer; assert(method > 0); assert(level > 0); #if 1 // set available bytes in dst if (*dst_len == 0) *dst_len = MemBuffer::getSizeForCompression(src_len); #else // force users to provide *dst_len assert(*dst_len != 0); #endif // for UPX, we always require a reasonably sized output buffer assert(*dst_len >= MemBuffer::getSizeForCompression(src_len)); if (!cresult) cresult = &cresult_buffer; cresult->reset(); #if 1 // debugging aid cresult->debug.method = method; cresult->debug.level = level; cresult->debug.u_len = src_len; cresult->debug.c_len = 0; #endif const unsigned orig_dst_len = *dst_len; if (__acc_cte(false)) { } #if (WITH_BZIP2) else if (M_IS_BZIP2(method)) r = upx_bzip2_compress(src, src_len, dst, dst_len, cb, method, level, cconf, cresult); #endif #if (WITH_LZMA) else if (M_IS_LZMA(method)) r = upx_lzma_compress(src, src_len, dst, dst_len, cb, method, level, cconf, cresult); #endif #if (WITH_NRV) else if ((M_IS_NRV2B(method) || M_IS_NRV2D(method) || M_IS_NRV2E(method)) && !opt->prefer_ucl) r = upx_nrv_compress(src, src_len, dst, dst_len, cb, method, level, cconf, cresult); #endif #if (WITH_UCL) else if (M_IS_NRV2B(method) || M_IS_NRV2D(method) || M_IS_NRV2E(method)) r = upx_ucl_compress(src, src_len, dst, dst_len, cb, method, level, cconf, cresult); #endif #if (WITH_ZSTD) else if (M_IS_ZSTD(method)) r = upx_zstd_compress(src, src_len, dst, dst_len, cb, method, level, cconf, cresult); #endif else { throwInternalError("unknown compression method %d", method); } #if 1 // debugging aid cresult->debug.c_len = *dst_len; #endif assert_noexcept(*dst_len <= orig_dst_len); return r; } /************************************************************************* // **************************************************************************/ int upx_decompress(const upx_bytep src, unsigned src_len, upx_bytep dst, unsigned *dst_len, int method, const upx_compress_result_t *cresult) { int r = UPX_E_ERROR; assert(*dst_len > 0); assert(src_len < *dst_len); // must be compressed if (cresult && cresult->debug.method == 0) cresult = nullptr; const unsigned orig_dst_len = *dst_len; if (__acc_cte(false)) { } #if (WITH_BZIP2) else if (M_IS_BZIP2(method)) r = upx_bzip2_decompress(src, src_len, dst, dst_len, method, cresult); #endif #if (WITH_LZMA) else if (M_IS_LZMA(method)) r = upx_lzma_decompress(src, src_len, dst, dst_len, method, cresult); #endif #if (WITH_NRV) else if ((M_IS_NRV2B(method) || M_IS_NRV2D(method) || M_IS_NRV2E(method)) && !opt->prefer_ucl) r = upx_nrv_decompress(src, src_len, dst, dst_len, method, cresult); #endif #if (WITH_UCL) else if (M_IS_NRV2B(method) || M_IS_NRV2D(method) || M_IS_NRV2E(method)) r = upx_ucl_decompress(src, src_len, dst, dst_len, method, cresult); #endif #if (WITH_ZLIB) else if (M_IS_DEFLATE(method)) r = upx_zlib_decompress(src, src_len, dst, dst_len, method, cresult); #endif #if (WITH_ZSTD) else if (M_IS_ZSTD(method)) r = upx_zstd_decompress(src, src_len, dst, dst_len, method, cresult); #endif else { throwInternalError("unknown compression method %d", method); } assert_noexcept(*dst_len <= orig_dst_len); return r; } /************************************************************************* // **************************************************************************/ int upx_test_overlap(const upx_bytep buf, const upx_bytep tbuf, unsigned src_off, unsigned src_len, unsigned *dst_len, int method, const upx_compress_result_t *cresult) { int r = UPX_E_ERROR; if (cresult && cresult->debug.method == 0) cresult = nullptr; assert(*dst_len > 0); assert(src_len < *dst_len); // must be compressed unsigned overlap_overhead = src_off + src_len - *dst_len; assert((int) overlap_overhead > 0); const unsigned orig_dst_len = *dst_len; if (__acc_cte(false)) { } #if (WITH_BZIP2) else if (M_IS_BZIP2(method)) r = upx_bzip2_test_overlap(buf, tbuf, src_off, src_len, dst_len, method, cresult); #endif #if (WITH_LZMA) else if (M_IS_LZMA(method)) r = upx_lzma_test_overlap(buf, tbuf, src_off, src_len, dst_len, method, cresult); #endif #if (WITH_NRV) else if ((M_IS_NRV2B(method) || M_IS_NRV2D(method) || M_IS_NRV2E(method)) && !opt->prefer_ucl) r = upx_nrv_test_overlap(buf, tbuf, src_off, src_len, dst_len, method, cresult); #endif #if (WITH_UCL) else if (M_IS_NRV2B(method) || M_IS_NRV2D(method) || M_IS_NRV2E(method)) r = upx_ucl_test_overlap(buf, tbuf, src_off, src_len, dst_len, method, cresult); #endif #if (WITH_ZSTD) else if (M_IS_ZSTD(method)) r = upx_zstd_test_overlap(buf, tbuf, src_off, src_len, dst_len, method, cresult); #endif else { throwInternalError("unknown compression method %d", method); } assert_noexcept(*dst_len <= orig_dst_len); return r; } /* vim:set ts=4 sw=4 et: */