tcbufferedfilestream.pp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. unit tcbufferedfilestream;
  2. {$mode objfpc}{$H+}
  3. interface
  4. uses
  5. Classes, SysUtils, fpcunit, testregistry, bufstream;
  6. type
  7. { TTestBufferedFileStream }
  8. TTestBufferedFileStream= class(TTestCase)
  9. private
  10. const
  11. TEST_RANDOM_READS=10000;
  12. TEST_SEQUENTIAL_READS=1000000;
  13. TEST_FILENAME='testfile.bin';
  14. TEST_WRITEC_FILE='testwritecache.bin';
  15. TEST_WRITEF_FILE='testwritedirec.bin';
  16. private
  17. function CompareStreams(const aStream1: TStream; const aStream2: TStream): Boolean;
  18. protected
  19. procedure SetUp; override;
  20. procedure TearDown; override;
  21. published
  22. procedure TestCacheRead;
  23. procedure TestCacheWrite;
  24. procedure TestCacheSeek;
  25. end;
  26. implementation
  27. procedure TTestBufferedFileStream.TestCacheRead;
  28. var
  29. lBufferedStream: TBufferedFileStream;
  30. lStream: TFileStream;
  31. b: array [0..10000-1] of char;
  32. j,k: integer;
  33. lBytesToRead: integer;
  34. lEffectiveRead: integer;
  35. {$IFDEF CHECK_AGAINST_FILE}
  36. lEffectiveRead2: integer;
  37. {$ENDIF}
  38. lReadPosition: int64;
  39. lCheckInitV: integer;
  40. lTick: QWord;
  41. begin
  42. b[0]:=#0; // Avoid initalization hint
  43. lBufferedStream:=TBufferedFileStream.Create(TEST_FILENAME,fmOpenRead or fmShareDenyWrite);
  44. lStream:=TFileStream.Create(TEST_FILENAME,fmOpenRead or fmShareDenyWrite);
  45. try
  46. RandSeed:=1;
  47. Randomize;
  48. lTick:=GetTickCount64;
  49. for j := 0 to Pred(TEST_RANDOM_READS) do begin
  50. lBytesToRead:=Random(10000);
  51. lReadPosition:=Random(lBufferedStream.Size);
  52. lBufferedStream.Position:=lReadPosition;
  53. lEffectiveRead:=lBufferedStream.Read(b,lBytesToRead);
  54. {$IFDEF CHECK_AGAINST_FILE}
  55. // Now read without cache
  56. lStream.Position:=lReadPosition;
  57. lEffectiveRead2:=lStream.Read(b2,lBytesToRead);
  58. if lEffectiveRead<>lEffectiveRead2 then begin
  59. FAIL('Read length mismatch');
  60. end;
  61. if not CompareMem(@b[0],@b2[0],lEffectiveRead) then begin
  62. FAIL('Compare buffer data error');
  63. end;
  64. F.Position:=0;
  65. {$ELSE}
  66. lCheckInitV:=lReadPosition mod 10;
  67. for k := 0 to Pred(lEffectiveRead) do begin
  68. if b[k]<>char(ord('0')+lCheckInitV mod 10) then begin
  69. FAIL('Expected data error');
  70. end;
  71. inc(lCheckInitV);
  72. end;
  73. {$ENDIF}
  74. end;
  75. // Writeln('CACHE ',TEST_RANDOM_READS,' random reads in ',GetTickCount64-lTick,' ms.');
  76. RandSeed:=1;
  77. Randomize;
  78. // Writeln('Same operation without cache');
  79. lTick:=GetTickCount64;
  80. for j := 0 to Pred(TEST_RANDOM_READS) do begin
  81. lBytesToRead:=Random(10000);
  82. lReadPosition:=Random(lBufferedStream.Size);
  83. lStream.Position:=lReadPosition;
  84. lEffectiveRead:=lStream.Read(b,lBytesToRead);
  85. lCheckInitV:=lReadPosition mod 10;
  86. for k := 0 to Pred(lEffectiveRead) do begin
  87. if b[k]<>char(ord('0')+lCheckInitV mod 10) then begin
  88. FAIL('Expected data error');
  89. end;
  90. inc(lCheckInitV);
  91. end;
  92. end;
  93. // Writeln('FILE ',TEST_RANDOM_READS,' random reads in ',GetTickCount64-lTick,' ms.');
  94. // Writeln('Check sequential read');
  95. RandSeed:=1;
  96. Randomize;
  97. lTick:=GetTickCount64;
  98. lBytesToRead:=1;
  99. lReadPosition:=0;
  100. lBufferedStream.Position:=lReadPosition;
  101. lStream.Position:=lReadPosition;
  102. for j := 0 to Pred(TEST_SEQUENTIAL_READS) do begin
  103. lEffectiveRead:=lBufferedStream.Read(b,lBytesToRead);
  104. {$IFDEF CHECK_AGAINST_FILE}
  105. // Now read without cache
  106. lEffectiveRead2:=lStream.Read(b2,lBytesToRead);
  107. if lEffectiveRead<>lEffectiveRead2 then begin
  108. FAIL('Read length mismatch');
  109. end;
  110. if not CompareMem(@b[0],@b2[0],lEffectiveRead) then begin
  111. FAIL('Compare buffer data error');
  112. end;
  113. F.Position:=0;
  114. {$ELSE}
  115. lCheckInitV:=lReadPosition mod 10;
  116. for k := 0 to Pred(lEffectiveRead) do begin
  117. if b[k]<>char(ord('0')+lCheckInitV mod 10) then begin
  118. FAIL('Expected data error');
  119. end;
  120. inc(lCheckInitV);
  121. end;
  122. {$ENDIF}
  123. inc(lReadPosition,lBytesToRead);
  124. end;
  125. // Writeln('CACHE ',TEST_SEQUENTIAL_READS,' byte sequential reads in ',GetTickCount64-lTick,' ms.');
  126. RandSeed:=1;
  127. Randomize;
  128. lTick:=GetTickCount64;
  129. lBytesToRead:=1;
  130. lReadPosition:=0;
  131. lStream.Position:=lReadPosition;
  132. for j := 0 to Pred(TEST_SEQUENTIAL_READS) do begin
  133. lEffectiveRead:=lStream.Read(b,lBytesToRead);
  134. lCheckInitV:=lReadPosition mod 10;
  135. for k := 0 to Pred(lEffectiveRead) do begin
  136. if b[k]<>char(ord('0')+lCheckInitV mod 10) then begin
  137. FAIL('Expected data error');
  138. end;
  139. inc(lCheckInitV);
  140. end;
  141. inc(lReadPosition,lBytesToRead);
  142. end;
  143. // Writeln('FILE ',TEST_SEQUENTIAL_READS,' byte sequential reads in ',GetTickCount64-lTick,' ms.');
  144. // Writeln('CACHE Trying read beyond limits');
  145. lBufferedStream.Position:=lBufferedStream.Size-1;
  146. lEffectiveRead:=lBufferedStream.Read(b,2);
  147. if lEffectiveRead<>1 then begin
  148. FAIL('Read beyond limits, returned bytes: '+inttostr(lEffectiveRead));
  149. end else begin
  150. // Writeln('CACHE OK, read beyond limits returns 0 bytes.');
  151. end;
  152. finally
  153. lBufferedStream.Free;
  154. lStream.Free;
  155. end;
  156. end;
  157. procedure TTestBufferedFileStream.TestCacheWrite;
  158. const
  159. EXPECTED_SIZE=10000000;
  160. TEST_ROUNDS=100000;
  161. var
  162. lBufferedStream: TBufferedFileStream;
  163. lStream: TFileStream;
  164. lVerifyStream1,lVerifyStream2: TFileStream;
  165. b: array [0..10000-1] of char;
  166. j: integer;
  167. lBytesToWrite: integer;
  168. lWritePosition: int64;
  169. begin
  170. // Writeln('Testing write cache');
  171. // All test should return the same random sequence
  172. RandSeed:=1;
  173. Randomize;
  174. for j := 0 to Pred(10000) do begin
  175. b[j]:='0';
  176. end;
  177. lBufferedStream:=TBufferedFileStream.Create(TEST_WRITEC_FILE,fmCreate);
  178. lStream:=TFileStream.Create(TEST_WRITEF_FILE,fmCreate);
  179. try
  180. for j := 0 to Pred(EXPECTED_SIZE div Sizeof(b)) do begin
  181. lBufferedStream.Write(b,sizeof(b));
  182. lStream.Write(b,sizeof(b));
  183. end;
  184. for j := 0 to Pred(Sizeof(b)) do begin
  185. b[j]:=char(ord('0')+j mod 10);
  186. end;
  187. finally
  188. lBufferedStream.Free;
  189. lStream.Free;
  190. end;
  191. lBufferedStream:=TBufferedFileStream.Create(TEST_WRITEC_FILE,fmOpenReadWrite);
  192. lStream:=TFileStream.Create(TEST_WRITEF_FILE,fmOpenWrite);
  193. try
  194. for j := 0 to Pred(TEST_ROUNDS) do begin
  195. if lStream.Size<>lBufferedStream.Size then begin
  196. FAIL('Mismatched lengths');
  197. end;
  198. lWritePosition:=Random(EXPECTED_SIZE);
  199. lBytesToWrite:=Random(sizeof(b));
  200. lBufferedStream.Position:=lWritePosition;
  201. lStream.Position:=lWritePosition;
  202. lBufferedStream.Write(b,lBytesToWrite);
  203. lStream.Write(b,lBytesToWrite);
  204. // if j mod 1273 = 0 then write(j,' / ',TEST_ROUNDS,#13);
  205. end;
  206. // Writeln(TEST_ROUNDS,' / ',TEST_ROUNDS);
  207. if lStream.Size<>lBufferedStream.Size then begin
  208. FAIL('Mismatched lengths');
  209. end;
  210. finally
  211. lBufferedStream.Free;
  212. lStream.Free;
  213. end;
  214. // Verify both generated files are identical.
  215. lVerifyStream1:=TFileStream.Create(TEST_WRITEC_FILE,fmOpenRead or fmShareDenyWrite);
  216. lVerifyStream2:=TFileStream.Create(TEST_WRITEF_FILE,fmOpenRead or fmShareDenyWrite);
  217. try
  218. if not CompareStreams(lVerifyStream1,lVerifyStream2) then begin
  219. FAIL('Streams are different!!');
  220. end else begin
  221. // Writeln('Streams are identical. OK.');
  222. end;
  223. finally
  224. lVerifyStream1.Free;
  225. lVerifyStream2.Free;
  226. end;
  227. end;
  228. procedure TTestBufferedFileStream.TestCacheSeek;
  229. var
  230. lBufferedStream: TBufferedFileStream;
  231. lStream: TFileStream;
  232. bBuffered: array [0..10000] of BYTE;
  233. bStream: array [0..10000] of BYTE;
  234. bread : Integer;
  235. begin
  236. bBuffered[0]:=0; // Avoid initalization hint
  237. bStream[0]:=0; // Avoid initalization hint
  238. lBufferedStream:=TBufferedFileStream.Create(TEST_FILENAME,fmOpenRead or fmShareDenyWrite);
  239. lStream:=TFileStream.Create(TEST_FILENAME,fmOpenRead or fmShareDenyWrite);
  240. try
  241. // Writeln('Set position=-1');
  242. lStream.Position:=-1;
  243. // Writeln('TFileStream position=',lStream.Position);
  244. lBufferedStream.Position:=-1;
  245. // Writeln('Buffered position=',lBufferedStream.Position);
  246. if lStream.Position<>lBufferedStream.Position then begin
  247. FAIL('Positions are not the same.');
  248. end else begin
  249. // Writeln('Positions are the same.');
  250. end;
  251. // Writeln('Read data when position=-1');
  252. bread:=lStream.Read(bBuffered[0],10);
  253. // Writeln('TFileStream read bytes : ',bread);
  254. // Writeln('TFileStream end position: ',lStream.Position);
  255. bread:=lBufferedStream.Read(bStream[0],10);
  256. // Writeln('Buffered read bytes: ',bread);
  257. // Writeln('Buffered end position: ',lBufferedStream.Position);
  258. if (not CompareMem(@bBuffered[0],@bStream[0],10)) or (lStream.Position<>lBufferedStream.Position) then begin
  259. FAIL('Read data or positions are not the same.');
  260. end else begin
  261. // Writeln('Read data at -1 is the same.');
  262. end;
  263. // Writeln('Testing Seek operations');
  264. // Writeln('Seek -1 from beginning');
  265. bread:=lStream.Seek(-1,soBeginning);
  266. // Writeln('Stream seek result : ',bread);
  267. bread:=lBufferedStream.Seek(-1,soBeginning);
  268. // Writeln('Buffered seek result: ',);
  269. // Writeln('Read data when Seek -1');
  270. bread:=lStream.Read(bBuffered[0],10);
  271. // Writeln('TFileStream read bytes : ',bread);
  272. // Writeln('TFileStream end position: ',lStream.Position);
  273. bread:=lBufferedStream.Read(bStream[0],10);
  274. // Writeln('Buffered read bytes: ',bread);
  275. // Writeln('Buffered end position: ',lBufferedStream.Position);
  276. if (not CompareMem(@bBuffered[0],@bStream[0],10)) or (lStream.Position<>lBufferedStream.Position) then begin
  277. FAIL('Read data or positions are not the same.');
  278. end else begin
  279. // Writeln('Read data at -1 is the same.');
  280. end;
  281. // Writeln('Seek -current*2 from current');
  282. bread:=lStream.Seek(lStream.Position*-2,soCurrent);
  283. // Writeln('Stream seek result : ',bread);
  284. bread:=lBufferedStream.Seek(lBufferedStream.Position*-2,soCurrent);
  285. // Writeln('Buffered seek result: ',bread);
  286. // Writeln('Read data when Seek from current -current*2');
  287. bread:=lStream.Read(bBuffered[0],10);
  288. // Writeln('TFileStream read bytes : ',bread);
  289. // Writeln('TFileStream end position: ',lStream.Position);
  290. bread:=lBufferedStream.Read(bStream[0],10);
  291. // Writeln('Buffered read bytes: ',);
  292. // Writeln('Buffered end position: ',lBufferedStream.Position);
  293. if (not CompareMem(@bBuffered[0],@bStream[0],10)) or (lStream.Position<>lBufferedStream.Position) then begin
  294. FAIL('Read data or positions are not the same.');
  295. end else begin
  296. // Writeln('Read data at -current*2 is the same.');
  297. end;
  298. finally
  299. lBufferedStream.Free;
  300. lStream.Free;
  301. end;
  302. end;
  303. procedure TTestBufferedFileStream.SetUp;
  304. var
  305. F: TFileStream;
  306. b: array [0..10000-1] of char;
  307. j: integer;
  308. begin
  309. for j := 0 to Pred(10000) do begin
  310. b[j]:=char(ord('0')+j mod 10);
  311. end;
  312. F:=TFileStream.Create(TEST_FILENAME,fmCreate);
  313. for j := 0 to Pred(1000) do begin
  314. F.Write(b,sizeof(b));
  315. end;
  316. F.Free;
  317. end;
  318. procedure TTestBufferedFileStream.TearDown;
  319. begin
  320. DeleteFile(TEST_FILENAME);
  321. DeleteFile(TEST_WRITEC_FILE);
  322. DeleteFile(TEST_WRITEF_FILE);
  323. end;
  324. function TTestBufferedFileStream.CompareStreams(const aStream1: TStream;
  325. const aStream2: TStream): Boolean;
  326. const
  327. BUFFER_SIZE=5213; // Odd number
  328. var
  329. b1: array [0..BUFFER_SIZE-1] of BYTE;
  330. b2: array [0..BUFFER_SIZE-1] of BYTE;
  331. lReadBytes: integer;
  332. lAvailable: integer;
  333. lEffectiveRead1: integer;
  334. lEffectiveRead2: integer;
  335. begin
  336. b1[0]:=0; // Avoid initalization hint
  337. b2[0]:=0; // Avoid initalization hint
  338. Result:=false;
  339. if aStream1.Size<>aStream2.Size then exit;
  340. aStream1.Position:=0;
  341. aStream2.Position:=0;
  342. while aStream1.Position<aStream1.Size do begin
  343. lAvailable:=aStream1.Size-aStream1.Position;
  344. if lAvailable>=BUFFER_SIZE then begin
  345. lReadBytes:=BUFFER_SIZE;
  346. end else begin
  347. lReadBytes:=aStream1.Size-aStream1.Position;
  348. end;
  349. lEffectiveRead1:=aStream1.Read(b1[0],lReadBytes);
  350. lEffectiveRead2:=aStream2.Read(b2[0],lReadBytes);
  351. if lEffectiveRead1<>lEffectiveRead2 then exit;
  352. if not CompareMem(@b1[0],@b2[0],lEffectiveRead1) then exit;
  353. end;
  354. Result:=true;
  355. end;
  356. initialization
  357. RegisterTest(TTestBufferedFileStream);
  358. end.