spress.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. #include "spress.h"
  2. static spress_header_t spress_hdr = {
  3. .magic = SPRESS_HEADER_MAGIC,
  4. .major_version = 0,
  5. .minor_version = 1,
  6. .file_hdr_sz = sizeof(spress_header_t),
  7. .chunk_hdr_sz = sizeof(chunk_header_t),
  8. .chunk_sz = 0,
  9. .compress_sz = 0,
  10. .decompress_sz = 0,
  11. };
  12. size_t spr_size_decompressed(const char *source)
  13. {
  14. spress_header_t* hdr = (spress_header_t*)source;
  15. return hdr->decompress_sz;
  16. }
  17. size_t spr_size_compressed(const char *source)
  18. {
  19. spress_header_t* hdr = (spress_header_t*)source;
  20. return hdr->compress_sz;
  21. }
  22. static char* spr_split_next(const char* src, size_t size, uint32_t* praw) {
  23. uint32_t raw = 0;
  24. uint32_t cnt = 0;
  25. uint32_t* ptr = (uint32_t*)src;
  26. uint32_t val = *ptr;
  27. uint32_t tmp;
  28. if (size > SPRESS_BLK_MAX * SPRESS_BLK_SIZE) {
  29. size = SPRESS_BLK_MAX * SPRESS_BLK_SIZE;
  30. }
  31. while ((char*)ptr < (src + size)) {
  32. tmp = *ptr++;
  33. if (tmp == val) {
  34. cnt++;
  35. if (raw && (cnt >= SPRESS_BLK_MIN)) {
  36. ptr -= cnt;
  37. break;
  38. }
  39. if (!raw && (cnt >= SPRESS_BLK_MAX)) {
  40. break;
  41. }
  42. } else {
  43. if (cnt >= SPRESS_BLK_MIN) {
  44. raw = 0;
  45. ptr --;
  46. break;
  47. }
  48. raw = 1;
  49. cnt = 1;
  50. val = tmp;
  51. }
  52. }
  53. *praw = raw;
  54. return (char*)ptr;
  55. }
  56. static void spr_copy_data(uint32_t* dst, uint32_t* src, uint32_t cnt) {
  57. while (cnt-- > 0) {
  58. *dst++ = *src++;
  59. }
  60. }
  61. static void spr_fill_data(uint32_t* dst, uint32_t dat, uint32_t cnt) {
  62. while (cnt-- > 0) {
  63. *dst++ = dat;
  64. }
  65. }
  66. size_t spr_compress_index(const char* source, char* destination, size_t size)
  67. {
  68. size_t rsize = size;
  69. chunk_header_t* hdr = (chunk_header_t*)(destination + sizeof(spress_header_t));
  70. char* src = (char*)source;
  71. char* next;
  72. uint32_t raw, len;
  73. // write chunk
  74. while (rsize > 0) {
  75. // split chunk
  76. next = spr_split_next(src, rsize, &raw);
  77. len = next - src;
  78. if (raw) {
  79. // raw
  80. hdr->data_sz = (len >> 1) | CHUNK_TYPE_RAW;
  81. } else {
  82. // fill
  83. hdr->data_sz = (len >> 1) | CHUNK_TYPE_FILL;
  84. }
  85. hdr++;
  86. src = next;
  87. rsize -= len;
  88. }
  89. // algin process
  90. hdr = (chunk_header_t*)((char*)hdr + ((SPRESS_BLK_SIZE - ((uintptr_t)hdr & 0x3)) & 0x3));
  91. // write header
  92. spress_hdr.chunk_sz = (uint16_t)((char*)hdr - destination - sizeof(spress_header_t));
  93. return spress_hdr.chunk_sz;
  94. }
  95. size_t spr_compress_data(const char* source, char* destination, size_t size)
  96. {
  97. size_t rsize = size;
  98. chunk_header_t* hdr = (chunk_header_t*)(destination + sizeof(spress_header_t));
  99. char* src = (char*)source;
  100. char* dst = (char*)hdr + spress_hdr.chunk_sz;
  101. uint32_t raw, len;
  102. // write data
  103. while (rsize > 0) {
  104. raw = ((hdr->data_sz & CHUNK_TYPE_MASK) == CHUNK_TYPE_RAW);
  105. len = (hdr->data_sz & ~CHUNK_TYPE_MASK) << 1;
  106. if (len == 0) {
  107. len = SPRESS_BLK_MAX * SPRESS_BLK_SIZE;
  108. }
  109. if (raw) {
  110. // raw
  111. spr_copy_data((uint32_t*)dst, (uint32_t*)src, len / SPRESS_BLK_SIZE);
  112. dst += len;
  113. } else {
  114. // fill
  115. spr_fill_data((uint32_t*)dst, *(uint32_t*)src, 1);
  116. dst += SPRESS_BLK_SIZE;
  117. }
  118. hdr++;
  119. src += len;
  120. rsize -= len;
  121. }
  122. // write header
  123. spress_hdr.compress_sz = (uint32_t)(dst - destination);
  124. spress_hdr.decompress_sz = size;
  125. memcpy(destination, &spress_hdr, sizeof(spress_header_t));
  126. return spress_hdr.compress_sz;
  127. }
  128. size_t spr_compress(const char* source, char* destination, size_t size)
  129. {
  130. size_t rsize = size;
  131. chunk_header_t* hdr = (chunk_header_t*)(destination + sizeof(spress_header_t));
  132. char* src = (char*)source;
  133. char* dst;
  134. char* next;
  135. uint32_t raw, len;
  136. // write chunk
  137. while (rsize > 0) {
  138. // split chunk
  139. next = spr_split_next(src, rsize, &raw);
  140. len = next - src;
  141. if (raw) {
  142. // raw
  143. hdr->data_sz = (len >> 1) | CHUNK_TYPE_RAW;
  144. } else {
  145. // fill
  146. hdr->data_sz = (len >> 1) | CHUNK_TYPE_FILL;
  147. }
  148. hdr++;
  149. src = next;
  150. rsize -= len;
  151. }
  152. // algin process
  153. hdr = (chunk_header_t*)((char*)hdr + ((SPRESS_BLK_SIZE - ((uintptr_t)hdr & 0x3)) & 0x3));
  154. // write header
  155. spress_hdr.chunk_sz = (uint16_t)((char*)hdr - destination - sizeof(spress_header_t));
  156. // write data
  157. rsize = size;
  158. src = (char*)source;
  159. dst = (char*)hdr;
  160. hdr = (chunk_header_t*)(destination + sizeof(spress_header_t));
  161. while (rsize > 0) {
  162. raw = ((hdr->data_sz & CHUNK_TYPE_MASK) == CHUNK_TYPE_RAW);
  163. len = (hdr->data_sz & ~CHUNK_TYPE_MASK) << 1;
  164. if (len == 0) {
  165. len = SPRESS_BLK_MAX * SPRESS_BLK_SIZE;
  166. }
  167. if (raw) {
  168. // raw
  169. spr_copy_data((uint32_t*)dst, (uint32_t*)src, len / SPRESS_BLK_SIZE);
  170. dst += len;
  171. } else {
  172. // fill
  173. spr_fill_data((uint32_t*)dst, *(uint32_t*)src, 1);
  174. dst += SPRESS_BLK_SIZE;
  175. }
  176. hdr++;
  177. src += len;
  178. rsize -= len;
  179. }
  180. // write header
  181. spress_hdr.compress_sz = (uint32_t)(dst - destination);
  182. spress_hdr.decompress_sz = size;
  183. memcpy(destination, &spress_hdr, sizeof(spress_header_t));
  184. return spress_hdr.compress_sz;
  185. }
  186. size_t spr_decompress(const char* source, char* destination)
  187. {
  188. spress_header_t* shdr = (spress_header_t*)source;
  189. size_t rsize = shdr->compress_sz - sizeof(spress_header_t) - shdr->chunk_sz;
  190. chunk_header_t* hdr = (chunk_header_t*)(source + sizeof(spress_header_t));
  191. char* src = (char*)source + sizeof(spress_header_t) + shdr->chunk_sz;
  192. char* dst = destination;
  193. uint16_t raw, len;
  194. // write chunk
  195. while (rsize > 0) {
  196. // parse chunk
  197. raw = ((hdr->data_sz & CHUNK_TYPE_MASK) == CHUNK_TYPE_RAW);
  198. len = (hdr->data_sz & ~CHUNK_TYPE_MASK) << 1;
  199. if (len == 0) {
  200. len = SPRESS_BLK_MAX * SPRESS_BLK_SIZE;
  201. }
  202. if (raw) {
  203. // raw
  204. spr_copy_data((uint32_t*)dst, (uint32_t*)src, len / SPRESS_BLK_SIZE);
  205. dst += len;
  206. } else {
  207. // fill
  208. spr_fill_data((uint32_t*)dst, *(uint32_t*)src, len / SPRESS_BLK_SIZE);
  209. dst += len;
  210. len = SPRESS_BLK_SIZE;
  211. }
  212. hdr++;
  213. src += len;
  214. rsize -= len;
  215. }
  216. return shdr->decompress_sz;
  217. }