This allows readFile() to indicate that a file doesn't exist, and might eliminate some large string copying.
		
			
				
	
	
		
			96 lines
		
	
	
	
		
			2.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			96 lines
		
	
	
	
		
			2.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "compression.hh"
 | |
| #include "types.hh"
 | |
| 
 | |
| #include <lzma.h>
 | |
| #include <cstdio>
 | |
| 
 | |
| namespace nix {
 | |
| 
 | |
| /* RAII wrapper around lzma_stream. */
 | |
| struct LzmaStream
 | |
| {
 | |
|     lzma_stream strm;
 | |
|     LzmaStream() : strm(LZMA_STREAM_INIT) { };
 | |
|     ~LzmaStream() { lzma_end(&strm); };
 | |
|     lzma_stream & operator()() { return strm; }
 | |
| };
 | |
| 
 | |
| std::string compressXZ(const std::string & in)
 | |
| {
 | |
|     LzmaStream strm;
 | |
| 
 | |
|     // FIXME: apply the x86 BCJ filter?
 | |
| 
 | |
|     lzma_ret ret = lzma_easy_encoder(
 | |
|         &strm(), 6, LZMA_CHECK_CRC64);
 | |
|     if (ret != LZMA_OK)
 | |
|         throw Error("unable to initialise lzma encoder");
 | |
| 
 | |
|     lzma_action action = LZMA_RUN;
 | |
|     uint8_t outbuf[BUFSIZ];
 | |
|     string res;
 | |
|     strm().next_in = (uint8_t *) in.c_str();
 | |
|     strm().avail_in = in.size();
 | |
|     strm().next_out = outbuf;
 | |
|     strm().avail_out = sizeof(outbuf);
 | |
| 
 | |
|     while (true) {
 | |
| 
 | |
|         if (strm().avail_in == 0)
 | |
|             action = LZMA_FINISH;
 | |
| 
 | |
|         lzma_ret ret = lzma_code(&strm(), action);
 | |
| 
 | |
|         if (strm().avail_out == 0 || ret == LZMA_STREAM_END) {
 | |
|             res.append((char *) outbuf, sizeof(outbuf) - strm().avail_out);
 | |
|             strm().next_out = outbuf;
 | |
|             strm().avail_out = sizeof(outbuf);
 | |
|         }
 | |
| 
 | |
|         if (ret == LZMA_STREAM_END)
 | |
|             return res;
 | |
| 
 | |
|         if (ret != LZMA_OK)
 | |
|             throw Error("error while decompressing xz file");
 | |
|     }
 | |
| }
 | |
| 
 | |
| ref<std::string> decompressXZ(const std::string & in)
 | |
| {
 | |
|     LzmaStream strm;
 | |
| 
 | |
|     lzma_ret ret = lzma_stream_decoder(
 | |
|         &strm(), UINT64_MAX, LZMA_CONCATENATED);
 | |
|     if (ret != LZMA_OK)
 | |
|         throw Error("unable to initialise lzma decoder");
 | |
| 
 | |
|     lzma_action action = LZMA_RUN;
 | |
|     uint8_t outbuf[BUFSIZ];
 | |
|     ref<std::string> res = make_ref<std::string>();
 | |
|     strm().next_in = (uint8_t *) in.c_str();
 | |
|     strm().avail_in = in.size();
 | |
|     strm().next_out = outbuf;
 | |
|     strm().avail_out = sizeof(outbuf);
 | |
| 
 | |
|     while (true) {
 | |
| 
 | |
|         if (strm().avail_in == 0)
 | |
|             action = LZMA_FINISH;
 | |
| 
 | |
|         lzma_ret ret = lzma_code(&strm(), action);
 | |
| 
 | |
|         if (strm().avail_out == 0 || ret == LZMA_STREAM_END) {
 | |
|             res->append((char *) outbuf, sizeof(outbuf) - strm().avail_out);
 | |
|             strm().next_out = outbuf;
 | |
|             strm().avail_out = sizeof(outbuf);
 | |
|         }
 | |
| 
 | |
|         if (ret == LZMA_STREAM_END)
 | |
|             return res;
 | |
| 
 | |
|         if (ret != LZMA_OK)
 | |
|             throw Error("error while decompressing xz file");
 | |
|     }
 | |
| }
 | |
| 
 | |
| }
 |