1 /**
2 Copyright: Copyright (c) 2015-2017 Andrey Penechko.
3 License: $(WEB boost.org/LICENSE_1_0.txt, Boost License 1.0).
4 Authors: Andrey Penechko.
5 */
6 
7 module voxelman.utils.rlecompression;
8 
9 import std.range : put;
10 
11 ubyte[] rleEncode(in ubyte[] data, ubyte[] outBuffer)
12 {
13 	if (data.length == 0) return null;
14 
15 	ubyte[] original = outBuffer;
16 	ubyte count = 1;
17 	ubyte current = data[0];
18 
19 	foreach(item; data[1..$])
20 	{
21 		if (current == item && count < ubyte.max)
22 		{
23 			++count;
24 		}
25 		else
26 		{
27 			put(outBuffer, count);
28 			put(outBuffer, current);
29 			count = 1;
30 			current = item;
31 		}
32 	}
33 	put(outBuffer, count);
34 	put(outBuffer, current);
35 
36 	return original[0..$-outBuffer.length];
37 }
38 
39 ubyte[] rleDecode(in ubyte[] data, ubyte[] outBuffer)
40 {
41 	if (data.length == 0) return null;
42 
43 	assert(data.length % 2 == 0);
44 	ubyte[] original = outBuffer;
45 
46 	foreach(i; 0..data.length/2)
47 	{
48 		ubyte count = data[i*2];
49 		ubyte item = data[i*2 + 1];
50 
51 		outBuffer[0..count] = item;
52 		outBuffer = outBuffer[count..$];
53 	}
54 
55 	return original[0..$-outBuffer.length];
56 }
57 
58 unittest
59 {
60 	ubyte[32] data = [0,0,0,0,0,1,1,1,1,2,2,2,2,0,0,0,0,0,1,0,4,0,1,0,0,0,0,0,0,0,0,1];
61 	ubyte[32] outBuffer;
62 	assert(rleEncode(data, outBuffer) == [5,0,4,1,4,2,5,0,1,1,1,0,1,4,1,0,1,1,8,0,1,1]);
63 	assert(rleDecode(rleEncode(data, outBuffer).dup, outBuffer) == data);
64 }