adpthuff.pas 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. CONST
  2. bufSize = 32768;
  3. Type
  4. PTree = ^TTree;
  5. TTree = record
  6. info : byte;
  7. ti : longint;
  8. Left : PTree;
  9. Right: PTree;
  10. end;
  11. ps = ^el;
  12. el = record
  13. data : PTree;
  14. prev : ps;
  15. end;
  16. CONST
  17. fname : string[13] = 'input.txt';
  18. fout : string[13] = 'input.huf';
  19. VAR
  20. INP,OUT : FILE;
  21. USED : set of byte;
  22. Nums : ps;
  23. THead : PTree;
  24. {-----}
  25. Code : longint;
  26. CodeLen : byte;
  27. {-----}
  28. GO,GNum : byte;
  29. done : boolean;
  30. Size,Csize : longint;
  31. {---------------------------------}
  32. var
  33. {FReadBuf}
  34. RBBuf, RBread : ^BYTE;
  35. RBcount : word;
  36. RBsize : word;
  37. RBFirst, RBLast : boolean;
  38. {FReadBuf}
  39. Function FReadBuf(var F:FILE;var val:byte):boolean;
  40. begin
  41. if RBFirst then
  42. begin
  43. RBfirst:=false;
  44. GetMem(RBBuf,BUFSize);
  45. RBcount:=0;
  46. RBSize:=0;
  47. end;
  48. if RBLast then
  49. begin
  50. RBLast:=false;
  51. FreeMem(RBBuf,BufSize);
  52. RBFirst:=true;
  53. exit;
  54. end;
  55. If RBcount=RBsize then
  56. begin
  57. BlockRead(F,RBBuf^,BufSize,RBsize);
  58. if Rbsize=0 then
  59. begin
  60. FReadBuf:=false;
  61. exit;
  62. end;
  63. RBcount:=0;
  64. end;
  65. RBread:=PTR(SeG(RBBUF^),Ofs(RBbuf^)+RBcount);
  66. inc(RBcount);
  67. val:=RBread^;
  68. FReadBuf:=true;
  69. end; {FReadBuf}
  70. {-----------------------------------------}
  71. var
  72. {FWriteBuf}
  73. WBBuf,WBwrt : ^BYTE;
  74. WBcount : word;
  75. WBFirst, WBLast : boolean;
  76. {FWritebuf}
  77. Procedure FWriteBuf(var F:FILE;val:byte);
  78. begin
  79. if WBFirst then
  80. begin
  81. WBfirst:=false;
  82. GetMem(WBBuf,BUFSize);
  83. WBcount:=0;
  84. end;
  85. if WBLast then
  86. begin
  87. WBLast:=false;
  88. BlockWrite(F,WBBuf^,WBcount);
  89. FreeMem(WBBuf,BufSize);
  90. WBFirst:=true;
  91. exit;
  92. end;
  93. WBwrt:=PTR(SeG(WBBUF^),Ofs(WBbuf^)+WBcount);
  94. inc(WBcount);
  95. WBwrt^:=val;
  96. If WBcount=BUFsize then
  97. begin
  98. BlockWrite(F,WBBuf^,WBcount);
  99. WBcount:=0;
  100. end;
  101. end;{FWriteBuf}
  102. {-----------------------------------------}
  103. Procedure DelBranch(br:PTree);
  104. begin
  105. if br^.left <> nil then DelBranch(br^.left);
  106. if br^.right <> nil then DelBranch(br^.right);
  107. Dispose(br);
  108. end;{DelBranch}
  109. {------------------------------------------}
  110. Procedure AddItem(var HEAD:ps; VAL : PTree);
  111. var
  112. c,NEWs : ps;
  113. Begin
  114. c := HEAD;
  115. New(News);
  116. News^.data := VAL;
  117. News^.prev := nil;
  118. if c=nil then begin HEAD := NEWs; exit end;
  119. if val^.ti <= C^.DATA^.ti then begin News^.prev:=C; HEAD:=News; exit end;
  120. while (c^.prev <> nil) and
  121. (not ((val^.ti>C^.data^.ti) and (val^.ti<=C^.prev^.data^.ti))) do
  122. c:=c^.prev;
  123. News^.prev := c^.prev;
  124. C^.prev := News;
  125. end;{AddItem}
  126. {---------------------------------}
  127. Procedure RemoveItem(var HEAD:ps; VAL : PTree);
  128. var
  129. c,cur : ps;
  130. Begin
  131. c := HEAD;
  132. if c=nil then exit;
  133. if c^.data = val then begin HEAD:=C^.prev; Dispose(C); exit end;
  134. while (c^.prev <> nil) and (C^.prev^.data <> val) do c:=c^.prev;
  135. if c^.prev^.data <> val then exit;
  136. cur := c^.prev;
  137. C^.prev := cur^.prev;
  138. Dispose(cur);
  139. end;{RemoveItem}
  140. {---------------------------------}
  141. Procedure DeleteAll(var HEAD:ps);
  142. var
  143. c:ps;
  144. begin
  145. while HEAD <> nil do
  146. begin
  147. c:=HEAD^.prev;
  148. Dispose(HEAD);
  149. HEAD := C;
  150. end;
  151. end;{DeleteAll}
  152. {---------------------------------}
  153. Procedure GetCode(VAL:Byte;BRA:PTree;cur:longint;len:byte);
  154. begin
  155. if (Bra^.left=nil) and (Bra^.right=nil) and (BRA^.info = VAL) then
  156. begin
  157. Code := cur;
  158. CodeLen:=len;
  159. end;
  160. if (Bra^.left<>nil) then begin
  161. GetCode(val,Bra^.left,(cur shl 1) + 1,len+1);
  162. GetCode(val,Bra^.right,(cur shl 1),len+1)
  163. end;
  164. end;{GetCode}
  165. {---------------------------------}
  166. Function GetByte(cur:string):byte;
  167. var ret:byte;
  168. begin
  169. ret:=0;
  170. while length(cur)>0 do
  171. begin
  172. ret:=ret shl 1 + ord(cur[1])-ord('0');
  173. delete(cur,1,1);
  174. end;
  175. GetByte:=ret;
  176. end;{GetByte}
  177. {---------------------------------}
  178. Procedure IfNeedWrite(var gg:longint;var F:FILE);
  179. var
  180. FFF : byte;
  181. begin
  182. FFF := gg and $0FF;
  183. FWriteBuf(F,Fff);
  184. end;{IfNeedWrite}
  185. {---------------------------------}
  186. Function GetBit(var F:File):boolean;
  187. begin
  188. if GNum = 0 then
  189. begin
  190. GNum:=8;
  191. inc(CSize);
  192. done:=FReadBuf(F,GO);
  193. if not done then
  194. begin
  195. RBLast:=true;
  196. FReadBuf(F,GO);
  197. WBLast:=true;
  198. FWriteBuf(OUT,0);
  199. close(INP);
  200. Close(OUT);
  201. Halt(0);
  202. end;
  203. if csize=Size-1 then done:=FReadBuf(F,GNum);
  204. end;
  205. Dec(GNum);
  206. GetBit := (GO and 128) = 128;
  207. Go := GO shl 1;
  208. end;{GetBit}
  209. {---------------------------------}
  210. VAR
  211. CURCODE : longint;
  212. curlen : byte;
  213. CUR : byte;
  214. i : byte;
  215. Ctr : Ptree;
  216. BEGIN
  217. if (ParamCount<1) then
  218. begin
  219. WriteLn('Usage: HUF.exe <FILENAME.EXT> [OUTNAME.ext]');
  220. Halt(10);
  221. end;
  222. Fname := ParamStr(1);
  223. for size := 1 to Length(Fname) do
  224. fname[size] := UpCase(fname[size]);
  225. if pos('.HUF',fname) = 0 then
  226. begin
  227. if Paramcount = 2 then fout := ParamStr(2) else
  228. fout := Copy(Fname,1,pos('.',Fname)) + 'HUF';
  229. Assign(INP,Fname);
  230. Reset(INP,1);
  231. Size:=FileSize(INP);
  232. Assign(OUT,Fout);
  233. Rewrite(OUT,1);
  234. BlockWrite(OUT,Fname,14);
  235. curcode:=0;
  236. curlen:=0;
  237. csize:=0;
  238. RBFirst:=true;
  239. RBLast:=false;
  240. WBFirst:=true;
  241. WBLast:=false;
  242. while FReadBuf(INP,CUR) do
  243. begin
  244. inc(csize);
  245. if cur in USED then
  246. begin
  247. AddItem(HEAD)
  248. end
  249. else
  250. begin
  251. end;
  252. Write(#13'Complete: ',round((csize/size)*100),'%');
  253. end;
  254. if (curlen <> 0) then
  255. begin
  256. curcode := curcode shl (7-curlen);
  257. ifNeedWrite(curcode,OUT);
  258. end
  259. else
  260. curlen:=7;
  261. FWriteBuf(OUT,curlen);
  262. RBLast:=true;
  263. FReadBuf(INP,cur);
  264. WBLast:=true;
  265. FWriteBuf(OUT,0);
  266. Close(OUT);
  267. Close(INP);
  268. end
  269. else
  270. begin
  271. end;
  272. END.