explode.cpp
上传用户:yisoukefu
上传日期:2020-08-09
资源大小:39506k
文件大小:15k
源码类别:

其他游戏

开发平台:

Visual C++

  1. #include <windows.h>
  2. #include <assert.h>
  3. #include <string.h>
  4. #include "mpq.h"
  5. #include "explode.h"
  6. /* Tables */
  7. static unsigned char pkzip_dist_bits[] = {
  8. 0x02, 0x04, 0x04, 0x05, 0x05, 0x05, 0x05, 0x06, 0x06, 0x06, 0x06, 0x06, 0x06, 0x06, 0x06, 0x06,
  9. 0x06, 0x06, 0x06, 0x06, 0x06, 0x06, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07,
  10. 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07, 0x07,
  11. 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08
  12. };
  13. static unsigned char pkzip_dist_code[] = {
  14. 0x03, 0x0D, 0x05, 0x19, 0x09, 0x11, 0x01, 0x3E, 0x1E, 0x2E, 0x0E, 0x36, 0x16, 0x26, 0x06, 0x3A,
  15. 0x1A, 0x2A, 0x0A, 0x32, 0x12, 0x22, 0x42, 0x02, 0x7C, 0x3C, 0x5C, 0x1C, 0x6C, 0x2C, 0x4C, 0x0C,
  16. 0x74, 0x34, 0x54, 0x14, 0x64, 0x24, 0x44, 0x04, 0x78, 0x38, 0x58, 0x18, 0x68, 0x28, 0x48, 0x08,
  17. 0xF0, 0x70, 0xB0, 0x30, 0xD0, 0x50, 0x90, 0x10, 0xE0, 0x60, 0xA0, 0x20, 0xC0, 0x40, 0x80, 0x00
  18. };
  19. static unsigned char pkzip_clen_bits[] = {
  20. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08
  21. };
  22. static unsigned short pkzip_len_base[] = {
  23. 0x0000, 0x0001, 0x0002, 0x0003, 0x0004, 0x0005, 0x0006, 0x0007,
  24. 0x0008, 0x000A, 0x000E, 0x0016, 0x0026, 0x0046, 0x0086, 0x0106
  25. };
  26. static unsigned char pkzip_slen_bits[] = {
  27. 0x03, 0x02, 0x03, 0x03, 0x04, 0x04, 0x04, 0x05, 0x05, 0x05, 0x05, 0x06, 0x06, 0x06, 0x07, 0x07
  28. };
  29. static unsigned char pkzip_len_code[] = {
  30. 0x05, 0x03, 0x01, 0x06, 0x0A, 0x02, 0x0C, 0x14, 0x04, 0x18, 0x08, 0x30, 0x10, 0x20, 0x40, 0x00
  31. };
  32. static unsigned char pkzip_bits_asc[] = {
  33. 0x0B, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x08, 0x07, 0x0C, 0x0C, 0x07, 0x0C, 0x0C,
  34. 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0D, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C,
  35. 0x04, 0x0A, 0x08, 0x0C, 0x0A, 0x0C, 0x0A, 0x08, 0x07, 0x07, 0x08, 0x09, 0x07, 0x06, 0x07, 0x08,
  36. 0x07, 0x06, 0x07, 0x07, 0x07, 0x07, 0x08, 0x07, 0x07, 0x08, 0x08, 0x0C, 0x0B, 0x07, 0x09, 0x0B,
  37. 0x0C, 0x06, 0x07, 0x06, 0x06, 0x05, 0x07, 0x08, 0x08, 0x06, 0x0B, 0x09, 0x06, 0x07, 0x06, 0x06,
  38. 0x07, 0x0B, 0x06, 0x06, 0x06, 0x07, 0x09, 0x08, 0x09, 0x09, 0x0B, 0x08, 0x0B, 0x09, 0x0C, 0x08,
  39. 0x0C, 0x05, 0x06, 0x06, 0x06, 0x05, 0x06, 0x06, 0x06, 0x05, 0x0B, 0x07, 0x05, 0x06, 0x05, 0x05,
  40. 0x06, 0x0A, 0x05, 0x05, 0x05, 0x05, 0x08, 0x07, 0x08, 0x08, 0x0A, 0x0B, 0x0B, 0x0C, 0x0C, 0x0C,
  41. 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D,
  42. 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D,
  43. 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D,
  44. 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C,
  45. 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C,
  46. 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C,
  47. 0x0D, 0x0C, 0x0D, 0x0D, 0x0D, 0x0C, 0x0D, 0x0D, 0x0D, 0x0C, 0x0D, 0x0D, 0x0D, 0x0D, 0x0C, 0x0D,
  48. 0x0D, 0x0D, 0x0C, 0x0C, 0x0C, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D, 0x0D
  49. };
  50. static unsigned short pkzip_code_asc[] = {
  51. 0x0490, 0x0FE0, 0x07E0, 0x0BE0, 0x03E0, 0x0DE0, 0x05E0, 0x09E0,
  52. 0x01E0, 0x00B8, 0x0062, 0x0EE0, 0x06E0, 0x0022, 0x0AE0, 0x02E0,
  53. 0x0CE0, 0x04E0, 0x08E0, 0x00E0, 0x0F60, 0x0760, 0x0B60, 0x0360,
  54. 0x0D60, 0x0560, 0x1240, 0x0960, 0x0160, 0x0E60, 0x0660, 0x0A60,
  55. 0x000F, 0x0250, 0x0038, 0x0260, 0x0050, 0x0C60, 0x0390, 0x00D8,
  56. 0x0042, 0x0002, 0x0058, 0x01B0, 0x007C, 0x0029, 0x003C, 0x0098,
  57. 0x005C, 0x0009, 0x001C, 0x006C, 0x002C, 0x004C, 0x0018, 0x000C,
  58. 0x0074, 0x00E8, 0x0068, 0x0460, 0x0090, 0x0034, 0x00B0, 0x0710,
  59. 0x0860, 0x0031, 0x0054, 0x0011, 0x0021, 0x0017, 0x0014, 0x00A8,
  60. 0x0028, 0x0001, 0x0310, 0x0130, 0x003E, 0x0064, 0x001E, 0x002E,
  61. 0x0024, 0x0510, 0x000E, 0x0036, 0x0016, 0x0044, 0x0030, 0x00C8,
  62. 0x01D0, 0x00D0, 0x0110, 0x0048, 0x0610, 0x0150, 0x0060, 0x0088,
  63. 0x0FA0, 0x0007, 0x0026, 0x0006, 0x003A, 0x001B, 0x001A, 0x002A,
  64. 0x000A, 0x000B, 0x0210, 0x0004, 0x0013, 0x0032, 0x0003, 0x001D,
  65. 0x0012, 0x0190, 0x000D, 0x0015, 0x0005, 0x0019, 0x0008, 0x0078,
  66. 0x00F0, 0x0070, 0x0290, 0x0410, 0x0010, 0x07A0, 0x0BA0, 0x03A0,
  67. 0x0240, 0x1C40, 0x0C40, 0x1440, 0x0440, 0x1840, 0x0840, 0x1040,
  68. 0x0040, 0x1F80, 0x0F80, 0x1780, 0x0780, 0x1B80, 0x0B80, 0x1380,
  69. 0x0380, 0x1D80, 0x0D80, 0x1580, 0x0580, 0x1980, 0x0980, 0x1180,
  70. 0x0180, 0x1E80, 0x0E80, 0x1680, 0x0680, 0x1A80, 0x0A80, 0x1280,
  71. 0x0280, 0x1C80, 0x0C80, 0x1480, 0x0480, 0x1880, 0x0880, 0x1080,
  72. 0x0080, 0x1F00, 0x0F00, 0x1700, 0x0700, 0x1B00, 0x0B00, 0x1300,
  73. 0x0DA0, 0x05A0, 0x09A0, 0x01A0, 0x0EA0, 0x06A0, 0x0AA0, 0x02A0,
  74. 0x0CA0, 0x04A0, 0x08A0, 0x00A0, 0x0F20, 0x0720, 0x0B20, 0x0320,
  75. 0x0D20, 0x0520, 0x0920, 0x0120, 0x0E20, 0x0620, 0x0A20, 0x0220,
  76. 0x0C20, 0x0420, 0x0820, 0x0020, 0x0FC0, 0x07C0, 0x0BC0, 0x03C0,
  77. 0x0DC0, 0x05C0, 0x09C0, 0x01C0, 0x0EC0, 0x06C0, 0x0AC0, 0x02C0,
  78. 0x0CC0, 0x04C0, 0x08C0, 0x00C0, 0x0F40, 0x0740, 0x0B40, 0x0340,
  79. 0x0300, 0x0D40, 0x1D00, 0x0D00, 0x1500, 0x0540, 0x0500, 0x1900,
  80. 0x0900, 0x0940, 0x1100, 0x0100, 0x1E00, 0x0E00, 0x0140, 0x1600,
  81. 0x0600, 0x1A00, 0x0E40, 0x0640, 0x0A40, 0x0A00, 0x1200, 0x0200,
  82. 0x1C00, 0x0C00, 0x1400, 0x0400, 0x1800, 0x0800, 0x1000, 0x0000  
  83. };
  84. /* Local variables */
  85. static char copyright[] = "PKWARE Data Compression Library for Win32rn"
  86.                           "Copyright 1989-1995 PKWARE Inc.  All Rights Reservedrn"
  87.                           "Patent No. 5,051,745rn"
  88.                           "PKWARE Data Compression Library Reg. U.S. Pat. and Tm. Off.rn"
  89.                           "Version 1.11rn";
  90. /* Local functions */
  91. static void libmpq_pkzip_gen_decode_tabs(long count, unsigned char *bits, unsigned char *code, unsigned char *buf2) {
  92. long i;
  93. for (i = count-1; i >= 0; i--) { /* EBX - count */
  94. unsigned long idx1 = code[i];
  95. unsigned long idx2 = 1 << bits[i];
  96. do {
  97. buf2[idx1] = (unsigned char)i;
  98. idx1      += idx2;
  99. } while (idx1 < 0x100);
  100. }
  101. }
  102. static void libmpq_pkzip_gen_asc_tabs(pkzip_data_cmp *mpq_pkzip) {
  103. unsigned short *code_asc = &pkzip_code_asc[0xFF];
  104. unsigned long acc, add;
  105. unsigned short count;
  106. for (count = 0x00FF; code_asc >= pkzip_code_asc; code_asc--, count--) {
  107. unsigned char *bits_asc = mpq_pkzip->bits_asc + count;
  108. unsigned char bits_tmp = *bits_asc;
  109. if (bits_tmp <= 8) {
  110. add = (1 << bits_tmp);
  111. acc = *code_asc;
  112. do {
  113. mpq_pkzip->offs_2c34[acc] = (unsigned char)count;
  114. acc += add;
  115. } while (acc < 0x100);
  116. } else {
  117. if ((acc = (*code_asc & 0xFF)) != 0) {
  118. mpq_pkzip->offs_2c34[acc] = 0xFF;
  119. if (*code_asc & 0x3F) {
  120. bits_tmp -= 4;
  121. *bits_asc = bits_tmp;
  122. add = (1 << bits_tmp);
  123. acc = *code_asc >> 4;
  124. do {
  125. mpq_pkzip->offs_2d34[acc] = (unsigned char)count;
  126. acc += add;
  127. } while (acc < 0x100);
  128. } else {
  129. bits_tmp -= 6;
  130. *bits_asc = bits_tmp;
  131. add = (1 << bits_tmp);
  132. acc = *code_asc >> 6;
  133. do {
  134. mpq_pkzip->offs_2e34[acc] = (unsigned char)count;
  135. acc += add;
  136. } while (acc < 0x80);
  137. }
  138. } else {
  139. bits_tmp -= 8;
  140. *bits_asc = bits_tmp;
  141. add = (1 << bits_tmp);
  142. acc = *code_asc >> 8;
  143. do {
  144. mpq_pkzip->offs_2eb4[acc] = (unsigned char)count;
  145. acc += add;
  146. } while (acc < 0x100);
  147. }
  148. }
  149. }
  150. }
  151. /*
  152.  *  Skips given number of bits in bit buffer. Result is stored in mpq_pkzip->bit_buf
  153.  *  If no data in input buffer, returns true
  154.  */
  155. static int libmpq_pkzip_skip_bits(pkzip_data_cmp *mpq_pkzip, unsigned long bits) {
  156. /* If number of bits required is less than number of (bits in the buffer) ? */
  157. if (bits <= mpq_pkzip->extra_bits) {
  158. mpq_pkzip->extra_bits -= bits;
  159. mpq_pkzip->bit_buf >>= bits;
  160. return 0;
  161. }
  162. /* Load input buffer if necessary */
  163. mpq_pkzip->bit_buf >>= mpq_pkzip->extra_bits;
  164. if (mpq_pkzip->in_pos == mpq_pkzip->in_bytes) {
  165. mpq_pkzip->in_pos = sizeof(mpq_pkzip->in_buf);
  166. if ((mpq_pkzip->in_bytes = mpq_pkzip->read_buf((char *)mpq_pkzip->in_buf, &mpq_pkzip->in_pos, mpq_pkzip->param)) == 0) {
  167. return 1;
  168. }
  169. mpq_pkzip->in_pos = 0;
  170. }
  171. /* Update bit buffer */
  172. mpq_pkzip->bit_buf |= (mpq_pkzip->in_buf[mpq_pkzip->in_pos++] << 8);
  173. mpq_pkzip->bit_buf >>= (bits - mpq_pkzip->extra_bits);
  174. mpq_pkzip->extra_bits = (mpq_pkzip->extra_bits - bits) + 8;
  175. return 0;
  176. }
  177. /*
  178.  *  Decompress the imploded data using coded literals.
  179.  *  Returns: 0x000 - 0x0FF : One byte from compressed file.
  180.  *           0x100 - 0x305 : Copy previous block (0x100 = 1 byte)
  181.  *           0x306         : Out of buffer (?)
  182.  */
  183. static unsigned long libmpq_pkzip_explode_lit(pkzip_data_cmp *mpq_pkzip) {
  184. unsigned long bits; /* Number of bits to skip */
  185. unsigned long value; /* Position in buffers */
  186. /* Test the current bit in byte buffer. If is not set, simply return the next byte. */
  187. if (mpq_pkzip->bit_buf & 1) {
  188. /* Skip current bit in the buffer. */
  189. if (libmpq_pkzip_skip_bits(mpq_pkzip, 1)) {
  190. return 0x306;
  191. }
  192. /* The next bits are position in buffers. */
  193. value = mpq_pkzip->pos2[(mpq_pkzip->bit_buf & 0xFF)];
  194. /* Get number of bits to skip */
  195. if (libmpq_pkzip_skip_bits(mpq_pkzip, mpq_pkzip->slen_bits[value])) {
  196. return 0x306;
  197. }
  198. if ((bits = mpq_pkzip->clen_bits[value]) != 0) {
  199. unsigned long val2 = mpq_pkzip->bit_buf & ((1 << bits) - 1);
  200. if (libmpq_pkzip_skip_bits(mpq_pkzip, bits)) {
  201. if ((value + val2) != 0x10E) {
  202. return 0x306;
  203. }
  204. }
  205. value = mpq_pkzip->len_base[value] + val2;
  206. }
  207. return value + 0x100; /* Return number of bytes to repeat */
  208. }
  209. /* Skip one bit */
  210. if (libmpq_pkzip_skip_bits(mpq_pkzip, 1)) {
  211. return 0x306;
  212. }
  213. /* If the binary compression type, read 8 bits and return them as one byte. */
  214. if (mpq_pkzip->cmp_type == LIBMPQ_PKZIP_CMP_BINARY) {
  215. value = mpq_pkzip->bit_buf & 0xFF;
  216. if (libmpq_pkzip_skip_bits(mpq_pkzip, 8)) {
  217. return 0x306;
  218. }
  219. return value;
  220. }
  221. /* When ASCII compression ... */
  222. if (mpq_pkzip->bit_buf & 0xFF) {
  223. value = mpq_pkzip->offs_2c34[mpq_pkzip->bit_buf & 0xFF];
  224. if (value == 0xFF) {
  225. if (mpq_pkzip->bit_buf & 0x3F) {
  226. if (libmpq_pkzip_skip_bits(mpq_pkzip, 4)) {
  227. return 0x306;
  228. }
  229. value = mpq_pkzip->offs_2d34[mpq_pkzip->bit_buf & 0xFF];
  230. } else {
  231. if (libmpq_pkzip_skip_bits(mpq_pkzip, 6)) {
  232. return 0x306;
  233. }
  234. value = mpq_pkzip->offs_2e34[mpq_pkzip->bit_buf & 0x7F];
  235. }
  236. }
  237. } else {
  238. if (libmpq_pkzip_skip_bits(mpq_pkzip, 8)) {
  239. return 0x306;
  240. }
  241. value = mpq_pkzip->offs_2eb4[mpq_pkzip->bit_buf & 0xFF];
  242. }
  243. return libmpq_pkzip_skip_bits(mpq_pkzip, mpq_pkzip->bits_asc[value]) ? 0x306 : value;
  244. }
  245. /*
  246.  *  Retrieves the number of bytes to move back.
  247.  */
  248. static unsigned long libmpq_pkzip_explode_dist(pkzip_data_cmp *mpq_pkzip, unsigned long length) {
  249. unsigned long pos  = mpq_pkzip->pos1[(mpq_pkzip->bit_buf & 0xFF)];
  250. unsigned long skip = mpq_pkzip->dist_bits[pos]; /* Number of bits to skip */
  251. /* Skip the appropriate number of bits */
  252. if (libmpq_pkzip_skip_bits(mpq_pkzip, skip) == 1) {
  253. return 0;
  254. }
  255. if (length == 2) {
  256. pos = (pos << 2) | (mpq_pkzip->bit_buf & 0x03);
  257. if (libmpq_pkzip_skip_bits(mpq_pkzip, 2) == 1) {
  258. return 0;
  259. }
  260. } else {
  261. pos = (pos << mpq_pkzip->dsize_bits) | (mpq_pkzip->bit_buf & mpq_pkzip->dsize_mask);
  262. /* Skip the bits */
  263. if (libmpq_pkzip_skip_bits(mpq_pkzip, mpq_pkzip->dsize_bits) == 1) {
  264. return 0;
  265. }
  266. }
  267. return pos + 1;
  268. }
  269. static unsigned long libmpq_pkzip_expand(pkzip_data_cmp *mpq_pkzip) {
  270. unsigned int copy_bytes; /* Number of bytes to copy */
  271. unsigned long one_byte; /* One byte from compressed file */
  272. unsigned long result;
  273. mpq_pkzip->out_pos = 0x1000; /* Initialize output buffer position */
  274. /* If end of data or error, terminate decompress */
  275. while ((result = one_byte = libmpq_pkzip_explode_lit(mpq_pkzip)) < 0x305) {
  276. /* If one byte is greater than 0x100, means "Repeat n - 0xFE bytes" */
  277. if (one_byte >= 0x100) {
  278. unsigned char *source; /* ECX */
  279. unsigned char *target; /* EDX */
  280. unsigned long copy_length = one_byte - 0xFE;
  281. unsigned long move_back;
  282. /* Get length of data to copy */
  283. if ((move_back = libmpq_pkzip_explode_dist(mpq_pkzip, copy_length)) == 0) {
  284. result = 0x306;
  285. break;
  286. }
  287. /* Target and source pointer */
  288. target = &mpq_pkzip->out_buf[mpq_pkzip->out_pos];
  289. source = target - move_back;
  290. mpq_pkzip->out_pos += copy_length;
  291. while (copy_length-- > 0) {
  292. *target++ = *source++;
  293. }
  294. } else {
  295. mpq_pkzip->out_buf[mpq_pkzip->out_pos++] = (unsigned char)one_byte;
  296. }
  297. /*
  298.  * If number of extracted bytes has reached 1/2 of output buffer,
  299.  * flush output buffer.
  300.  */
  301. if (mpq_pkzip->out_pos >= 0x2000) {
  302. /* Copy decompressed data into user buffer. */
  303. copy_bytes = 0x1000;
  304. mpq_pkzip->write_buf((char *)&mpq_pkzip->out_buf[0x1000], &copy_bytes, mpq_pkzip->param);
  305. /* If there are some data left, keep them alive */
  306. memcpy(mpq_pkzip->out_buf, &mpq_pkzip->out_buf[0x1000], mpq_pkzip->out_pos - 0x1000);
  307. mpq_pkzip->out_pos -= 0x1000;
  308. }
  309. }
  310. copy_bytes = mpq_pkzip->out_pos - 0x1000;
  311. mpq_pkzip->write_buf((char *)&mpq_pkzip->out_buf[0x1000], &copy_bytes, mpq_pkzip->param);
  312. return result;
  313. }
  314. /*
  315.  * Main exploding function.
  316.  */
  317. unsigned int libmpq_pkzip_explode(
  318. unsigned int (*read_buf)(char *buf, unsigned  int *size, void *param),
  319. void (*write_buf)(char *buf, unsigned  int *size, void *param),
  320. char *work_buf,
  321. void *param) {
  322. pkzip_data_cmp *mpq_pkzip = (pkzip_data_cmp *)work_buf;
  323. /* Set the whole work buffer to zeros */
  324. memset(mpq_pkzip, 0, sizeof(pkzip_data_cmp));
  325. /* Initialize work struct and load compressed data */
  326. mpq_pkzip->read_buf   = read_buf;
  327. mpq_pkzip->write_buf  = write_buf;
  328. mpq_pkzip->param      = param;
  329. mpq_pkzip->in_pos     = sizeof(mpq_pkzip->in_buf);
  330. mpq_pkzip->in_bytes   = mpq_pkzip->read_buf((char *)mpq_pkzip->in_buf, &mpq_pkzip->in_pos, mpq_pkzip->param);
  331. if (mpq_pkzip->in_bytes <= 4) {
  332. return LIBMPQ_PKZIP_CMP_BAD_DATA;
  333. }
  334. mpq_pkzip->cmp_type   = mpq_pkzip->in_buf[0]; /* Get the compression type */
  335. mpq_pkzip->dsize_bits = mpq_pkzip->in_buf[1]; /* Get the dictionary size */
  336. mpq_pkzip->bit_buf    = mpq_pkzip->in_buf[2]; /* Initialize 16-bit bit buffer */
  337. mpq_pkzip->extra_bits = 0; /* Extra (over 8) bits */
  338. mpq_pkzip->in_pos     = 3; /* Position in input buffer */
  339. /* Test for the valid dictionary size */
  340. if (4 > mpq_pkzip->dsize_bits || mpq_pkzip->dsize_bits > 6) {
  341. return LIBMPQ_PKZIP_CMP_INV_DICTSIZE;
  342. }
  343. mpq_pkzip->dsize_mask = 0xFFFF >> (0x10 - mpq_pkzip->dsize_bits); /* Shifted by 'sar' instruction */
  344. if (mpq_pkzip->cmp_type != LIBMPQ_PKZIP_CMP_BINARY) {
  345. if (mpq_pkzip->cmp_type != LIBMPQ_PKZIP_CMP_ASCII) {
  346. return LIBMPQ_PKZIP_CMP_INV_MODE;
  347. }
  348. memcpy(mpq_pkzip->bits_asc, pkzip_bits_asc, sizeof(mpq_pkzip->bits_asc));
  349. libmpq_pkzip_gen_asc_tabs(mpq_pkzip);
  350. }
  351. memcpy(mpq_pkzip->slen_bits, pkzip_slen_bits, sizeof(mpq_pkzip->slen_bits));
  352. libmpq_pkzip_gen_decode_tabs(0x10, mpq_pkzip->slen_bits, pkzip_len_code, mpq_pkzip->pos2);
  353. memcpy(mpq_pkzip->clen_bits, pkzip_clen_bits, sizeof(mpq_pkzip->clen_bits));
  354. memcpy(mpq_pkzip->len_base, pkzip_len_base, sizeof(mpq_pkzip->len_base));
  355. memcpy(mpq_pkzip->dist_bits, pkzip_dist_bits, sizeof(mpq_pkzip->dist_bits));
  356. libmpq_pkzip_gen_decode_tabs(0x40, mpq_pkzip->dist_bits, pkzip_dist_code, mpq_pkzip->pos1);
  357. if (libmpq_pkzip_expand(mpq_pkzip) != 0x306) {
  358. return LIBMPQ_PKZIP_CMP_NO_ERROR;
  359. }
  360. return LIBMPQ_PKZIP_CMP_ABORT;
  361. }