C++程序  |  312行  |  7.58 KB

// LZ4frame API example : compress a file
// Based on sample code from Zbigniew Jędrzejewski-Szmek

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>

#include <lz4frame.h>

#define BUF_SIZE 16*1024
#define LZ4_HEADER_SIZE 19
#define LZ4_FOOTER_SIZE 4

static const LZ4F_preferences_t lz4_preferences = {
	{ LZ4F_max256KB, LZ4F_blockLinked, LZ4F_noContentChecksum, LZ4F_frame, 0, { 0, 0 } },
	0,   /* compression level */
	0,   /* autoflush */
	{ 0, 0, 0, 0 },  /* reserved, must be set to 0 */
};

static size_t compress_file(FILE *in, FILE *out, size_t *size_in, size_t *size_out) {
	LZ4F_errorCode_t r;
	LZ4F_compressionContext_t ctx;
	char *src, *buf = NULL;
	size_t size, n, k, count_in = 0, count_out, offset = 0, frame_size;

	r = LZ4F_createCompressionContext(&ctx, LZ4F_VERSION);
	if (LZ4F_isError(r)) {
		printf("Failed to create context: error %zu\n", r);
		return 1;
	}
	r = 1;

	src = malloc(BUF_SIZE);
	if (!src) {
		printf("Not enough memory\n");
		goto cleanup;
	}

	frame_size = LZ4F_compressBound(BUF_SIZE, &lz4_preferences);
	size =  frame_size + LZ4_HEADER_SIZE + LZ4_FOOTER_SIZE;
	buf = malloc(size);
	if (!buf) {
		printf("Not enough memory\n");
		goto cleanup;
	}

	n = offset = count_out = LZ4F_compressBegin(ctx, buf, size, &lz4_preferences);
	if (LZ4F_isError(n)) {
		printf("Failed to start compression: error %zu\n", n);
		goto cleanup;
	}

	printf("Buffer size is %zu bytes, header size %zu bytes\n", size, n);

	for (;;) {
		k = fread(src, 1, BUF_SIZE, in);
		if (k == 0)
			break;
		count_in += k;

		n = LZ4F_compressUpdate(ctx, buf + offset, size - offset, src, k, NULL);
		if (LZ4F_isError(n)) {
			printf("Compression failed: error %zu\n", n);
			goto cleanup;
		}

		offset += n;
		count_out += n;
		if (size - offset < frame_size + LZ4_FOOTER_SIZE) {
			printf("Writing %zu bytes\n", offset);

			k = fwrite(buf, 1, offset, out);
			if (k < offset) {
				if (ferror(out))
					printf("Write failed\n");
				else
					printf("Short write\n");
				goto cleanup;
			}

			offset = 0;
		}
	}

	n = LZ4F_compressEnd(ctx, buf + offset, size - offset, NULL);
	if (LZ4F_isError(n)) {
		printf("Failed to end compression: error %zu\n", n);
		goto cleanup;
	}

	offset += n;
	count_out += n;
	printf("Writing %zu bytes\n", offset);

	k = fwrite(buf, 1, offset, out);
	if (k < offset) {
		if (ferror(out))
			printf("Write failed\n");
		else
			printf("Short write\n");
		goto cleanup;
	}

	*size_in = count_in;
	*size_out = count_out;
	r = 0;
 cleanup:
	if (ctx)
		LZ4F_freeCompressionContext(ctx);
	free(src);
	free(buf);
	return r;
}

static size_t get_block_size(const LZ4F_frameInfo_t* info) {
	switch (info->blockSizeID) {
        case LZ4F_default:
		case LZ4F_max64KB:  return 1 << 16;
		case LZ4F_max256KB: return 1 << 18;
		case LZ4F_max1MB:   return 1 << 20;
		case LZ4F_max4MB:   return 1 << 22;
		default:
			printf("Impossible unless more block sizes are allowed\n");
			exit(1);
	}
}

