1*3117ece4Schristos /* 2*3117ece4Schristos * Copyright (c) Meta Platforms, Inc. and affiliates. 3*3117ece4Schristos * All rights reserved. 4*3117ece4Schristos * 5*3117ece4Schristos * This source code is licensed under both the BSD-style license (found in the 6*3117ece4Schristos * LICENSE file in the root directory of this source tree) and the GPLv2 (found 7*3117ece4Schristos * in the COPYING file in the root directory of this source tree). 8*3117ece4Schristos * You may select, at your option, one of the above-listed licenses. 9*3117ece4Schristos */ 10*3117ece4Schristos 11*3117ece4Schristos #include <stdio.h> // printf 12*3117ece4Schristos #include <stdlib.h> // free 13*3117ece4Schristos #include <string.h> // memcpy, strlen 14*3117ece4Schristos #include <zstd.h> // presumes zstd library is installed 15*3117ece4Schristos #include "common.h" // Helper functions, CHECK(), and CHECK_ZSTD() 16*3117ece4Schristos 17*3117ece4Schristos typedef struct { 18*3117ece4Schristos void* fBuffer; 19*3117ece4Schristos void* cBuffer; 20*3117ece4Schristos size_t fBufferSize; 21*3117ece4Schristos size_t cBufferSize; 22*3117ece4Schristos ZSTD_CCtx* cctx; 23*3117ece4Schristos } resources; 24*3117ece4Schristos 25*3117ece4Schristos /* 26*3117ece4Schristos * allocate memory for buffers big enough to compress all files 27*3117ece4Schristos * as well as memory for output file name (ofn) 28*3117ece4Schristos */ 29*3117ece4Schristos static resources createResources_orDie(int argc, const char** argv, char **ofn, size_t* ofnBufferLen) 30*3117ece4Schristos { 31*3117ece4Schristos size_t maxFilenameLength=0; 32*3117ece4Schristos size_t maxFileSize = 0; 33*3117ece4Schristos 34*3117ece4Schristos int argNb; 35*3117ece4Schristos for (argNb = 1; argNb < argc; argNb++) { 36*3117ece4Schristos const char* const filename = argv[argNb]; 37*3117ece4Schristos size_t const filenameLength = strlen(filename); 38*3117ece4Schristos size_t const fileSize = fsize_orDie(filename); 39*3117ece4Schristos 40*3117ece4Schristos if (filenameLength > maxFilenameLength) maxFilenameLength = filenameLength; 41*3117ece4Schristos if (fileSize > maxFileSize) maxFileSize = fileSize; 42*3117ece4Schristos } 43*3117ece4Schristos 44*3117ece4Schristos resources ress; 45*3117ece4Schristos ress.fBufferSize = maxFileSize; 46*3117ece4Schristos ress.cBufferSize = ZSTD_compressBound(maxFileSize); 47*3117ece4Schristos 48*3117ece4Schristos *ofnBufferLen = maxFilenameLength + 5; 49*3117ece4Schristos *ofn = (char*)malloc_orDie(*ofnBufferLen); 50*3117ece4Schristos ress.fBuffer = malloc_orDie(ress.fBufferSize); 51*3117ece4Schristos ress.cBuffer = malloc_orDie(ress.cBufferSize); 52*3117ece4Schristos ress.cctx = ZSTD_createCCtx(); 53*3117ece4Schristos CHECK(ress.cctx != NULL, "ZSTD_createCCtx() failed!"); 54*3117ece4Schristos return ress; 55*3117ece4Schristos } 56*3117ece4Schristos 57*3117ece4Schristos static void freeResources(resources ress, char *outFilename) 58*3117ece4Schristos { 59*3117ece4Schristos free(ress.fBuffer); 60*3117ece4Schristos free(ress.cBuffer); 61*3117ece4Schristos ZSTD_freeCCtx(ress.cctx); /* never fails */ 62*3117ece4Schristos free(outFilename); 63*3117ece4Schristos } 64*3117ece4Schristos 65*3117ece4Schristos /* compress with pre-allocated context (ZSTD_CCtx) and input/output buffers*/ 66*3117ece4Schristos static void compressFile_orDie(resources ress, const char* fname, const char* oname) 67*3117ece4Schristos { 68*3117ece4Schristos size_t fSize = loadFile_orDie(fname, ress.fBuffer, ress.fBufferSize); 69*3117ece4Schristos 70*3117ece4Schristos /* Compress using the context. 71*3117ece4Schristos * If you need more control over parameters, use the advanced API: 72*3117ece4Schristos * ZSTD_CCtx_setParameter(), and ZSTD_compress2(). 73*3117ece4Schristos */ 74*3117ece4Schristos size_t const cSize = ZSTD_compressCCtx(ress.cctx, ress.cBuffer, ress.cBufferSize, ress.fBuffer, fSize, 1); 75*3117ece4Schristos CHECK_ZSTD(cSize); 76*3117ece4Schristos 77*3117ece4Schristos saveFile_orDie(oname, ress.cBuffer, cSize); 78*3117ece4Schristos 79*3117ece4Schristos /* success */ 80*3117ece4Schristos printf("%25s : %6u -> %7u - %s \n", fname, (unsigned)fSize, (unsigned)cSize, oname); 81*3117ece4Schristos } 82*3117ece4Schristos 83*3117ece4Schristos int main(int argc, const char** argv) 84*3117ece4Schristos { 85*3117ece4Schristos const char* const exeName = argv[0]; 86*3117ece4Schristos 87*3117ece4Schristos if (argc<2) { 88*3117ece4Schristos printf("wrong arguments\n"); 89*3117ece4Schristos printf("usage:\n"); 90*3117ece4Schristos printf("%s FILE(s)\n", exeName); 91*3117ece4Schristos return 1; 92*3117ece4Schristos } 93*3117ece4Schristos 94*3117ece4Schristos /* memory allocation for outFilename and resources */ 95*3117ece4Schristos char* outFilename; 96*3117ece4Schristos size_t outFilenameBufferLen; 97*3117ece4Schristos resources const ress = createResources_orDie(argc, argv, &outFilename, &outFilenameBufferLen); 98*3117ece4Schristos 99*3117ece4Schristos /* compress files with shared context, input and output buffers */ 100*3117ece4Schristos int argNb; 101*3117ece4Schristos for (argNb = 1; argNb < argc; argNb++) { 102*3117ece4Schristos const char* const inFilename = argv[argNb]; 103*3117ece4Schristos size_t const inFilenameLen = strlen(inFilename); 104*3117ece4Schristos CHECK(inFilenameLen + 5 <= outFilenameBufferLen, "File name too long!"); 105*3117ece4Schristos memcpy(outFilename, inFilename, inFilenameLen); 106*3117ece4Schristos memcpy(outFilename+inFilenameLen, ".zst", 5); 107*3117ece4Schristos compressFile_orDie(ress, inFilename, outFilename); 108*3117ece4Schristos } 109*3117ece4Schristos 110*3117ece4Schristos /* free memory */ 111*3117ece4Schristos freeResources(ress,outFilename); 112*3117ece4Schristos 113*3117ece4Schristos printf("compressed %i files \n", argc-1); 114*3117ece4Schristos 115*3117ece4Schristos return 0; 116*3117ece4Schristos } 117