1 /* 2 * simple_buffer.c 3 * Copyright : Kyle Harper 4 * License : Follows same licensing as the lz4.c/lz4.h program at any given time. Currently, BSD 2. 5 * Description: Example program to demonstrate the basic usage of the compress/decompress functions within lz4.c/lz4.h. 6 * The functions you'll likely want are LZ4_compress_default and LZ4_decompress_safe. Both of these are documented in 7 * the lz4.h header file; I recommend reading them. 8 */ 9 10 /* Includes, for Power! */ 11 #include "lz4.h" // This is all that is required to expose the prototypes for basic compression and decompression. 12 #include <stdio.h> // For printf() 13 #include <string.h> // For memcmp() 14 #include <stdlib.h> // For exit() 15 16 /* 17 * Easy show-error-and-bail function. 18 */ 19 void run_screaming(const char *message, const int code) { 20 printf("%s\n", message); 21 exit(code); 22 return; 23 } 24 25 26 /* 27 * main 28 */ 29 int main(void) { 30 /* Introduction */ 31 // Below we will have a Compression and Decompression section to demonstrate. 32 // There are a few important notes before we start: 33 // 1) The return codes of LZ4_ functions are important. 34 // Read lz4.h if you're unsure what a given code means. 35 // 2) LZ4 uses char* pointers in all LZ4_ functions. 36 // This is baked into the API and probably not going to change. 37 // If your program uses pointers that are unsigned char*, void*, or otherwise different, 38 // you may need to do some casting or set the right -W compiler flags to ignore those warnings (e.g.: -Wno-pointer-sign). 39 40 /* Compression */ 41 // We'll store some text into a variable pointed to by *src to be compressed later. 42 const char* const src = "Lorem ipsum dolor sit amet, consectetur adipiscing elit."; 43 // The compression function needs to know how many bytes exist. Since we're using a string, we can use strlen() + 1 (for \0). 44 const int src_size = (int)(strlen(src) + 1); 45 // LZ4 provides a function that will tell you the maximum size of compressed output based on input data via LZ4_compressBound(). 46 const int max_dst_size = LZ4_compressBound(src_size); 47 // We will use that size for our destination boundary when allocating space. 48 char* compressed_data = malloc(max_dst_size); 49 if (compressed_data == NULL) 50 run_screaming("Failed to allocate memory for *compressed_data.", 1); 51 // That's all the information and preparation LZ4 needs to compress *src into *compressed_data. 52 // Invoke LZ4_compress_default now with our size values and pointers to our memory locations. 53 // Save the return value for error checking. 54 const int compressed_data_size = LZ4_compress_default(src, compressed_data, src_size, max_dst_size); 55 // Check return_value to determine what happened. 56 if (compressed_data_size < 0) 57 run_screaming("A negative result from LZ4_compress_default indicates a failure trying to compress the data. See exit code (echo $?) for value returned.", compressed_data_size); 58 if (compressed_data_size == 0) 59 run_screaming("A result of 0 means compression worked, but was stopped because the destination buffer couldn't hold all the information.", 1); 60 if (compressed_data_size > 0) 61 printf("We successfully compressed some data!\n"); 62 // Not only does a positive return_value mean success, the value returned == the number of bytes required. 63 // You can use this to realloc() *compress_data to free up memory, if desired. We'll do so just to demonstrate the concept. 64 compressed_data = (char *)realloc(compressed_data, compressed_data_size); 65 if (compressed_data == NULL) 66 run_screaming("Failed to re-alloc memory for compressed_data. Sad :(", 1); 67 68 /* Decompression */ 69 // Now that we've successfully compressed the information from *src to *compressed_data, let's do the opposite! We'll create a 70 // *new_src location of size src_size since we know that value. 71 char* const regen_buffer = malloc(src_size); 72 if (regen_buffer == NULL) 73 run_screaming("Failed to allocate memory for *regen_buffer.", 1); 74 // The LZ4_decompress_safe function needs to know where the compressed data is, how many bytes long it is, 75 // where the regen_buffer memory location is, and how large regen_buffer (uncompressed) output will be. 76 // Again, save the return_value. 77 const int decompressed_size = LZ4_decompress_safe(compressed_data, regen_buffer, compressed_data_size, src_size); 78 free(compressed_data); /* no longer useful */ 79 if (decompressed_size < 0) 80 run_screaming("A negative result from LZ4_decompress_safe indicates a failure trying to decompress the data. See exit code (echo $?) for value returned.", decompressed_size); 81 if (decompressed_size == 0) 82 run_screaming("I'm not sure this function can ever return 0. Documentation in lz4.h doesn't indicate so.", 1); 83 if (decompressed_size > 0) 84 printf("We successfully decompressed some data!\n"); 85 // Not only does a positive return value mean success, 86 // value returned == number of bytes regenerated from compressed_data stream. 87 88 /* Validation */ 89 // We should be able to compare our original *src with our *new_src and be byte-for-byte identical. 90 if (memcmp(src, regen_buffer, src_size) != 0) 91 run_screaming("Validation failed. *src and *new_src are not identical.", 1); 92 printf("Validation done. The string we ended up with is:\n%s\n", regen_buffer); 93 return 0; 94 } 95