static size_t decompress_file(FILE *in, FILE *out) {
	void* const src = malloc(BUF_SIZE);
	void* dst = NULL;
	size_t dstCapacity = 0;
	LZ4F_dctx *dctx = NULL;
	size_t ret;

	/* Initialization */
    if (!src) { perror("decompress_file(src)"); goto cleanup; }
	ret = LZ4F_createDecompressionContext(&dctx, 100);
	if (LZ4F_isError(ret)) {
		printf("LZ4F_dctx creation error: %s\n", LZ4F_getErrorName(ret));
		goto cleanup;
	}

	/* Decompression */
	ret = 1;
	while (ret != 0) {
		/* Load more input */
		size_t srcSize = fread(src, 1, BUF_SIZE, in);
		void* srcPtr = src;
		void* srcEnd = srcPtr + srcSize;
		if (srcSize == 0 || ferror(in)) {
			printf("Decompress: not enough input or error reading file\n");
			goto cleanup;
		}
		/* Allocate destination buffer if it isn't already */
		if (!dst) {
			LZ4F_frameInfo_t info;
			ret = LZ4F_getFrameInfo(dctx, &info, src, &srcSize);
			if (LZ4F_isError(ret)) {
				printf("LZ4F_getFrameInfo error: %s\n", LZ4F_getErrorName(ret));
				goto cleanup;
			}
			/* Allocating enough space for an entire block isn't necessary for
			 * correctness, but it allows some memcpy's to be elided.
			 */
			dstCapacity = get_block_size(&info);
			dst = malloc(dstCapacity);
            if (!dst) { perror("decompress_file(dst)"); goto cleanup; }
			srcPtr += srcSize;
			srcSize = srcEnd - srcPtr;
		}
		/* Decompress:
		 * Continue while there is more input to read and the frame isn't over.
		 * If srcPtr == srcEnd then we know that there is no more output left in the
		 * internal buffer left to flush.
		 */
		while (srcPtr != srcEnd && ret != 0) {
			/* INVARIANT: Any data left in dst has already been written */
			size_t dstSize = dstCapacity;
			ret = LZ4F_decompress(dctx, dst, &dstSize, srcPtr, &srcSize, /* LZ4F_decompressOptions_t */ NULL);
			if (LZ4F_isError(ret)) {
				printf("Decompression error: %s\n", LZ4F_getErrorName(ret));
				goto cleanup;
			}
			/* Flush output */
			if (dstSize != 0){
				size_t written = fwrite(dst, 1, dstSize, out);
				printf("Writing %zu bytes\n", dstSize);
				if (written != dstSize) {
					printf("Decompress: Failed to write to file\n");
					goto cleanup;
				}
			}
			/* Update input */
			srcPtr += srcSize;
			srcSize = srcEnd - srcPtr;
		}
	}
	/* Check that there isn't trailing input data after the frame.
	 * It is valid to have multiple frames in the same file, but this example
	 * doesn't support it.
	 */
	ret = fread(src, 1, 1, in);
	if (ret != 0 || !feof(in)) {
		printf("Decompress: Trailing data left in file after frame\n");
		goto cleanup;
	}

cleanup:
	free(src);
	free(dst);
	return LZ4F_freeDecompressionContext(dctx);   /* note : free works on NULL */
}

int compare(FILE* fp0, FILE* fp1)
{
	int result = 0;

	while(0 == result) {
		char b0[1024];
		char b1[1024];
		const size_t r0 = fread(b0, 1, sizeof(b0), fp0);
		const size_t r1 = fread(b1, 1, sizeof(b1), fp1);

		result = (int) r0 - (int) r1;

		if (0 == r0 || 0 == r1) {
			break;
		}
		if (0 == result) {
			result = memcmp(b0, b1, r0);
		}
	}

	return result;
}

int main(int argc, const char **argv) {
	char inpFilename[256] = { 0 };
	char lz4Filename[256] = { 0 };
	char decFilename[256] = { 0 };

	if(argc < 2) {
		printf("Please specify input filename\n");
		return 0;
	}

	snprintf(inpFilename, 256, "%s", argv[1]);
	snprintf(lz4Filename, 256, "%s.lz4", argv[1]);
	snprintf(decFilename, 256, "%s.lz4.dec", argv[1]);

	printf("inp = [%s]\n", inpFilename);
	printf("lz4 = [%s]\n", lz4Filename);
	printf("dec = [%s]\n", decFilename);

	/* compress */
	{   FILE* const inpFp = fopen(inpFilename, "rb");
		FILE* const outFp = fopen(lz4Filename, "wb");
		size_t sizeIn = 0;
		size_t sizeOut = 0;
		size_t ret;

		printf("compress : %s -> %s\n", inpFilename, lz4Filename);
		ret = compress_file(inpFp, outFp, &sizeIn, &sizeOut);
		if (ret) {
			printf("compress : failed with code %zu\n", ret);
			return ret;
		}
		printf("%s: %zu → %zu bytes, %.1f%%\n",
			inpFilename, sizeIn, sizeOut,
			(double)sizeOut / sizeIn * 100);
		printf("compress : done\n");

		fclose(outFp);
		fclose(inpFp);
	}

	/* decompress */
	{   FILE* const inpFp = fopen(lz4Filename, "rb");
		FILE* const outFp = fopen(decFilename, "wb");
		size_t ret;

		printf("decompress : %s -> %s\n", lz4Filename, decFilename);
		ret = decompress_file(inpFp, outFp);
		if (ret) {
			printf("decompress : failed with code %zu\n", ret);
			return ret;
		}
		printf("decompress : done\n");

		fclose(outFp);
		fclose(inpFp);
	}

	/* verify */
	{   FILE* const inpFp = fopen(inpFilename, "rb");
		FILE* const decFp = fopen(decFilename, "rb");

		printf("verify : %s <-> %s\n", inpFilename, decFilename);
		const int cmp = compare(inpFp, decFp);
		if(0 == cmp) {
			printf("verify : OK\n");
		} else {
			printf("verify : NG\n");
		}

		fclose(decFp);
		fclose(inpFp);
	}
}