hv_boot_Lzss.c 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. /**
  2. * @file hv_boot_Lzss.c
  3. * @brief Lzss source code.
  4. * @details This file provides the following functions: \n
  5. * (1) Initialization and de-initialization functions \n
  6. * (2) Start and stop functions \n
  7. * (3) Feed functions \n
  8. *
  9. * @author HiView SoC Software Team
  10. * @version 1.0.0
  11. * @date 2022-08-18
  12. * @copyright Copyright(c),2022-8, Hiview Software. All rights reserved.
  13. * @par History:
  14. * <table>
  15. * <tr><th>Author <th>Date <th>Change Description
  16. * <tr><td>HiView SoC Software Team <td>2022-08-23 <td>init
  17. * </table>
  18. */
  19. #include "hv_chip_Config.h"
  20. #include "hv_boot_Common.h"
  21. #define N 4096 /* size of ring buffer */
  22. #define F 18 /* upper limit for match_length */
  23. #define THRESHOLD 4 /* encode string into position and length if match_length is greater than this */
  24. #define NIL N /* index for root of binary search trees */
  25. static unsigned long int textsize = 0; /* text size counter */
  26. static unsigned long int codesize = 0; /* code size counter */
  27. static unsigned long int printcount = 0; /* counter for reporting progress every 1K bytes */
  28. static unsigned char text_buf[N + F - 1] = {0}; /* ring buffer of size N,with extra F-1 bytes to facilitate string comparison */
  29. static int match_position = 0;
  30. static int match_length = 0; /* of longest match. These are set by the InsertNode() procedure. */
  31. static int lson[N + 1] = {0};
  32. static int rson[N + 257] = {0};
  33. static int dad[N + 1] = {0}; /* left & right children &parents -- These constitute binary search trees. */
  34. /* input & output files */
  35. static unsigned char *infile = 0;
  36. static unsigned char *fileoff = 0;
  37. static unsigned char *outfile = 0;
  38. void InitTree(void) /* initialize trees */
  39. {
  40. int i;
  41. /* For i = 0 to N - 1, rson[i] and lson[i] will be the right and
  42. left children of node i. These nodes need not be initialized.
  43. Also, dad[i] is the parent of node i. These are initialized to
  44. NIL (= N), which stands for 'not used.'
  45. For i = 0 to 255, rson[N + i + 1] is the root of the tree
  46. for strings that begin with character i. These are initialized
  47. to NIL. Note there are 256 trees. */
  48. for (i = N + 1; i <= N + 256; i++) rson[i] = NIL;
  49. for (i = 0; i < N; i++) dad[i] = NIL;
  50. }
  51. void InsertNode(int r)
  52. /* Inserts string of length F, text_buf[r..r+F-1], into one of the
  53. trees (text_buf[r]'th tree) and returns the longest-match position
  54. and length via the global variables match_position and match_length.
  55. If match_length = F, then removes the old node in favor of the new
  56. one, because the old one will be deleted sooner.
  57. Note r plays double role, as tree node and position in buffer. */
  58. {
  59. int i, p, cmp;
  60. unsigned char *key;
  61. cmp = 1; key = &text_buf[r]; p = N + 1 + key[0];
  62. rson[r] = lson[r] = NIL; match_length = 0;
  63. for ( ; ; ) {
  64. if (cmp >= 0) {
  65. if (rson[p] != NIL) p = rson[p];
  66. else { rson[p] = r; dad[r] = p; return; }
  67. } else {
  68. if (lson[p] != NIL) p = lson[p];
  69. else { lson[p] = r; dad[r] = p; return; }
  70. }
  71. for (i = 1; i < F; i++)
  72. if ((cmp = key[i] - text_buf[p + i]) != 0) break;
  73. if (i > match_length) {
  74. match_position = p;
  75. if ((match_length = i) >= F) break;
  76. }
  77. }
  78. dad[r] = dad[p]; lson[r] = lson[p]; rson[r] = rson[p];
  79. dad[lson[p]] = r; dad[rson[p]] = r;
  80. if (rson[dad[p]] == p) rson[dad[p]] = r;
  81. else lson[dad[p]] = r;
  82. dad[p] = NIL; /* remove p */
  83. }
  84. void DeleteNode(int p) /* deletes node p from tree */
  85. {
  86. int q;
  87. if (dad[p] == NIL) return; /* not in tree */
  88. if (rson[p] == NIL) q = lson[p];
  89. else if (lson[p] == NIL) q = rson[p];
  90. else {
  91. q = lson[p];
  92. if (rson[q] != NIL) {
  93. do { q = rson[q]; } while (rson[q] != NIL);
  94. rson[dad[q]] = lson[q]; dad[lson[q]] = dad[q];
  95. lson[q] = lson[p]; dad[lson[p]] = q;
  96. }
  97. rson[q] = rson[p]; dad[rson[p]] = q;
  98. }
  99. dad[q] = dad[p];
  100. if (rson[dad[p]] == p) rson[dad[p]] = q; else lson[dad[p]] = q;
  101. dad[p] = NIL;
  102. }
  103. void Decode(void) /* Just the reverse of Encode(). */
  104. {
  105. int i, j, k, r, c;
  106. unsigned int flags;
  107. for (i = 0; i < N - F; i++) text_buf[i] = ' ';
  108. r = N - F; flags = 0;
  109. for ( ; ; ) {
  110. if (((flags >>= 1) & 256) == 0) {
  111. if (infile==fileoff)
  112. {
  113. break;
  114. }
  115. c = *infile++;
  116. flags = c | 0xff00; /* uses higher byte cleverly */
  117. } /* to count eight */
  118. if (flags & 1) {
  119. if (infile==fileoff)
  120. {
  121. break;
  122. }
  123. c = *infile++;
  124. *outfile++=c; text_buf[r++] = c; r &= (N - 1);
  125. } else {
  126. if (infile==fileoff)
  127. {
  128. break;
  129. }
  130. i = *infile++;
  131. if (infile==fileoff)
  132. {
  133. break;
  134. }
  135. j = *infile++;
  136. i |= ((j & 0xf0) << 4); j = (j & 0x0f) + THRESHOLD;
  137. for (k = 0; k <= j; k++) {
  138. c = text_buf[(i + k) & (N - 1)];
  139. *outfile++=c; text_buf[r++] = c; r &= (N - 1);
  140. }
  141. }
  142. }
  143. }
  144. int Hv_Boot_GetFileSize(void)
  145. {
  146. char *pcPoint = infile;
  147. int uiSize = 0;
  148. int iCount = 0;
  149. unsigned int uiMaigic = 0;
  150. Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> check size in\n");
  151. Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_INFO, (unsigned int)infile);
  152. /* check magic "HIVIEW!" */
  153. for (pcPoint = infile, iCount = 0; iCount < 4; iCount++, pcPoint++)
  154. {
  155. uiMaigic = (uiMaigic << 8) + (*pcPoint);
  156. }
  157. if (uiMaigic != 0x48495649)
  158. {
  159. Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_WAR, ">>> no hivi\n");
  160. return -1;
  161. }
  162. for (pcPoint = infile + 4, iCount = 0; iCount < 4; iCount++, pcPoint++)
  163. {
  164. uiMaigic = (uiMaigic << 8) + (*pcPoint);
  165. }
  166. if (uiMaigic != 0x4557210A)
  167. {
  168. Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_WAR, ">>> no iew!\n");
  169. return -2;
  170. }
  171. /* check size */
  172. for (pcPoint = infile + 8, iCount = 0;
  173. (*pcPoint != 0xa) && (*pcPoint < 0x3a) && (*pcPoint > 0x2f) && (iCount < 8);
  174. pcPoint++,iCount++)
  175. {
  176. uiSize = uiSize * 10 + (*pcPoint - 0x30);
  177. }
  178. if ((uiSize > 512) && (uiSize < 1458716))
  179. {
  180. infile += 32;
  181. fileoff = infile + uiSize;
  182. Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_INFO, uiSize);
  183. return 0;
  184. }
  185. Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_WAR, ">>> size wrong!\n");
  186. return -3;
  187. }
  188. int Hv_Boot_LzssDecompress(char *pcSrc, char *pcDst, int iChanegeEndian, int iSize)
  189. {
  190. int iLoop = 0;
  191. unsigned char ucTemp = 0;
  192. if (pcSrc != 0)
  193. {
  194. infile = pcSrc;
  195. }
  196. if (infile == 0)
  197. {
  198. return -1;
  199. }
  200. if (pcDst == 0)
  201. {
  202. return -1;
  203. }
  204. outfile = pcDst;
  205. if (Hv_Boot_GetFileSize())
  206. {
  207. if (iSize == 0)
  208. {
  209. return -1;
  210. } else {
  211. fileoff = infile + iSize;
  212. Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> valid size\n");
  213. Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_INFO, (unsigned int)iSize);
  214. }
  215. }
  216. Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> decompress bin\n");
  217. Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_INFO, (unsigned int)infile);
  218. Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_INFO, (unsigned int)fileoff);
  219. Decode();
  220. Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> decompress bin over\n");
  221. Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_INFO, (unsigned int)pcDst);
  222. Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_INFO, (unsigned int)outfile);
  223. if (iChanegeEndian == 0)
  224. {
  225. return 0;
  226. }
  227. for (iLoop = 0; iLoop < (fileoff - infile); iLoop += 4)
  228. {
  229. ucTemp = *(infile+iLoop+3);
  230. *(infile+iLoop+3) = *(infile+iLoop+0);
  231. *(infile+iLoop+0) = ucTemp;
  232. ucTemp = *(infile+iLoop+2);
  233. *(infile+iLoop+2) = *(infile+iLoop+1);
  234. *(infile+iLoop+1) = ucTemp;
  235. }
  236. return 0;
  237. }