comparison src/cs/drivers/drv_app/ffs/board/tcases.c @ 0:4e78acac3d88

src/{condat,cs,gpf,nucleus}: import from Selenite
author Mychaela Falconia <falcon@freecalypso.org>
date Fri, 16 Oct 2020 06:23:26 +0000
parents
children
comparison
equal deleted inserted replaced
-1:000000000000 0:4e78acac3d88
1 /******************************************************************************
2 * Flash File System (ffs)
3 * Idea, design and coding by Mads Meisner-Jensen, mmj@ti.com
4 *
5 * ffs test cases
6 *
7 * $Id: tcases.c 1.13.1.1.1.66 Thu, 08 Jan 2004 15:05:23 +0100 tsj $
8 *
9 ******************************************************************************/
10
11 #ifndef TARGET
12 #include "ffs.cfg"
13 #endif
14
15 #include "ffs/ffs_api.h" // Temp
16 #include "ffs/ffs.h"
17
18 #include "ffs/board/tffs.h"
19 #include "ffs/board/core.h" // only for block/object recovery test flags
20 #include "ffs/board/tdata.h"
21 #include "ffs/board/ffstrace.h"
22 #include "ffs/board/drv.h"
23 #include "ffs/pcm.h"
24
25 #if((TARGET == 1) || (RIV_ENV==1))
26 #include "rvf/rvf_api.h" // this include rv_general.h and rvf_target.h
27 #include "rvm/rvm_use_id_list.h"
28
29 #endif
30
31 #include <stdio.h>
32 #include <string.h>
33 #include <stdlib.h>
34 #include <limits.h>
35 #include <assert.h>
36
37 #define LUDIR "/europe/sweden"
38 #define MFDIR "/pcm"
39
40 // TODO: reimplement test case bfull (not valid any more).
41
42 // TODO: Cleanup in beginning of each test case for as many cases as
43 // possible, so they can be re-run.
44
45 // TODO: Make as many cases as possible fully independent of number of
46 // blocks and size of blocks in ffs.
47
48 // TODO: Every testcase should test if it is applicable to the current
49 // environment, e.g. some tests are impossible if e have a 2 or 3 block
50 // file system.
51
52 // TODO: test case for testing age functionality
53 // TODO: Implement case_ren
54 // TODO: Implement case_rm
55
56 // NOTEME: DO something with test case flags (PC, IT, RND)?
57
58 // Should we make a test case where we use case_lsr to make map of objects (
59 // ensuring mapsize is prime). In a loop, we select a random object (object
60 // index[rnd%size]) and perform a random operation on that object?
61
62 // Should test data (tdata[]) be of sizes 2^n+n or Xn+n where n = 0..15
63
64 // Should all test files have suffix ".<tdata-index>"? This would make very
65 // easy data checking!
66
67 // Add compiler define: WITH_PCM
68
69 //unsigned char ffs_image[4*4*1024];
70 //int ffs_ram_image_address = (int) &ffs_image;
71
72
73 /******************************************************************************
74 * Prototypes and Globals
75 ******************************************************************************/
76
77 // Helper functions
78 int case_cleanup(int min_space);
79 int case_mk_rand_file(char *dirname, int max_size, int min_size);
80 int ignore_file(char *pathname);
81 int case_trace_mask(int p0, int p1);
82 int case_reinit(int p0, int p1);
83 int case_debug_help(int p0, int p1);
84
85 struct object_s {
86 char *name;
87 struct xstat_s stat;
88 };
89
90 const struct testcase_s testcase[];
91
92 // Benchmark (not in use yet)
93 struct results_s {
94 int w16B;
95 int w256B;
96 int w4096B;
97 int rew4096B;
98 int r16B;
99 int r256B;
100 int r4096B;
101 int lfile;
102 int ldir;
103 int lonef;
104 };
105
106 /******************************************************************************
107 * Collective Test Cases
108 ******************************************************************************/
109
110 // test cases: all, alot, most, much + specific ones
111
112 int case_all(int p0, int p1)
113 {
114 #if (TARGET == 1)
115 UINT32 time_begin, elapsed;
116 time_begin = tffs_timer_begin();
117 #endif
118 // We have to run test case 'init' after test case 'format'. This is
119 // because 'init' test case calls test_ffs_params_get() which
120 // initializes variables used in many of the testcases.
121
122 error = test_run("ninit;format;i;world;eu;pcm;apiexc;"
123 "bigf;open;rw;seek;append;ren;mopen;"
124 "jnl;irec;drec;trunc;brec;stat;"
125 "lu;fc;root;dirs;frd;bfull;dsync;"
126 "ssym;find;ri;x;fwflags;query;octrl");
127 #if (TARGET == 1)
128 elapsed = tffs_timer_end(time_begin);
129 ttw(ttr(TTrAll, "Time elapsed %ds" NL, elapsed / 1000));
130 #endif
131
132 test_statistics_print();
133
134 return error;
135 // Not implemented test cases...
136 return test_run("rm;pcm;");
137 }
138
139 int case_alot(int p0, int p1)
140 {
141 return test_run("ninit;format;ri;i;ri;world;ri;eu;ri;fwflags;ri;"
142 "bigf;ri;open;ri;rw;ri;seek;ri;append;ri;ren;ri;"
143 "jnl;ri;irec;ri;brec;ri;stat;ri;trunc;ri;mopen;ri;"
144 "lu;ri;fc;ri;root;ri;dirs;ri;frd;ri;bfull;ri;"
145 "ssym;ri;find;ri;x;ri;dsync;ri;pcm;ri;query;ri;octrl");
146 }
147
148 int case_tall(int p0, int p1)
149 {
150 return test_run("ninit;format;i;world;eu;fwflags;"
151 "bigf;jnl;irec;drec;ren;apiexc;"
152 "stat;open;rw;seek;trunc;append;octrl;"
153 "lu;fc;root;dirs;frd;bfull;pcm;query;"
154 "ssym;find;x;ex;bf;nb;mopen;dsync;brec");
155 }
156
157 // Ad hoc test case
158 int case_test(int p0, int p1)
159 {
160 return test_run("format;i;world;eu;ri;mopen1;ri;fwflags;ri;"
161 "bigf;ri;jnl;ri;irec;ri;drec;ri;brec;ri;pcm;ri;"
162 "trunc1;ri;stat;open1;ri;rw1;ri;seek1;ri;ren;ri;"
163 "lu;ri;fc;ri;root;ri;dirs;ri;frd;ri;bfull;ri;"
164 "ssym;ri;find;ri;x;ri;append;ri;dsync;ri;");
165 }
166
167 extern struct dev_s dev;
168 // Agressive all. Run case 'all' for dev.numblocks in the range dev.numblocks..4
169 int case_aall(int p0, int p1)
170 {
171 char myname[20];
172
173 int i, failed = 0;
174
175 if (dev.numblocks * dev.blocksize > 1024*1024)
176 strcpy(myname, "/ffs/b999");
177 else
178 strcpy(myname, "/ffs/b99");
179
180 /** There's no need to test for i=127,126,125,124..3,2,1. So for i
181 * >= 20 we progress a little faster. */
182 for (i = dev.numblocks; i >= 3; i -= (i >= 20 ? i/4 : 1))
183 {
184 tw(tr(TR_FUNC, TrTest, "TEST aall. %d\n", i));
185 ttw(ttr(TTrTest, "TEST aall. %d" NL, i));
186
187 error = tffs_preformat(0xDEAD);
188 expect(error, EFFS_OK);
189
190 sprintf(myname, "/ffs/b%d", i);
191 tffs_format(myname, 0x2BAD);
192 expect(error, EFFS_OK);
193
194 failed += test_run("i;world;eu;bigf;bfull;"
195 "jnl;irec;drec;brec;stat;dsync;"
196 "open;rw;seek;trunc;ren;mopen;"
197 "lu;fc;root;dirs;frd;append;"
198 "ssym;find;ri;x");
199
200 }
201 return failed;
202 }
203
204 // This is a collection of all failing testcases to be investigated.
205 int case_fail(int p0, int p1)
206 {
207 int result;
208 const char imeifile[] = "/europe/norway/IMEI";
209
210 switch (p0) {
211 case 1:
212 tw(tr(TR_FUNC, TrTestHigh,
213 "remember to run case_fcontrol before this one\n"));
214 // Make symlink to imeifile and try to update it
215 error = tffs_symlink("/europe/imie", imeifile);
216 expect(error, EFFS_ACCESS);
217
218 error = tffs_fwrite("/europe/imie", TDATA(1));
219 expect(error, EFFS_ACCESS);
220 break;
221 case 2:
222 if (1) {
223 const char bigfile[] = "/iceberg";
224 int bytes_max, file_size;
225 char myname[] = "/ffs/b7";
226
227 error = tffs_preformat(0xDEAD);
228 expect(error, EFFS_OK);
229
230 error = tffs_format(myname, 0x2BAD);
231 expect(error, EFFS_OK);
232
233 ffs_query(Q_BYTES_FREE, (uint32 *) &bytes_max);
234
235 // File the system with this huge file
236 file_size = bytes_max;
237
238 ttw(ttr(TTrTest, "Bigfile of size %d" NL, file_size));
239 tw(tr(TR_FUNC, TrTestHigh, "Bigfile of size %d\n", file_size));
240
241 error = tffs_fcreate(bigfile, (char *) tdata[TDATA_HUGE], file_size);
242 expect(error, EFFS_OK);
243 }
244 break;
245
246 default:
247 result = 1;
248 }
249 return result;
250 }
251
252 // zzz to easy search ;-)
253 // NOTE: Move the below tests
254 extern char *tdata_huge;
255 // Test that a FFS API blocking call that not is able to send a mail to FFS fails as expected and don't lock the system (it must unlock and delete the mutex).
256 int case_okay(int p0, int p1)
257 {
258
259 error = tffs_fwrite("/test", tdata[TDATA_HUGE], 20);
260 expect_ok(error);
261
262 error = tffs_fread("/test", bigbuf, 20);
263 expect_ok(error);
264
265 bigbuf[20] = 0;
266
267 tw(tr(TR_FUNC, TrTestHigh, "String '%s'\n", bigbuf));
268
269 return EFFS_OK;
270 }
271
272 // Run testcases in a random order. Only the test cases in the test case
273 // table that are marked as re-runnable, are run.
274 int case_rand(int p0, int p1)
275 {
276 int i, n, seed, max = 0, error = 0;
277 const struct testcase_s *p;
278 // This is a way to activate trace at a defined test number. If rand
279 // test number 134 fail it is possible to activate trace at test number
280 // 133 etc. Note we have to change activate_trace_nr manual before compile.
281 int activate_trace_nr = 0;
282 int trace_mask = 0xFFDFF;
283 // NOTE: use p1 as active_trace_nr? or make it
284
285 p0 = (p0 == 0 ? 117 : p0); // Number of test cases to run
286 p1 = (p1 == 0 ? 567 : p1); // Initial seed
287 seed = p1;
288 if (seed == 1) {
289 ; // TODO: Set seed as a variable of current time
290 }
291
292 // TODO: Initialize seed from p1.
293
294 // First count total number of test cases
295 for (p = testcase; p->name; p++)
296 max++;
297
298 tw(tr(TR_FUNC, TrTestHigh,
299 "Number of available random test cases = %d\n", max));
300
301 for (i = 0; i < p0; i++)
302 {
303 do {
304 n = rand() % max;
305 } while ((testcase[n].flags & RND) == 0);
306
307 if ((i + 1) == activate_trace_nr) {
308 #if (TARGET == 0)
309 tr_init(trace_mask, 2, 0 );
310 #else
311 ttr_init(trace_mask);
312 #endif
313 tw(tr_bstat());
314 }
315
316 tw(tr(TR_FUNC, TrTest, "Nr: %d", i + 1));
317 ttw(ttr(TTrTest, "Nr: %d" NL, i + 1));
318
319 if ((error = test_run(testcase[n].name)))
320 break;
321 }
322
323 if (p1 == 1)
324 tw(tr(TR_FUNC, TrTestHigh, "Initial seed = %d\n", seed));
325
326 test_statistics_print();
327 return error;
328 }
329
330
331 /******************************************************************************
332 * Population Tests
333 ******************************************************************************/
334
335 int case_world(int p0, int p1)
336 {
337 int i;
338 const char *dirs[] = { "/antarctica", "/africa",
339 "/asia", "/europe",
340 "/north-america", "/south-america",
341 "/australia" };
342
343 // Cleanup
344 for (i = 0; i < sizeof(dirs)/sizeof(char *); i++) {
345 tffs_remove(dirs[i]);
346 }
347
348 for (i = 0; i < sizeof(dirs)/sizeof(char *); i++) {
349 error = tffs_mkdir(dirs[i]);
350 expect(error, EFFS_OK);
351 }
352 return 0;
353 }
354
355
356 int case_europe(int p0, int p1)
357 {
358 int i;
359 const char *dirs[] = { "/europe/denmark", "/europe/sweden",
360 "/europe/norway", "/europe/finland" };
361
362 // Cleanup
363 for (i = 0; i < sizeof(dirs)/sizeof(char *); i++) {
364 error = tffs_remove(dirs[i]);
365 }
366
367 for (i = 0; i < sizeof(dirs)/sizeof(char *); i++) {
368 error = tffs_mkdir(dirs[i]);
369 expect(error, EFFS_OK);
370 }
371 return 0;
372 }
373
374 int case_denmark(int p0, int p1)
375 {
376 // Cleanup
377 tffs_remove("/europe/denmark/jutland");
378 tffs_remove("/europe/denmark/sealand");
379
380 error = tffs_mkdir("/europe/denmark/jutland");
381 expect(error, EFFS_OK);
382 error = tffs_mkdir("/europe/denmark/sealand");
383 expect(error, EFFS_OK);
384
385 return 0;
386 }
387
388
389 /******************************************************************************
390 * Atomic Tests
391 ******************************************************************************/
392
393 int case_init(int p0, int p1)
394 {
395 error = tffs_initialize();
396 // ignore error
397 error = test_ffs_params_get();
398 expect(error, EFFS_OK);
399 return 0;
400 }
401
402 int case_exit(int p0, int p1)
403 {
404 error = tffs_exit();
405 expect(error, EFFS_OK);
406 return 0;
407 }
408
409 int case_only_preformat(int p0, int p1)
410 {
411 tffs_preformat(0xDEAD);
412 expect(error, EFFS_OK);
413 return 0;
414 }
415
416 int case_only_format(int p0, int p1)
417 {
418 tffs_format("/ffs", 0x2BAD);
419 expect(error, EFFS_OK);
420 return 0;
421 }
422
423 int case_reset(int p0, int p1)
424 {
425 error = tffs_initialize();
426 // ignore result
427 error = tffs_preformat(0xDEAD);
428 expect(error, EFFS_OK);
429 error = tffs_format("/ffs", 0x2BAD);
430 expect(error, EFFS_OK);
431 error = tffs_exit();
432 expect(error, EFFS_OK);
433 return 0;
434 }
435
436 int case_status(int p0, int p1)
437 {
438 error = test_run("lsr");
439 return error;
440 }
441
442
443 /******************************************************************************
444 * Special Tests
445 ******************************************************************************/
446
447 // Test FFS with only 2 blocks in device
448 int case_twob(int p0, int p1)
449 {
450 int space_left;
451 int size, i;
452 char myname[] = "/two_block99";
453
454 // Format ffs to use only 2 block no journal file and no reserved space.
455 // Make some directoryes.
456 // Fill ffs allmost to the edge and try to write over the edge.
457 // Read back all the files.
458 error = tffs_preformat(0xDEAD);
459 expect(error, EFFS_OK);
460 error = tffs_format("/ffs/b2j0m0", 0x2BAD);
461 expect(error, EFFS_OK);
462
463 error = test_run("i;world;eu;x");
464 if(error) return 1;
465
466 // Calculate the amount of space that is left to user data because some space
467 // is used to directoryes
468 space_left = dev.blocksize - (13 * 20);
469 tw(tr(TR_FUNC, TrTestHigh, "Space left: %d\n", space_left));
470
471 size = space_left;
472 for(i = 0; i < 6; i++) {
473 size = size/2;
474 sprintf(myname, "/two_block%d", i);
475 error = tffs_fwrite(myname, (char*)tdata[TDATA_HUGE],
476 size >= 20 ? size - 20 : 0);
477 expect(error, EFFS_OK);
478 }
479
480 error = tffs_fwrite(myname, (char*)tdata[TDATA_HUGE], space_left/5);
481 expect(error, EFFS_NOSPACE);
482
483 // 20 is the average space used to the file name + word align.
484 size = space_left;
485 for(i = 0; i < 6; i++) {
486 size = size/2;
487 sprintf(myname, "/two_block%d", i);
488 error = tffs_fread(myname, bigbuf, bigbuf_size);
489 expect(error, size >= 20 ? size - 20 : 0);
490 }
491
492 //error = ffs_object_control(0, OC_DEV_DEVICE, 0x080D);
493 //if (error < 0) return error;
494
495 return 0;
496 }
497
498 // Test FFS with only 3 blocks in device
499 int case_threeb(int p0, int p1)
500 {
501 char mytest[20];
502
503 // Format ffs to use only 3 blocks, make directoryes and run test case
504 // rand <p0> number of times.
505 error = tffs_preformat(0xDEAD);
506 expect(error, EFFS_OK);
507 error = tffs_format("/ffs/b3r24", 0x2BAD);
508 expect(error, EFFS_OK);
509
510 error = test_run("i;world;eu;x");
511 if (error) return 1;
512
513 if(p0 == 0) p0 = 233;
514
515 sprintf(mytest, "rand%d", p0);
516 return test_run(mytest);
517 }
518
519
520 // Simulate a typical usage of ffs. We expect one part of the data will be
521 // what we call 'static' data (calibration files etc) the other part is the
522 // 'dynamic' data (wap, sms etc) we do also expect that they don't fill FFS
523 // completely thus we have a 'free' part.
524 int case_customer(int nblocks, int nloops)
525 {
526 int free_part = 5, static_part = 47, dynamic_part = 47;
527 int free_size, static_size, dynamic_size;
528 int bytes_free, bytes_max, bytes_used;
529 int file_size_max, file_size_min, file_size_avg;
530 int i, j, max_obj_block, num_static_files;
531 char myname[] = "/ffs/bxxx/";
532 char file_name[FFS_FILENAME_MAX + 1];
533 char path[FFS_FILENAME_MAX * 2];
534 char static_dir[] ="/READONLY";
535 char dynamic_dir[] = "/rand";
536 struct dir_s dir;
537
538 if (nblocks == 0)
539 nblocks = 7;
540 if (nloops == 0)
541 nloops = 300;
542
543 error = tffs_preformat(0xDEAD); expect(error, EFFS_OK);
544 sprintf(myname, "/ffs/b%d", nblocks);
545 error = tffs_format(myname, 0x2BAD); expect(error, EFFS_OK);
546
547 tffs_mkdir(dynamic_dir); expect(error, EFFS_OK);
548 tffs_mkdir(static_dir); expect(error, EFFS_OK);
549
550 ffs_query(Q_BYTES_FREE, &bytes_max);
551 free_size = bytes_max * free_part / 100;
552 static_size = bytes_max * static_part / 100;
553 dynamic_size = bytes_max * dynamic_part / 100;
554 tw(tr(TR_FUNC, TrTest, "Free:%d,%dkB. Static:%d,%dkB, Dynamic:%d,%dkB\n",
555 free_part, free_size/1024, static_part, static_size/1024,
556 dynamic_part, dynamic_size/1024));
557
558 // Find max average objects per block
559 max_obj_block = fs.block_files_max;
560 if (max_obj_block > fs.objects_max / (dev.numblocks -1 - fs.blocks_free_min))
561 max_obj_block = fs.objects_max / (dev.numblocks -1 - fs.blocks_free_min);
562 tw(tr(TR_FUNC, TrTest, "Max average objects per block %d\n", max_obj_block));
563 ttw(ttr(TTrTest, "Max avg obj per block %d" NL, max_obj_block));
564
565 // Average (minimum) file size (avoid to reach block_files_max and
566 // objects_max)
567 file_size_avg = dev.blocksize / max_obj_block;
568
569 ffs_query(Q_BYTES_USED, &bytes_used);
570 tw(tr(TR_FUNC, TrTest, "Write static files (avg size %d)\n", file_size_avg));
571 ttw(ttr(TTrTest, "avg size %d" NL, file_size_avg));
572 do {
573 i++;
574 file_size_min = 5;
575 // For each 5 file we make one huge.
576 file_size_max = i % 5 ? file_size_avg * 2 : file_size_avg * 15;
577 // For each 6 file we make one small.
578 file_size_max = i % 6 ? file_size_max : 5;
579
580 // Saturate, avoid to make one file that parses the total static_size
581 if (file_size_max > static_size - bytes_used)
582 file_size_max = file_size_min = static_size - bytes_used;
583
584 // For each 7 file we create one dynamic (If the block is full with only
585 // valid objects do we risk that this block never will be reclaimed!).
586 if (i % 7)
587 error = case_mk_rand_file(dynamic_dir, file_size_max, file_size_min);
588 else
589 error = case_mk_rand_file(static_dir, file_size_max, file_size_min);
590 expect_ok(error);
591 ffs_query(Q_BYTES_USED, &bytes_used);
592 } while (bytes_used < static_size);
593
594 num_static_files = tffs_opendir(static_dir, &dir);
595 expect_ok(num_static_files);
596 tw(tr(TR_FUNC, TrTest, "Written %d files\n", num_static_files));
597 ttw(ttr(TTrTest, "Written %d files" NL, num_static_files));
598 // In a loop we continue to write data until only the wanted free_part is
599 // left, after that we remove half of the just created files and then write
600 // again and again...
601 j = 0;
602 tw(tr(TR_FUNC, TrTest, "Write and remove dynamic files\n"));
603 ttw(ttr(TTrTest, "Write and remove dynamic files" NL));
604 for (i = 0; i < nloops; i++) {
605 if (i % 10 == 0) {
606 tw(tr(TR_FUNC, TrTest, "loop %d\n", i));
607 ttw(ttr(TTrTest, "loop %d" NL, i));
608 }
609 ffs_query(Q_BYTES_FREE, &bytes_free);
610 do {
611 j++;
612 file_size_min = 5;
613 file_size_max = j % 5 ? file_size_avg * 2 : file_size_avg * 15;
614 file_size_max = j % 6 ? file_size_max : 5;
615 // Saturate, avoid using the free space.
616 if (file_size_max > bytes_free - free_size)
617 file_size_max = file_size_min = bytes_free - free_size;
618
619 if (file_size_max < 0)
620 break;
621
622 error = case_mk_rand_file(dynamic_dir, file_size_max, file_size_min);
623 // NOTE it is very difficult to avoid EFFS_FSFULL because cleanup will
624 // sometimes only remove the big files thus there will exist a lot of small
625 // that use all the inodes.
626 if (error == EFFS_FSFULL) {
627 tw(tr(TR_FUNC, TrTest, "Warning no free inodes\n"));
628 ttw(ttr(TTrTest, "Warning no free inodes" NL));
629 error = case_cleanup(free_size + dynamic_size);
630 }
631 else
632 expect_ok(error);
633
634 ffs_query(Q_BYTES_FREE, &bytes_free);
635 } while (bytes_free > free_size);
636
637 error = case_cleanup(free_size + dynamic_size / 2);
638 expect_ok(error);
639 }
640
641 error = case_reinit(0, 0);
642 expect(EFFS_OK, error);
643
644 tw(tr(TR_FUNC, TrTest, "Verify all files\n"));
645 ttw(ttr(TTrTest, "Verify all files" NL));
646 for (i = 0; i < 2; i++) {
647 if (i == 0) {
648 error = tffs_opendir(static_dir, &dir);
649 expect_eq(error, num_static_files);
650 }
651 else {
652 error = tffs_opendir(dynamic_dir, &dir);
653 expect_ok(error);
654 }
655
656 while ((error = tffs_readdir(&dir, file_name, sizeof(file_name))) > 0) {
657 if (i == 0)
658 strcpy(path, static_dir);
659 else
660 strcpy(path, dynamic_dir);
661 strcat(path, "/");
662 strcat(path, file_name);
663
664 error = tffs_stat(path, &stat);
665 expect_ok(error);
666 error = test_expect_file(path, tdata[TDATA_HUGE], stat.size);
667 if (error) return 1;
668 }
669 }
670
671 test_statistics_print();
672 return 0;
673 }
674
675
676 // TODO cleanup this test. Use stat and verify some of the files etc.
677
678 // Used to test ConQuest issue FFS_FIX-11368 We fill FFS with a lot of small
679 // files. Then by turn remove some files and write new files, always very
680 // close to max avaible bytes.
681
682 // Note the below test is not able to run in big configuration because it
683 // creates a lot of small files thus we reach EFFS_FSFULL 'out of inodes'
684 int case_ffull(int p0, int p1)
685 {
686 int error, i, bytes_free, file_size, max_write_size = 1000;
687 char myname[] = "/ffs/bxxx/";
688
689 if (p0 == 0)
690 p0 = 100; // Default 100 loops
691 if (p1 == 0 || p1 > 999)
692 p1 = 3; // Default run in 3 blocks
693
694 error = tffs_preformat(0xDEAD);
695 expect(error, EFFS_OK);
696
697 sprintf(myname, "/ffs/b%d", p1);
698 error = tffs_format(myname, 0x2BAD);
699 expect(error, EFFS_OK);
700
701 tffs_mkdir("/rand");
702 expect(error, EFFS_OK);
703
704 tw(tr(TR_FUNC, TrTest, "Fill it with files until it fails\n"));
705 ttw(ttr(TTrTest, "Fill it with files until it fails" NL));
706
707 // We have to avoid too many small files or else do we reach objects_max
708 ffs_query(Q_BYTES_FREE, &bytes_free);
709 max_write_size = bytes_free / fs.objects_max * 2.5;
710
711 do {
712 error = case_mk_rand_file("/rand", max_write_size, 0);
713 } while (error >= 0 );
714
715 tw(tr(TR_FUNC, TrTest, "Fill done (%d)\n" ,error));
716 ttw(ttr(TTrTest, "Fill done (%d)" NL, error));
717 expect(error, EFFS_NOSPACE); // Now are we sure that it is full
718
719 tw(tr(TR_FUNC, TrTest, "Cleanup and fill again \n"));
720 for (i = 0; i < (1 + p0); i++) {
721 tw(tr(TR_FUNC, TrTestHigh, "Loop %d \n", i + 1));
722 ttw(ttr(TTrTest, "Loop %d" NL, i + 1));
723
724 tw(tr_bstat());
725 error = case_cleanup(max_write_size * 3);
726 tw(tr_bstat());
727
728 do {
729 ffs_query(Q_BYTES_FREE, &bytes_free);
730 // Saturate size
731 file_size = bytes_free > max_write_size ? max_write_size :bytes_free;
732 // Query BYTES_FREE don't count the below data use.
733 //file_size -= (FFS_FILENAME_MAX + 1 + dev.atomsize);
734 if (file_size < 0)
735 break;
736
737 error = case_mk_rand_file("/rand", file_size - (FFS_FILENAME_MAX+1), 0);
738
739 if (error == EFFS_FSFULL) { // Out of inodes
740 error = case_cleanup(max_write_size * 6);
741 // Create bigger files to avoid '-11', auto fit :-)
742 max_write_size *= 1.1; // Add 10%
743 tw(tr(TR_FUNC, TrTestHigh, "Out of inodes, write size: %d \n",
744 max_write_size));
745 continue;
746 }
747
748 expect_ok(error);
749
750 } while (file_size != bytes_free);
751 }
752
753 tw(tr(TR_FUNC, TrTest, "Test finish (%d) \n", error));
754 ttw(ttr(TTrTest, "Test finish (%d)" NL, error));
755
756 if (error >= 0)
757 return EFFS_OK;
758
759 return error;
760 }
761
762 int case_rivtype(int p0, int p1)
763 {
764 // Use the compiler to test if the types are correctly defined. We don't
765 // need to run this test just make a compilation.
766 T_FFS_OPEN_FLAGS flags = FFS_O_RDONLY;
767 T_FFS_SIZE size = 20;
768 T_FFS_RET error = -10;
769 T_FFS_WHENCE whence = FFS_SEEK_SET;
770 T_FFS_STAT stat;
771
772 T_FFS_DIR dir;
773 T_FFS_FD fd = FFS_FD_OFFSET;
774 T_RV_RETURN callback;
775 T_FFS_FILE_CNF confirm;
776 T_FFS_OBJECT_TYPE objt = OT_FILE;
777
778 // Use the defined variables to avoid the boring warning: unused
779 // variable 'xx'
780 if (flags != FFS_O_RDONLY) return 1;
781 if (size != 20) return 1;
782 if (error != -10) return 1;
783 if (whence != FFS_SEEK_SET) return 1;
784 if (fd != FFS_FD_OFFSET) return 1;
785 if (objt != OT_FILE) return 1;
786
787 stat.type = 1;
788 if (stat.type != 1) return 1;
789 dir.this = 5;
790 if (dir.this != 5) return 1;
791 callback.addr_id = 3;
792 if (callback.addr_id != 3) return 1;
793 confirm.error = -1;
794 if (confirm.error != -1) return 1;
795
796 return 0;
797 }
798
799 /******************************************************************************
800 * Pseudo Tests
801 ******************************************************************************/
802
803 // Recursively read objects in directory. Does NOT check for buffer
804 // overflow!
805 int case_dir_list(char *dirname, struct object_s **pplist, char **ppnames)
806 {
807 struct object_s *plist, *plist_start, *plist_end;
808 char pathname[6 * 20];
809 struct dir_s dir;
810 char *pnames, *pn;
811 int i, pathlen;
812
813 //tw(tr(TR_BEGIN, TrTestHigh, "dir_list('%s', %d, 0x%x)\n",
814 // dirname, (int)*pplist/sizeof(struct object_s), *ppnames));
815 plist_start = plist = *pplist;
816 pnames = *ppnames;
817
818 strcpy(pathname, dirname);
819 pathlen = strlen(pathname);
820
821 // remove trailing slash. It is tiring to handle the root directory
822 // differently from other directories. In a future ffs revision, this
823 // trailing slash should be allowed!
824 if (pathname[pathlen - 1] == '/') {
825 pathname[pathlen - 1] = 0;
826 pathlen--;
827 }
828
829 if (strlen(pathname) == 0)
830 error = tffs_opendir("/", &dir);
831 else
832 error = tffs_opendir(pathname, &dir);
833 expect_ok(error);
834
835 pn = pathname + strlen(pathname);
836 *pn++ = '/';
837 *pn = 0;
838
839 error = 1;
840 for (i = 0; (error = tffs_readdir(&dir, pn, 21)) > 0; i++)
841 {
842 error = tffs_xlstat(pathname, &plist->stat);
843 expect(error, EFFS_OK);
844
845 // Copy full object pathname to buffer, working downwards.
846 pnames -= strlen(pathname) + 1;
847 // Check for buffer overflow (if pnames <= plist)
848 expect_gt((int) pnames, (int) plist);
849 strcpy(pnames, pathname);
850 plist->name = pnames;
851
852 plist++;
853 }
854 *pplist = plist;
855 *ppnames = pnames;
856
857 // For each directory in the retrieved list, recurse.
858 plist_end = plist;
859 for (plist = plist_start; plist < plist_end; plist++) {
860 if (plist->stat.type == OT_DIR)
861 i += case_dir_list(plist->name, pplist, ppnames);
862 }
863 // tw(tr(TR_END, TrTestHigh, "} %d\n", i));
864 return i;
865 }
866
867 int case_find(int p0, int p1)
868 {
869 struct object_s *plist, *plist_start;
870 char *pnames, *pnames_start;
871 int n, names_used, list_used;
872
873 plist = plist_start = (struct object_s *) bigbuf;
874 pnames = pnames_start = bigbuf + bigbuf_size;
875 n = case_dir_list("/", &plist, &pnames);
876
877 list_used = n * sizeof(struct object_s);
878 names_used = pnames_start - pnames;
879
880 tw(tr(TR_FUNC, TrTestHigh, "Buffer space used: %d + %d = %d\n",
881 list_used, names_used, list_used + names_used));
882 ttw(ttr(TTrTest, "Buffer space used: %d + %d = %d" NL,
883 list_used, names_used, list_used + names_used));
884
885 return 0;
886 }
887
888 // TODO: We should accumulate all stat.space and check it vs. the number of
889 // bytes used!
890 int case_lsr(int p0, int p1)
891 {
892 struct object_s *plist, *plist_start;
893 char *pnames, *pnames_start;
894 char of[3];
895 int i, n;
896
897 plist = plist_start = (struct object_s *) bigbuf;
898 pnames = pnames_start = bigbuf + bigbuf_size;
899 n = case_dir_list("/", &plist, &pnames);
900 tw(tr(TR_FUNC, TrTestHigh, "Total %d objects.\n", n));
901 ttw(ttr(TTrTest, "Total %d objects" NL, n));
902
903 plist = plist_start;
904 for (i = 0; i < n; i++, plist++) {
905 strcpy(of, " ");
906 switch (plist->stat.type) {
907 case OT_FILE: of[0] = ' '; break;
908 case OT_DIR: of[0] = 'd'; break;
909 case OT_LINK: of[0] = 'l'; break;
910 }
911 if (plist->stat.flags & OF_READONLY)
912 of[1] = 'r';
913
914 #if (TARGET == 0)
915 printf("%3d: %s %3d %2d/%04X (%4d,%4d) %5d %s\n",
916 i, of,
917 plist->stat.inode, plist->stat.block,
918 plist->stat.location, plist->stat.sequence, plist->stat.updates,
919 plist->stat.size, &plist->name[1]);
920 #else
921 ttw(ttr(TTrTest, "%3d: %s %3d %2d/%04X (%4d,%4d) %5d %s" NL,
922 i, of,
923 plist->stat.inode, plist->stat.block,
924 plist->stat.location, plist->stat.sequence, plist->stat.updates,
925 plist->stat.size, &plist->name[1]));
926 #endif
927 }
928
929 return 0;
930 }
931
932
933 /******************************************************************************
934 * Normal Tests
935 ******************************************************************************/
936
937 int case_format(int p0, int p1)
938 {
939 error = tffs_preformat(0xD00D);
940 expect(error, EFFS_INVALID);
941 error = tffs_preformat(0xDEAD);
942 expect(error, EFFS_OK);
943 error = tffs_format("/ffs/", 0xDEAD);
944 expect(error, EFFS_INVALID);
945 error = tffs_format("ffs", 0x2BAD);
946 expect(error, EFFS_BADNAME);
947 error = tffs_format("", 0x2BAD);
948 expect(error, EFFS_BADNAME);
949 error = tffs_format("/", 0x2BAD);
950 expect(error, EFFS_OK);
951 error = tffs_format("/ffs", 0x2BAD);
952 expect(error, EFFS_NOPREFORMAT);
953 error = tffs_preformat(0xDEAD);
954 expect(error, EFFS_OK);
955 error = tffs_format(0, 0x2BAD);
956 //error = tffs_format("/ffs/i256o128", 0x2BAD);
957 expect(error, EFFS_OK);
958 return 0;
959 }
960
961 // Test that it is illegal to modify root inode as if it was a normal file
962 // or directory
963 int case_root(int p0, int p1)
964 {
965 error = tffs_opendir("/", &dir);
966 expect_ok(error);
967 error = tffs_fcreate("/", "foo", 3);
968 expect(error, EFFS_EXISTS);
969 error = tffs_fupdate("/", "bar", 3);
970 expect(error, EFFS_NOTAFILE);
971 error = tffs_fwrite("/", "foo", 3);
972 expect(error, EFFS_NOTAFILE);
973 error = tffs_remove("/");
974 expect(error, EFFS_ACCESS);
975 return 0;
976 }
977
978 // Test object lookup, object names etc.
979 int case_lookup(int p0, int p1)
980 {
981 // Swedish Provinces: Blekinge, Smaaland, Halland, Vaermland, Dalarna,
982 // Dalsland, Gotland, Gaestrikland, Haelsingland, Bohuslaen,
983 // Haerjedalen, Jaemtland, Lappland, Medelpad, Norrbotten, Naerke,
984 // Soedermanland, Uppland, Vaesterbotten
985
986 // Smaaland Cities: Vetlanda, Bodafors, Rottne, Ljungby, Nybro,
987 // Hultsfred, Oskarshamn, Vimmerby, Hyltebruk, Joenkoeping, Vaexjoe
988
989 tffs_mkdir("/europe");
990 tffs_mkdir("/europe/sweden");
991
992 // test init
993 error = tffs_mkdir(LUDIR "/Smaaland");
994 expect(error, EFFS_OK);
995 // test BAD_FILENAME
996 error = tffs_fcreate("", TDATA(0));
997 expect(error, EFFS_BADNAME);
998 // test EFFS_EXISTS
999 error = tffs_fcreate(LUDIR "/Smaaland/vetlanda", TDATA(0));
1000 expect(error, EFFS_OK);
1001 error = tffs_fcreate(LUDIR "/Smaaland/vetlanda", TDATA(0));
1002 expect(error, EFFS_EXISTS);
1003 error = tffs_mkdir(LUDIR "/Smaaland");
1004 expect(error, EFFS_EXISTS);
1005 error = tffs_fwrite(LUDIR "/Smaaland/vetlanda", TDATA(1));
1006 expect(error, EFFS_OK);
1007 // test EFFS_BADNAME
1008 error = tffs_fcreate(LUDIR "/Smaaland/A_Zaz.0+9-7!", TDATA(2));
1009 expect(error, EFFS_BADNAME);
1010 error = tffs_fcreate(LUDIR "/Smaaland/A_Zaz.0+9-7#%$", TDATA(2));
1011 expect(error, EFFS_OK);
1012 // test ending slash
1013 error = tffs_mkdir(LUDIR "/Smaaland/Vaexjoe/");
1014 expect(error, EFFS_NOTADIR);
1015 error = tffs_fcreate(LUDIR "/Smaaland/Vaexjoe/", TDATA(3));
1016 expect(error, EFFS_NOTADIR);
1017 // test EFFS_NAMETOOLONG
1018 error = tffs_fcreate(LUDIR "/Smaaland/Hultsfred-is-21-chars", TDATA(4));
1019 expect(error, EFFS_NAMETOOLONG);
1020 error = tffs_fcreate(LUDIR "/Smaaland/Bodafors-is-20-chars", TDATA(4));
1021 expect(error, EFFS_OK);
1022 error = tffs_mkdir(LUDIR "/Vaermland-is-21-chars");
1023 expect(error, EFFS_NAMETOOLONG);
1024 error = tffs_mkdir(LUDIR "/Dalsland-is-20-chars");
1025 expect(error, EFFS_OK);
1026 // test EFFS_NOTADIR
1027 error = tffs_mkdir(LUDIR "/DontMakeSeveral/DirsAt/TheSameTime");
1028 expect(error, EFFS_NOTADIR);
1029 error = tffs_fread(LUDIR "/Lappland/Bodafors-is-20-chars", bigbuf, 1024);
1030 tw(tr(TR_END, TrApi, "} %d\n", error)); // Avoid wrong indent
1031 expect(error, EFFS_NOTADIR);
1032 error = tffs_fread(LUDIR "/Lappland/", bigbuf, 1024);
1033 tw(tr(TR_END, TrApi, "} %d\n", error)); // Avoid wrong indent
1034 expect(error, EFFS_NOTADIR);
1035 error = tffs_fread(LUDIR "/Smaaland/Bodafors", TDATA(4));
1036 tw(tr(TR_END, TrApi, "} %d\n", error)); // Avoid wrong indent
1037 expect(error, EFFS_NOTFOUND);
1038 // test EFFS_PATHTOODEEP
1039 error = tffs_mkdir(LUDIR "/Gotland"); // 3. level
1040 expect(error, EFFS_OK);
1041 error = tffs_mkdir(LUDIR "/Gotland/Visby"); // 4. level
1042 expect(error, EFFS_OK);
1043 error = tffs_mkdir(LUDIR "/Gotland/Visby/level5"); // 5. level
1044 expect(error, EFFS_OK);
1045 error = tffs_mkdir(LUDIR "/Gotland/Visby/level5/level6"); // 6. level
1046 expect(error, EFFS_OK);
1047 error = tffs_mkdir(LUDIR "/Gotland/Visby/level5/level6/level7"); // 7. level
1048 expect(error, EFFS_PATHTOODEEP);
1049 error = tffs_fcreate(LUDIR "/Gotland/Visby/level5/level6/level7",
1050 TDATA(5)); // 7. level
1051 expect(error, EFFS_PATHTOODEEP);
1052
1053 // final checks
1054 error = test_expect_file(LUDIR "/Smaaland/vetlanda", TDATA(1));
1055 if (error) return 1;
1056 error = test_expect_file(LUDIR "/Smaaland/Bodafors-is-20-chars", TDATA(4));
1057 if (error) return 1;
1058 error = tffs_opendir(LUDIR "/Dalsland-is-20-chars", &dir);
1059 expect_ok(error);
1060 error = tffs_opendir(LUDIR "/Gotland/Visby/level5/level6", &dir);
1061 expect_ok(error);
1062
1063 // cleanup
1064 error = tffs_remove(LUDIR "/Smaaland/A_Zaz.0+9-7");
1065 return 0;
1066 }
1067
1068
1069 // Test fcontrol and read-only semantics. TODO: We still need to perform
1070 // same tests on a dir and a symlink and thru a symlink.
1071 int case_fcontrol(int p0, int p1)
1072 {
1073 struct ffs_state_s old, new;
1074 const char rofile[] = "/europe/norway/rofile";
1075 const char imeifile[] = "/europe/norway/IMEI";
1076 fd_t fdi;
1077
1078 // Cleanup
1079 tffs_remove(rofile);
1080 tffs_remove("/europe/norway");
1081 tffs_remove("/europe");
1082
1083 // Initialize
1084 error = tffs_mkdir("/europe");
1085 error = tffs_mkdir("/europe/norway");
1086 error = tffs_fcreate(rofile, TDATA(2));
1087 expect(error, EFFS_OK);
1088 error = tffs_stat(rofile, &stat);
1089 expect(error, EFFS_OK);
1090 expect_eq(stat.flags, 0);
1091
1092 test_ffs_state_get(&old);
1093
1094 // set read-only flag
1095 error = tffs_fcontrol(rofile, OC_FLAGS, OF_READONLY);
1096 expect(error, EFFS_OK);
1097 error = tffs_stat(rofile, &stat);
1098 expect(error, EFFS_OK);
1099 expect_eq(stat.flags, OF_READONLY);
1100
1101 test_ffs_state_get(&new);
1102 expect_eq(new.objects_total, old.objects_total);
1103 expect_ne(new.inodes_used, old.inodes_used);
1104 test_ffs_state_copy(&old, &new);
1105
1106 // Set illegal flags. Try to fupdate file. Then try to set read-only
1107 // flag again.
1108 error = tffs_fcontrol(rofile, OC_FLAGS, 1<<0);
1109 expect(error, EFFS_INVALID);
1110 error = tffs_fcontrol(rofile, OC_FLAGS, 1<<3);
1111 expect(error, EFFS_INVALID);
1112 error = tffs_fcontrol(rofile, OC_FLAGS, 1<<5);
1113 expect(error, EFFS_INVALID);
1114 error = tffs_fcontrol(rofile, OC_FLAGS, 1<<6);
1115 expect(error, EFFS_INVALID);
1116 error = tffs_fupdate(rofile, TDATA(3));
1117 expect(error, EFFS_OK);
1118
1119 error = tffs_fcreate("/europe/norway/tease", TDATA(2));
1120 expect(error, EFFS_OK);
1121 tffs_remove("/europe/norway/tease");
1122
1123 error = tffs_fcontrol(rofile, OC_FLAGS, OF_READONLY);
1124 expect(error, EFFS_OK);
1125 error = tffs_stat(rofile, &stat);
1126 expect(error, EFFS_OK);
1127 expect_eq(stat.flags, OF_READONLY);
1128
1129 test_ffs_state_get(&new);
1130 expect_eq(new.objects_total, old.objects_total);
1131 expect_ne(new.inodes_used, old.inodes_used);
1132 test_ffs_state_copy(&old, &new);
1133
1134 // clear read-only flag (this works because ffs_is_modifiable() by
1135 // default returns true for all objects except for object names ending
1136 // in "IMEI".
1137 error = tffs_fcontrol(rofile, OC_FLAGS, 0);
1138 expect(error, EFFS_OK);
1139 error = tffs_stat(rofile, &stat);
1140 expect(error, EFFS_OK);
1141 expect_eq(stat.flags, 0);
1142
1143 test_ffs_state_get(&new);
1144 expect_eq(new.objects_total, old.objects_total);
1145 expect_ne(new.inodes_used, old.inodes_used);
1146 test_ffs_state_copy(&old, &new);
1147
1148 // Set read-only flag (again)
1149 error = tffs_fcontrol(rofile, OC_FLAGS, OF_READONLY);
1150 expect(error, EFFS_OK);
1151 error = tffs_stat(rofile, &stat);
1152 expect(error, EFFS_OK);
1153 expect_eq(stat.flags, OF_READONLY);
1154
1155 test_ffs_state_get(&new);
1156 expect_eq(new.objects_total, old.objects_total);
1157 expect_ne(new.inodes_used, old.inodes_used);
1158 test_ffs_state_copy(&old, &new);
1159
1160 // Set read-only flag of IMEI file
1161 error = tffs_fcreate(imeifile, TDATA(2));
1162 expect(error, EFFS_OK);
1163 error = tffs_stat(imeifile, &stat);
1164 expect(error, EFFS_OK);
1165 expect_eq(stat.flags, 0);
1166 error = tffs_fcontrol(imeifile, OC_FLAGS, OF_READONLY);
1167 expect(error, EFFS_OK);
1168 error = tffs_stat(imeifile, &stat);
1169 expect(error, EFFS_OK);
1170 expect_eq(stat.flags, OF_READONLY);
1171
1172 test_ffs_state_get(&new);
1173 expect_eq(new.objects_total, old.objects_total + 1);
1174 expect_ne(new.inodes_used, old.inodes_used);
1175 test_ffs_state_copy(&old, &new);
1176
1177 // Try to remove, fupdate, fwrite and fcontrol IMEI file.
1178 error = tffs_remove(imeifile);
1179 expect(error, EFFS_ACCESS);
1180 error = tffs_fupdate(imeifile, TDATA(0));
1181 expect(error, EFFS_ACCESS);
1182 error = tffs_fwrite(imeifile, TDATA(0));
1183 expect(error, EFFS_ACCESS);
1184 error = tffs_fcontrol(imeifile, OC_FLAGS, 0);
1185 expect(error, EFFS_ACCESS);
1186
1187 // Try to open IMEI file in write-only and read-only
1188 fdi = tffs_open(imeifile, FFS_O_WRONLY | FFS_O_APPEND);
1189 expect(fdi, EFFS_ACCESS);
1190
1191 fdi = tffs_open(imeifile, FFS_O_RDONLY);
1192 expect(fdi, FFS_FD_OFFSET);
1193 error = tffs_close(fdi);
1194 expect(error, EFFS_OK);
1195
1196 return 0;
1197 }
1198
1199 // Test symlink functionality of simple symlink implementation.
1200 // Fixme: add remove file through symlink, stat a dir through a symlink.
1201 int case_ssym(int p0, int p1)
1202 {
1203 int size;
1204 fd_t fdi;
1205 // two links, read files thru links
1206 // link to link to file, open file
1207 // link to non-valid object
1208 //
1209 // make three test files, a link to each of these, one link to dir, one
1210 // link to non-existing object, one link to link to file
1211
1212 tffs_mkdir("/europe");
1213 tffs_mkdir("/europe/denmark");
1214
1215 error = tffs_fwrite("/europe/denmark/aalborg", TDATA(1));
1216 expect(error, EFFS_OK);
1217 error = tffs_symlink("/europe/aal", "/europe/denmark/aalborg");
1218 expect(error, EFFS_OK);
1219 error = tffs_symlink("/dk", "/europe/denmark");
1220 expect(error, EFFS_OK);
1221 error = tffs_fwrite("/europe/denmark/aarhus", TDATA(2));
1222 expect(error, EFFS_OK);
1223 error = tffs_symlink("/europe/aar", "/europe/denmark/aarhus");
1224 expect(error, EFFS_OK);
1225 error = tffs_symlink("/europe/se", "/europe/non-existing");
1226 expect(error, EFFS_OK);
1227 error = tffs_fwrite("/europe/denmark/billund", TDATA(3));
1228 expect(error, EFFS_OK);
1229 error = tffs_symlink("/europe/bil", "/europe/denmark/billund");
1230 expect(error, EFFS_OK);
1231 error = tffs_symlink("/lego", "/europe/bil");
1232 expect(error, EFFS_OK);
1233 error = tffs_fwrite("/europe/denmark/norresundby", TDATA(2));
1234 expect(error, EFFS_OK);
1235 error = tffs_symlink("/europe/nor", "/europe/denmark/norresundby");
1236 expect(error, EFFS_OK);
1237
1238 // Test link to dir
1239 error = tffs_opendir("/dk", &dir);
1240 expect(error, EFFS_NOTAFILE); // TODO: strange error!
1241 error = tffs_stat("/dk", &stat);
1242 expect(error, EFFS_NOTAFILE); // TODO: strange error!
1243 error = tffs_linkstat("/dk", &stat);
1244 expect(error, EFFS_OK);
1245
1246 // Test link to link to file
1247 error = tffs_stat("/lego", &stat);
1248 expect(error, EFFS_NOTAFILE); // TODO: strange error?!
1249 error = tffs_linkstat("/lego", &stat);
1250 expect(error, EFFS_OK);
1251 error = tffs_fread("/lego", bigbuf, bigbuf_size);
1252 tw(tr(TR_END, TrApi, "} %d\n", error)); // Avoid wrong indent
1253 expect(error, EFFS_NOTAFILE);
1254
1255 // Test link to non-existing object
1256 error = tffs_opendir("/europe/se", &dir);
1257 expect(error, EFFS_NOTFOUND);
1258 error = tffs_stat("/europe/se", &stat);
1259 expect(error, EFFS_NOTFOUND);
1260 error = tffs_linkstat("/europe/se", &stat);
1261 expect(error, EFFS_OK);
1262
1263 // Read files through links
1264 error = test_expect_file("/europe/aal", TDATA(1));
1265 if (error) return 1;
1266 error = test_expect_file("/europe/aar", TDATA(2));
1267 if (error) return 1;
1268 error = test_expect_file("/europe/bil", TDATA(3));
1269 if (error) return 1;
1270
1271 // Write files through links
1272 error = tffs_fwrite("/europe/aal", TDATA(3));
1273 expect(error, EFFS_OK);
1274 error = tffs_fupdate("/europe/aar", TDATA(4));
1275 expect(error, EFFS_OK);
1276 error = tffs_fupdate("/europe/bil", TDATA(5));
1277 expect(error, EFFS_OK);
1278
1279 // Open file and write and read files through links
1280 fdi = tffs_open("/europe/nor", FFS_O_WRONLY | FFS_O_TRUNC | FFS_O_APPEND);
1281 expect(fdi, FFS_FD_OFFSET);
1282 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], 50);
1283 expect(size, 50);
1284 error = tffs_close(fdi);
1285 expect(error, 0);
1286
1287 fdi = tffs_open("/europe/nor", FFS_O_RDONLY);
1288 expect(fdi, FFS_FD_OFFSET);
1289 size = tffs_read(fdi, bigbuf, 62);
1290 expect(size, 50);
1291 error = test_expect_data((char*)tdata[TDATA_HUGE], bigbuf, 50);
1292 if (error) return 1;
1293 error = tffs_close(fdi);
1294 expect(error, 0);
1295
1296 // remove "lego" link, recreate it to link to "billund", read file data
1297 // through link, re-write file data through link
1298 error = tffs_symlink("/lego", "/europe/denmark/billund");
1299 expect(error, EFFS_NOTAFILE); // TODO: strange error?!
1300 error = tffs_fupdate("/lego", "/europe/denmark/billund",
1301 strlen("/europe/denmark/billund"));
1302 expect(error, EFFS_NOTAFILE);
1303 error = tffs_remove("/lego");
1304 expect(error, EFFS_OK);
1305 error = tffs_symlink("/lego", "/europe/denmark/billund");
1306 expect(error, EFFS_OK);
1307 error = test_expect_file("/lego", TDATA(5));
1308 if (error) return 1;
1309 error = tffs_fupdate("/lego", TDATA(2));
1310 expect(error, EFFS_OK);
1311
1312 // Re-Read files through links
1313 error = test_expect_file("/europe/aal", TDATA(3));
1314 if (error) return 1;
1315 error = test_expect_file("/europe/aar", TDATA(4));
1316 if (error) return 1;
1317 error = test_expect_file("/europe/bil", TDATA(2));
1318 if (error) return 1;
1319
1320 // Clean up
1321 error = tffs_remove("/europe/aal");
1322 expect(error, EFFS_OK);
1323 error = tffs_remove("/dk");
1324 expect(error, EFFS_OK);
1325 error = tffs_remove("/europe/aar");
1326 expect(error, EFFS_OK);
1327 error = tffs_remove("/europe/se");
1328 expect(error, EFFS_OK);
1329 error = tffs_remove("/europe/bil");
1330 expect(error, EFFS_OK);
1331 error = tffs_remove("/lego");
1332 expect(error, EFFS_OK);
1333 error = tffs_remove("/europe/nor");
1334 expect(error, EFFS_OK);
1335
1336 return 0;
1337 }
1338
1339 // Test symlink functionality of full symlink implementation
1340 int case_fsym(int p0, int p1)
1341 {
1342 fd_t fdi;
1343 // make relative link to directory, stat link and directory
1344 error = tffs_symlink("/sa", "/south-america"); // relative link
1345 expect(error, EFFS_OK);
1346
1347 error = tffs_linkstat("/sa", &stat);
1348 expect(error, EFFS_OK);
1349 expect_eq(stat.type, OT_LINK);
1350
1351 error = tffs_stat("/sa", &stat);
1352 expect(error, EFFS_OK);
1353 expect_eq(stat.type, OT_DIR);
1354
1355 // create directory thru a symlink
1356 error = tffs_mkdir("/sa/brazil");
1357 expect(error, EFFS_OK);
1358 error = tffs_symlink("/br.ba", "/brazil/buenos-aires"); // relative link
1359 expect(error, EFFS_OK);
1360
1361 // create file via symlink, stat the new file
1362 error = tffs_fcreate("/sa/brazil/buenos-aires", TDATA(2));
1363 expect(error, EFFS_OK);
1364 error = tffs_stat("/sa/br.ba", &stat);
1365 expect(error, EFFS_OK);
1366 expect_eq(stat.type, OT_FILE);
1367
1368 fdi = tffs_open("/sa/peru", FFS_O_WRONLY | FFS_O_TRUNC |
1369 FFS_O_APPEND | FFS_O_CREATE);
1370 expect(fdi, FFS_FD_OFFSET);
1371 error = tffs_close(fdi);
1372 expect(error, 0);
1373
1374 // create file thru a symlink, stat the new file. Will this work?
1375 error = tffs_symlink("/br.br", "/brazil/brasilia"); // relative link
1376 expect(error, EFFS_OK);
1377 error = tffs_fcreate("/sa/br.br", TDATA(3)); // ???
1378 expect(error, EFFS_OK);
1379 error = tffs_stat("/sa/br.br", &stat);
1380 expect(error, EFFS_OK);
1381 expect_eq(stat.type, OT_FILE);
1382
1383 // Create symlink that is a link to absolute link path
1384 error = tffs_symlink("/south-america/cape-horn", "/sa/ar"); // absolute link
1385 expect(error, EFFS_OK);
1386
1387 error = tffs_fcreate("/south-america/argentina", TDATA(0));
1388
1389 expect(error, EFFS_OK);
1390 error = tffs_symlink("/sa/ar", "/south-america/argentina");
1391 expect(error, EFFS_OK);
1392
1393 // TODO: Test very deep path
1394
1395 // TODO: Test circular link
1396
1397 // TODO: Test if ending slash is allowed on dirs and symlinks
1398
1399 tw(tr(TR_FUNC, TrTest, "WARNING: Test case not implemented!\n"));
1400 ttw(str(TTrTest, "Test case not implemented!" NL));
1401
1402 return 0;
1403 }
1404
1405 // Check every combination of name length and data size within the device
1406 // atomsize,
1407 int case_fread(int p0, int p1)
1408 {
1409 const char fullname[] = "123456789ABCDEF0123456789ABCDEF";
1410 int i, j, dirlen, mysize;
1411 char *mydata;
1412 char myname[20+20+32] = "/australia/f";
1413
1414 case_cleanup(0x11 * param.atomsize * param.atomsize);
1415
1416 tffs_mkdir("/australia");
1417
1418 mydata = (char *) tdata[TDATA_HUGE];
1419 dirlen = strlen(myname);
1420
1421 for (j = 0; j < param.atomsize; j++) {
1422 mysize = j * 0x11 /* + 0x100 */;
1423 ttw(ttr(TTrTest, "frd: size = %x" NL, mysize));
1424 tw(tr(TR_FUNC, TrTestHigh, "frd: size = %x\n", mysize));
1425 // remove files
1426 for (i = 0; i < param.atomsize; i++) {
1427 strncpy(&myname[dirlen], fullname, i);
1428 myname[dirlen + i] = 0;
1429 error = tffs_remove(myname);
1430 }
1431 // fcreate files with varying name lengths but same alignment.
1432 for (i = 0; i < param.atomsize; i++) {
1433 strncpy(&myname[dirlen], fullname, i);
1434 myname[dirlen + i] = 0;
1435 error = tffs_fwrite(myname, mydata, mysize);
1436 expect(error, EFFS_OK);
1437 }
1438 // Now check all the files written
1439 for (i = 0; i < param.atomsize; i++) {
1440 strncpy(&myname[dirlen], fullname, i);
1441 myname[dirlen + i] = 0;
1442 //tw(tr(TR_FUNC, TrTestHigh, "frd: txf('f...', %x, %d)\n",
1443 // mydata, mysize));
1444 error = test_expect_file(myname, mydata, mysize);
1445 if (error) return 1;
1446 }
1447 mydata += 0x10;
1448 }
1449
1450 for (i = 0; i < param.atomsize; i++) {
1451 strncpy(&myname[dirlen], fullname, i);
1452 myname[dirlen + i] = 0;
1453 error = tffs_remove(myname);
1454 expect(error, EFFS_OK);
1455 }
1456 return 0;
1457 }
1458
1459
1460 /******************************************************************************
1461 * Non-finished
1462 ******************************************************************************/
1463
1464 // Test Directories
1465 int case_dirs(int p0, int p1)
1466 {
1467 // remove empty dir, non-empty dir
1468 // open/readdir empty dir, non-empty dir
1469 char name[21];
1470 int i, j;
1471 struct dir_s dir[3];
1472 const char *names[3][5] = {
1473 { "china", "japan", "korea", "india", 0 },
1474 { "bombay", "newdelhi", 0, 0, 0 },
1475 { "hongkong", "shanghai", "hk", 0, 0 }
1476 };
1477
1478 // Cleanup
1479 tffs_mkdir("/asia");
1480 tffs_remove("/asia/india/bombay");
1481 tffs_remove("/asia/india/newdelhi");
1482 tffs_remove("/asia/india");
1483
1484 error = tffs_mkdir("/asia/china");
1485 expect(error, EFFS_OK);
1486 error = tffs_mkdir("/asia/china/beijing");
1487 expect(error, EFFS_OK);
1488 error = tffs_mkdir("/asia/china/hongkong");
1489 expect(error, EFFS_OK);
1490 error = tffs_fcreate("/asia/china/hongkong/hkfile1", TDATA(1));
1491 expect(error, EFFS_OK);
1492 error = tffs_fcreate("/asia/china/shanghai", TDATA(2));
1493 expect(error, EFFS_OK);
1494 error = tffs_symlink("/asia/china/hk", "/asia/china/hongkong");
1495 expect(error, EFFS_OK);
1496 error = tffs_mkdir("/asia/japan");
1497 expect(error, EFFS_OK);
1498 error = tffs_fcreate("/asia/thailand", TDATA(0));
1499 expect(error, EFFS_OK);
1500 error = tffs_fcreate("/asia/korea", TDATA(2));
1501 expect(error, EFFS_OK);
1502 error = tffs_fcreate("/asia/japan/tokyo", TDATA(3));
1503 expect(error, EFFS_OK);
1504 error = tffs_fupdate("/asia/japan/tokyo", TDATA(4));
1505 expect(error, EFFS_OK);
1506 error = tffs_mkdir("/asia/india");
1507 expect(error, EFFS_OK);
1508 error = tffs_fcreate("/asia/india/bombay", TDATA(0));
1509 expect(error, EFFS_OK);
1510 error = tffs_fcreate("/asia/india/newdelhi", TDATA(1));
1511 expect(error, EFFS_OK);
1512 error = tffs_fcreate("/asia/india/calcutta", TDATA(2));
1513 expect(error, EFFS_OK);
1514
1515 error = tffs_opendir("/asia", &dir[0]);
1516 expect(error, 5);
1517 error = tffs_opendir("/asia/india", &dir[1]);
1518 expect(error, 3);
1519 error = tffs_opendir("/asia/china", &dir[2]);
1520 expect(error, 4);
1521
1522 // remove first, middle and last entry in a dir
1523 error = tffs_remove("/asia/china/beijing");
1524 expect(error, EFFS_OK);
1525 error = tffs_remove("/asia/thailand");
1526 expect(error, EFFS_OK);
1527 error = tffs_remove("/asia/india/calcutta");
1528 expect(error, EFFS_OK);
1529
1530 for (j = 0; j < 5; j++) {
1531 for (i = 0; i < 3; i++) {
1532 error = tffs_readdir(&dir[i], name, 21);
1533 if (names[i][j] == NULL) {
1534 expect(error, EFFS_OK);
1535 }
1536 else {
1537 expect_gt(error, EFFS_OK);
1538 tw(tr(TR_FUNC, TrTestHigh, "dir[%d]: %10s, expected: %s\n",
1539 i, name, names[i][j]));
1540 test_expect_data(name, TDATA_STRING(names[i][j]));
1541 }
1542 }
1543 }
1544
1545 error = tffs_remove("/asia/china");
1546 expect(error, EFFS_DIRNOTEMPTY);
1547 error = tffs_remove("/asia/china/hongkong");
1548 expect(error, EFFS_DIRNOTEMPTY);
1549 error = tffs_remove("/asia/china/hongkong/hkfile1");
1550 expect(error, EFFS_OK);
1551 error = tffs_remove("/asia/china/shanghai");
1552 expect(error, EFFS_OK);
1553 error = tffs_remove("/asia/china/hk");
1554 expect(error, EFFS_OK);
1555 error = tffs_remove("/asia/china");
1556 expect(error, EFFS_DIRNOTEMPTY);
1557 error = tffs_remove("/asia/china/hongkong");
1558 expect(error, EFFS_OK);
1559 error = tffs_opendir("/asia/china", &dir[2]);
1560 expect(error, 0);
1561 error = tffs_remove("/asia/china");
1562 expect(error, EFFS_OK);
1563
1564 error = tffs_remove("/asia/korea");
1565 expect(error, EFFS_OK);
1566 error = tffs_remove("/asia/japan/tokyo");
1567 expect(error, EFFS_OK);
1568 error = tffs_remove("/asia/japan");
1569 expect(error, EFFS_OK);
1570
1571 return 0;
1572 }
1573
1574 // Check that expected stat data was read
1575 int case_expect_stat(const char *name, int n)
1576 {
1577 error = tffs_stat(name, &stat);
1578 expect(error, EFFS_OK);
1579 // test type, size, flags, space?
1580 // test range of location, inode, block
1581 return 0;
1582 }
1583
1584 // Test stat
1585 int case_stat(int p0, int p1)
1586 {
1587 struct xstat_s xstat;
1588 struct stat_s stat;
1589 char myname[] = "/Stat_file";
1590 char sym_name[] = "/sf";
1591 char stream_name[] = "/Stat_stream";
1592 fd_t fdi;
1593 int size, i, file_size = 0;
1594
1595 // test stat on dirs, symlinks and files
1596 // check stat.block (by writing two HUGE files)
1597
1598 case_cleanup(fs.chunk_size_max / 5 * 20);
1599
1600 // NOTEME: this is a very limited test
1601 error = tffs_fwrite(myname, (char *)tdata[TDATA_HUGE], 99);
1602 expect_ok(error);
1603
1604 error = tffs_stat(myname, &stat);
1605 expect(error, EFFS_OK);
1606 expect(stat.type, OT_FILE);
1607 expect(stat.flags, 0);
1608 expect_ok(stat.inode);
1609 expect(stat.size, 99);
1610
1611 // error = tffs_xstat(myname, &xstat);
1612 // expect(error, EFFS_OK);
1613
1614 // expect(error, EFFS_OK);
1615 // expect(xstat.type, OT_FILE);
1616 // expect(xstat.flags, 0);
1617 // expect_ok(xstat.inode);
1618 // expect(xstat.size, 99);
1619
1620 // expect(xstat.space, 112);
1621 // expect_ok(xstat.location);
1622 // expect_ok(xstat.block);
1623 // expect_ok(xstat.sequence);
1624 // expect_ok(xstat.updates);
1625
1626 error = tffs_linkstat(myname, &stat);
1627 expect(error, EFFS_OK);
1628
1629 expect(error, EFFS_OK);
1630 expect(stat.type, OT_FILE);
1631 expect(stat.flags, 0);
1632 expect_ok(stat.inode);
1633 expect(stat.size, 99);
1634
1635 error = tffs_xlstat(myname, &xstat);
1636 expect(error, EFFS_OK);
1637
1638 expect(error, EFFS_OK);
1639 expect(xstat.type, OT_FILE);
1640 expect(xstat.flags, 0);
1641 expect_ok(xstat.inode);
1642 expect(xstat.size, 99);
1643
1644 expect(xstat.space, 112);
1645 expect_ok(xstat.location);
1646 expect_ok(xstat.block);
1647 expect_ok(xstat.sequence);
1648 expect_ok(xstat.updates);
1649
1650 // Symlink
1651 tffs_symlink(sym_name, myname);
1652
1653 error = tffs_xlstat(sym_name, &xstat);
1654 expect(error, EFFS_OK);
1655
1656 expect(error, EFFS_OK);
1657 expect(xstat.type, OT_LINK);
1658 expect(xstat.flags, 0);
1659 expect_ok(xstat.inode);
1660 expect(xstat.size, sizeof(myname));
1661
1662 expect(xstat.space, 16);
1663 expect_ok(xstat.location);
1664 expect_ok(xstat.block);
1665 expect_ok(xstat.sequence);
1666 expect_ok(xstat.updates);
1667
1668 // Stream stat test
1669 // Use xstat to return the size of the file
1670 fdi = tffs_open(stream_name, FFS_O_WRONLY | FFS_O_CREATE |
1671 FFS_O_TRUNC | FFS_O_APPEND);
1672 expect(fdi, FFS_FD_OFFSET);
1673
1674 for (i = 0; i < 20; i++) {
1675 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE],
1676 fs.chunk_size_max / 5);
1677
1678 expect(size, fs.chunk_size_max / 5);
1679 file_size +=size;
1680
1681 error = tffs_xlstat(stream_name, &xstat);
1682 expect(error, EFFS_OK);
1683 expect_eq(xstat.size, file_size);
1684
1685 error = tffs_fstat(fdi, &stat);
1686 expect(error, EFFS_OK);
1687 expect_eq(stat.size, file_size);
1688 }
1689
1690 error = tffs_close(fdi);
1691 expect(error, EFFS_OK);
1692
1693 // Test if the file size is right if opened in read-only
1694 fdi = tffs_open(stream_name, FFS_O_RDONLY);
1695 expect(fdi, FFS_FD_OFFSET);
1696
1697 error = tffs_xlstat(stream_name, &xstat);
1698 expect(error, EFFS_OK);
1699 expect_eq(xstat.size, file_size);
1700
1701 // Test a lot of difference fdi
1702 for (i = -3; i < FFS_FD_OFFSET + fs.fd_max + 3; i++) {
1703 error = tffs_fstat(i, &stat);
1704 if (i == FFS_FD_OFFSET) {
1705 expect(error, EFFS_OK);
1706 }
1707 else expect(error, EFFS_BADFD);
1708 }
1709 error = tffs_close(fdi);
1710 expect(error, EFFS_OK);
1711
1712 return 0;
1713 }
1714
1715 int case_remove(int p0, int p1)
1716 {
1717 tw(tr(TR_FUNC, TrTest, "WARNING: Test case not implemented!\n"));
1718 ttw(str(TTrTest, "Test case not implemented!" NL));
1719
1720 return 0;
1721 }
1722
1723 int case_rename(int p0, int p1)
1724 {
1725 fd_t fdi;
1726 int size, i;
1727 struct dir_s dir;
1728 // create file A
1729 // rename A to B
1730 // check file A is gone
1731 // check file B is same as file A
1732 // do all the same for directory
1733 // do all the same for symlink
1734 // test and ensure that dir rename "foo" to "foo/bar" is impossible!
1735
1736 // Cleanup before run
1737 tffs_remove("/Walt_Disney/RUP");
1738 tffs_remove("/Uncle_Scrooge");
1739 tffs_remove("/Walt_Disney/Minnie");
1740 tffs_remove("/Walt_Disney");
1741 tffs_remove("/Duck");
1742
1743 // Init
1744 tffs_mkdir("/Disney");
1745 tffs_mkdir("/Hell");
1746
1747 error = tffs_fwrite("/IMEI", TDATA(2));
1748 if (error < 0 && error != EFFS_ACCESS) return 1;
1749 error = tffs_fcontrol("/IMEI", OC_FLAGS, OF_READONLY);
1750 if (error < 0 && error != EFFS_ACCESS) return 1;
1751
1752 tw(tr(TR_FUNC, TrTestHigh, "Rename file\n")); /* Rename file */
1753
1754 error = tffs_fwrite("/Mickey", TDATA(3));
1755 expect(error, EFFS_OK);
1756
1757 error = tffs_fwrite("/Pluto", TDATA(2));
1758 expect(error, EFFS_OK);
1759
1760 error = tffs_rename("/Pluto", "/Dog");
1761 expect(error, EFFS_OK);
1762
1763 error = tffs_fupdate("/Pluto", TDATA(2));
1764 expect(error, EFFS_NOTFOUND);
1765
1766 error = test_expect_file("/Dog", TDATA(2));
1767 if (error) return 1;
1768
1769 error = tffs_rename("/Dog", "/Hell/RIP");
1770 expect(error, EFFS_OK);
1771
1772 error = tffs_rename("/Hell/RIP", "/RAP");
1773 expect(error, EFFS_OK);
1774
1775 error = tffs_rename("/RAP", "/Disney/RUP");
1776 expect(error, EFFS_OK);
1777
1778 error = test_expect_file("/Disney/RUP", TDATA(2));
1779 if (error) return 1;
1780
1781 // Rename a file to an existing file
1782 error = tffs_rename("/Mickey", "/Disney/RUP");
1783 expect(error, EFFS_OK);
1784
1785 // Have the data changed?
1786 error = test_expect_file("/Disney/RUP", TDATA(3));
1787 if (error) return 1;
1788
1789 // Try rename a file to an exisitng read-only file
1790 error = tffs_rename("/Disney/RUP", "/IMEI");
1791 expect(error, EFFS_ACCESS);
1792
1793 // Try rename a file to an existing dir
1794 error = tffs_rename("/Disney/RUP", "/Hell");
1795 expect(error, EFFS_NOTAFILE);
1796
1797 error = tffs_rename("/Disney/RUP", "/BADNAME?");
1798 expect(error, EFFS_BADNAME);
1799
1800 error = tffs_remove("/Hell");
1801 expect(error, EFFS_OK);
1802
1803 tw(tr(TR_FUNC, TrTestHigh, "Rename symlink\n")); /* Rename symlink */
1804
1805 tffs_symlink("/Goose", "/Disney/RUP"); // 5
1806
1807 error = tffs_rename("/Goose", "/Duck");
1808 expect(error, EFFS_OK);
1809
1810 error = tffs_fupdate("/Goose", TDATA(2));
1811 expect(error, EFFS_NOTFOUND);
1812
1813 error = test_expect_file("/Duck", TDATA(3));
1814 if (error) return 1;
1815
1816 error = test_expect_file("/Disney/RUP", TDATA(3));
1817 if (error) return 1;
1818
1819 tw(tr(TR_FUNC, TrTestHigh, "Rename dir\n")); /* Rename dir */
1820
1821 // * FIXME BUG * FIXME BUG * FIXME BUG * The below test to not fail instead
1822 // the directory is removed
1823 // error = tffs_rename("/Disney", "/Disney/foo");
1824 // expect(error, EFFS_OK);
1825
1826 tffs_mkdir("/Disney/Donald_Duck");
1827 expect(error, EFFS_OK);
1828 error = ffs_rename("/Disney/Donald_Duck", "/Uncle_Scrooge");
1829 expect(error, EFFS_OK);
1830 error = tffs_opendir("/Uncle_Scrooge", &dir);
1831 expect(error, EFFS_OK);
1832 error = tffs_opendir("/Disney/Donald_Duck", &dir);
1833 expect(error, EFFS_NOTFOUND);
1834
1835 error = tffs_rename("/Disney", "/Walt_Disney");
1836 expect(error, EFFS_OK);
1837
1838 tffs_mkdir("/Disney"); // Create 'Disney' dir again
1839 expect(error, EFFS_OK);
1840
1841 // Try rename to existing dir
1842 error = tffs_rename("/Disney", "/Walt_Disney");
1843 expect(error, EFFS_EXISTS);
1844
1845 // Try rename to existing file
1846 error = tffs_rename("/Disney", "/Walt_Disney/RUP");
1847 expect(error, EFFS_EXISTS);
1848
1849 tw(tr(TR_FUNC, TrTestHigh, "Rename seghead\n")); /* Rename seghead */
1850
1851 fdi = tffs_open("/Walt_Disney/Mickey", FFS_O_WRONLY | FFS_O_CREATE);
1852 expect(fdi, FFS_FD_OFFSET);
1853 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], fs.chunk_size_max + 1);
1854 expect(size, fs.chunk_size_max + 1);
1855 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE] + size, fs.chunk_size_max);
1856 expect(size, fs.chunk_size_max);
1857
1858 error = tffs_rename("/Walt_Disney/Mickey", "/Walt_Disney/Minnie");
1859 expect(error, EFFS_LOCKED);
1860 tffs_close(fdi);
1861
1862 error = tffs_rename("/Walt_Disney/Mickey", "/Walt_Disney/Minnie");
1863 expect(error, EFFS_OK);
1864
1865 fdi = tffs_open("/Walt_Disney/Minnie", FFS_O_RDONLY);
1866 expect(fdi, FFS_FD_OFFSET);
1867 i = 0;
1868 do {
1869 size = tffs_read(fdi, bigbuf, bigbuf_size);
1870 error = test_expect_data((char*)tdata[TDATA_HUGE] + i, bigbuf, size);
1871 if (error) return 1;
1872 i += size;
1873 } while (size);
1874 error = tffs_close(fdi);
1875 expect(error, 0);
1876
1877 error = test_expect_file("/Walt_Disney/Minnie", (char *)tdata[TDATA_HUGE],
1878 2 * fs.chunk_size_max + 1);
1879 if (error) return 1;
1880
1881 // Rename a big file to an existing big file
1882 error = tffs_fwrite("/Mickey", (char *)tdata[TDATA_HUGE] + 5,
1883 2.5 * fs.chunk_size_max);
1884 expect(error, EFFS_OK);
1885
1886 error = tffs_rename("/Mickey", "/Walt_Disney/Minnie");
1887 expect(error, EFFS_OK);
1888
1889 error = test_expect_file("/Walt_Disney/Minnie", (char *)tdata[TDATA_HUGE] + 5,
1890 2.5 * fs.chunk_size_max);
1891 if (error) return 1;
1892
1893 error = tffs_fread("/Mickey", 0 , 0);
1894 expect(error, EFFS_NOTFOUND);
1895
1896 return 0;
1897 }
1898
1899 // One of the problems with the rename function is that we need to copy the
1900 // data part from the old obj to the new obj. When we allocate data for the
1901 // new obj we risk that the data alloc caused a data reclaim which relocated
1902 // the old obj thus the source have been moved!
1903 int case_rename_extended(int p0, int p1)
1904 {
1905 int i, old_drec_most_lost, rename_file_relocated = 0;
1906 int fsize, offset;
1907 char myname1[] = "/rename1/rename1/foo";
1908 char myname2[] = "/rename2/rename2/bar";
1909
1910 if (p0 == 0) p0 = 100;
1911 if (p1 == 0) p1 = 5;
1912
1913 error = tffs_preformat(0xDEAD);
1914 expect(error, EFFS_OK);
1915 error = tffs_format("/ffs/b3r24", 0x2BAD);
1916 expect(error, EFFS_OK);
1917
1918 fsize = 2.5 * fs.chunk_size_max; // Test with more than one chunk
1919
1920 tffs_mkdir("/rename1");
1921 tffs_mkdir("/rename1/rename1");
1922 tffs_mkdir("/rename2");
1923 tffs_mkdir("/rename2/rename2");
1924
1925 error = tffs_file_write(myname1, (char*)tdata[TDATA_HUGE], fsize,
1926 FFS_O_CREATE | FFS_O_TRUNC);
1927
1928 // Test if rename can handle when the objects are relocaded while the
1929 // rename is in progress
1930 for (i = 0; i < p0; i++) {
1931 tw(tr(TR_FUNC, TrTest, "Rename number %d\n", i));
1932
1933 offset = i % 2 ? 3 : 9;
1934 error = tffs_file_write(myname2, (char*)tdata[TDATA_HUGE] + offset, fsize,
1935 FFS_O_CREATE | FFS_O_TRUNC);
1936
1937 // Get data reclaim candidate most_lost
1938 old_drec_most_lost = stats.drec.most_lost;
1939 error = tffs_rename(myname2, myname1);
1940 expect(error, EFFS_OK);
1941
1942 error = test_expect_file(myname1, tdata[TDATA_HUGE] + offset, fsize);
1943 expect(error, EFFS_OK);
1944
1945 error = tffs_fread(myname2, 0, 0);
1946 expect(error, EFFS_NOTFOUND);
1947
1948 // Has the rename triggered a data_reclaim()?
1949 if (old_drec_most_lost != stats.drec.most_lost) {
1950 rename_file_relocated++;
1951 tw(tr(TR_FUNC, TrTest, "Rename objects relocated %d\n",
1952 rename_file_relocated));
1953 if (rename_file_relocated >= p1)
1954 return EFFS_OK;
1955 }
1956 }
1957
1958 return 1;
1959 }
1960
1961
1962 // Test Big files
1963 int case_bigfile(int p0, int p1)
1964 {
1965 struct ffs_state_s old, new;
1966 const char bigfile[] = "/antarctica/iceberg";
1967 int bytes_max, file_size;
1968
1969 #if 1
1970 ttw(ttr(TTrTest, "WARNING: We need to re-implement this test. Skip test" NL));
1971 tw(tr(TR_FUNC, TrTest, "WARNING: We need to re-implement this test. Skip test\n"));
1972 return 0;
1973 #endif
1974
1975 if (param.data_blocks <= 1) {
1976 ttw(ttr(TTrTest, "WARNING: Too few blocks to run. Skip test" NL));
1977 tw(tr(TR_FUNC, TrTest, "WARNING: Too few blocks to run. Skip test\n"));
1978 return 0;
1979 }
1980
1981 error = case_cleanup(fs.filesize_max);
1982 expect(error, EFFS_NOSPACE); // We can not cleanup this amount.
1983
1984 ffs_query(Q_BYTES_FREE, (uint32 *) &bytes_max);
1985 test_ffs_state_get(&old);
1986
1987 // We don't have enough space for this amount because the name dos also
1988 // use some space
1989 file_size = bytes_max + 1;
1990
1991 // Try to make a file of a size that is bigger than the total free space
1992 // in the file system.
1993 ttw(ttr(TTrTest, "Bigfile of size %d" NL, file_size));
1994 tw(tr(TR_FUNC, TrTestHigh, "Bigfile of size %d\n", file_size));
1995 error = tffs_fcreate(bigfile,
1996 (char *) tdata[TDATA_HUGE], file_size);
1997 expect(error, EFFS_NOSPACE);
1998
1999 test_ffs_state_get(&new);
2000 expect_eq(old.bytes_free, new.bytes_free);
2001
2002 // File the system with this huge file
2003 file_size = bytes_max - FFS_FILENAME_MAX - dev.atomsize;
2004 ttw(ttr(TTrTest, "Bigfile of size %d" NL, file_size));
2005 tw(tr(TR_FUNC, TrTestHigh, "Bigfile of size %d\n", file_size));
2006 error = tffs_fcreate(bigfile,
2007 (char *) tdata[TDATA_HUGE], file_size);
2008 expect(error, EFFS_OK);
2009
2010 error = test_expect_file((char *) bigfile,
2011 (char *) tdata[TDATA_HUGE], file_size);
2012 if (error) return 1;
2013
2014 test_ffs_state_get(&new);
2015 expect_gt(old.bytes_free, new.bytes_free - file_size);
2016
2017 tffs_remove(bigfile);
2018
2019 return 0;
2020 }
2021
2022
2023 /******************************************************************************
2024 * Reclaim Test Cases
2025 ******************************************************************************/
2026
2027 // Test that ffs internal state is same after a re-ffs_init()
2028 int case_reinit(int p0, int p1)
2029 {
2030 struct ffs_state_s old, new;
2031
2032 test_ffs_state_get(&old);
2033 error = tffs_initialize();
2034 expect(error, EFFS_OK);
2035 test_ffs_state_get(&new);
2036 error = test_expect_state(&old, &new);
2037 if (error) return 1;
2038
2039 return 0;
2040 }
2041
2042 // Test inodes reclaim
2043 int case_irec(int p0, int p1)
2044 {
2045 // African states: kenya, egypt, marocco, namibia, nigeria, mozambique,
2046 // tanzania, ghana, togo, liberia, mali, congo
2047 struct ffs_state_s old, new;
2048 int i, j, updates;
2049
2050 if (p0 == 0)
2051 p0 = 7;
2052
2053 tffs_mkdir("/africa");
2054
2055 // Initialize
2056 tffs_fwrite("/africa/kenya", TDATA(0)); // 1
2057 tffs_mkdir("/africa/east"); // 2
2058 tffs_fwrite("/africa/tanzania", TDATA(1)); // 3
2059 tffs_fwrite("/africa/east/egypt", TDATA(2)); // 4
2060 tffs_symlink("/africa/et", "africa/east/egypt"); // 5
2061 tffs_mkdir("/africa/west"); // 7
2062 tffs_fupdate("/africa/et", TDATA(3)); // 6
2063 tffs_fwrite("/africa/west/liberia", TDATA(0)); // 8
2064
2065 for (j = 0; j < p0; j++) {
2066 test_ffs_state_get(&old);
2067 updates = param.inodes_max - old.inodes_used;
2068 ttw(ttr(TTrTest, "loop %d: updates = %d" NL, j, updates));
2069 tw(tr(TR_FUNC, TrTestHigh, "loop %d: updates = %d\n", j, updates));
2070 for (i = 0; i < (updates / 10) + 10; i++) {
2071 // cleanup...
2072 error = tffs_remove("/africa/east/egypt");
2073 expect(error, EFFS_OK);
2074 error = tffs_remove("/africa/east");
2075 expect(error, EFFS_OK);
2076 error = tffs_remove("/africa/et");
2077 expect(error, EFFS_OK);
2078 error = tffs_remove("/africa/west/liberia");
2079 expect(error, EFFS_OK);
2080 error = tffs_remove("/africa/west");
2081 expect(error, EFFS_OK);
2082 // use 10 inodes...
2083 error = tffs_fwrite("/africa/kenya", TDATA(0)); // 1
2084 expect(error, EFFS_OK);
2085 error = tffs_mkdir("/africa/east"); // 2
2086 expect(error, EFFS_OK);
2087 error = tffs_fwrite("/africa/tanzania", (char *) tdata[0], 0); // 3
2088 expect(error, EFFS_OK);
2089 error = tffs_fcreate("/africa/east/egypt", TDATA(2)); // 4
2090 expect(error, EFFS_OK);
2091 error = tffs_symlink("/africa/et", "/africa/east/egypt"); // 5
2092 expect(error, EFFS_OK);
2093 error = tffs_mkdir("/africa/west"); // 7
2094 expect(error, EFFS_OK);
2095 error = tffs_fupdate("/africa/et", TDATA(3)); // 6
2096 expect(error, EFFS_OK);
2097 error = tffs_fcreate("/africa/west/liberia", TDATA(0)); // 8
2098 expect(error, EFFS_OK);
2099 error = tffs_fupdate("/africa/west/liberia", TDATA(1)); // 9
2100 expect(error, EFFS_OK);
2101 error = tffs_fupdate("/africa/tanzania", TDATA(2)); // 10
2102 expect(error, EFFS_OK);
2103 }
2104 test_ffs_state_get(&new);
2105 expect_eq(old.objects_total, new.objects_total);
2106 error = test_expect_file("/africa/tanzania", TDATA(2));
2107 if (error) return 1;
2108 error = test_expect_file("/africa/west/liberia", TDATA(1));
2109 if (error) return 1;
2110 error = test_expect_file("/africa/east/egypt", TDATA(3));
2111 if (error) return 1;
2112 error = test_expect_file("/africa/kenya", TDATA(0));
2113 if (error) return 1;
2114 }
2115
2116 return 0;
2117 }
2118
2119 #if 1
2120 // FIXME: Is this test case still valid after the new ffs_file_write() which
2121 // makes several chunks instead of on big?
2122
2123 // Test data reclaim. Use up to maximum half the total space available.
2124 // drec params: percentage of avail to use, numupdates. We must not have too
2125 // big files due to fragmentation problems
2126 #define DREC_DIR "/north-america/usa"
2127 #define DREC_CHUNKS 6
2128 int case_drec(int p0, int p1)
2129 {
2130 static struct test_file_s files[] =
2131 {
2132 { DREC_DIR "/alaska", 0, 0 },
2133 { DREC_DIR "/arkansas", 0, 0 },
2134 { DREC_DIR "/california", 0, 0 },
2135 { DREC_DIR "/colorado", 0, 0 },
2136 { DREC_DIR "/dakota", 0, 0 }, // north, south?
2137 { DREC_DIR "/DistrictOfColumbia", 0, 0 }, // state?
2138 { DREC_DIR "/florida", 0, 0 },
2139 { DREC_DIR "/Georgia", 0, 0 },
2140 { DREC_DIR "/hawaii", 0, 0 },
2141 { DREC_DIR "/Idaho", 0, 0 },
2142 { DREC_DIR "/Illinois", 0, 0 },
2143 { DREC_DIR "/Iowa", 0, 0 },
2144 { DREC_DIR "/kentucky", 0, 0 },
2145 { DREC_DIR "/maine", 0, 0 },
2146 { DREC_DIR "/Massachusettes", 0, 0 }, // spelling?
2147 { DREC_DIR "/michigan", 0, 0 },
2148 { DREC_DIR "/minnesota", 0, 0 },
2149 { DREC_DIR "/mississippi", 0, 0 },
2150 { DREC_DIR "/missouri", 0, 0 },
2151 { DREC_DIR "/Montana", 0, 0 },
2152 { DREC_DIR "/Nevada", 0, 0 },
2153 { DREC_DIR "/NewHampshire", 0, 0 },
2154 { DREC_DIR "/NewJersey", 0, 0 },
2155 { DREC_DIR "/NewMexico", 0, 0 },
2156 { DREC_DIR "/NewYork", 0, 0 }, // state?
2157 { DREC_DIR "/north-carolina", 0, 0 },
2158 { DREC_DIR "/ohio", 0, 0 },
2159 { DREC_DIR "/oklahoma", 0, 0 },
2160 { DREC_DIR "/Oregon", 0, 0 },
2161 { DREC_DIR "/Pensylvania", 0, 0 },
2162 { DREC_DIR "/RhodeIsland", 0, 0 }, // state?
2163 { DREC_DIR "/south-carolina", 0, 0 },
2164 { DREC_DIR "/Tennesee", 0, 0 },
2165 { DREC_DIR "/texas", 0, 0 },
2166 { DREC_DIR "/utah", 0, 0 },
2167 { DREC_DIR "/Vermont", 0, 0 },
2168 { DREC_DIR "/virginia", 0, 0 },
2169 { DREC_DIR "/washington", 0, 0 },
2170 { DREC_DIR "/Wyoming", 0, 0 },
2171
2172 { DREC_DIR "/40", 0, 0 },
2173 { DREC_DIR "/41", 0, 0 },
2174 { DREC_DIR "/42", 0, 0 },
2175 { DREC_DIR "/43", 0, 0 },
2176 { DREC_DIR "/44", 0, 0 },
2177 { DREC_DIR "/45", 0, 0 },
2178 { DREC_DIR "/46", 0, 0 },
2179 { DREC_DIR "/47", 0, 0 },
2180 { DREC_DIR "/48", 0, 0 },
2181 { DREC_DIR "/49", 0, 0 },
2182 { DREC_DIR "/50", 0, 0 },
2183 { DREC_DIR "/51", 0, 0 },
2184 { DREC_DIR "/52", 0, 0 },
2185 };
2186 struct chunk_s {
2187 int ratio;
2188 int high;
2189 int low;
2190 };
2191 const struct chunk_s chunk[DREC_CHUNKS] = {
2192 { 15, 65536, 32768 },
2193 { 15, 32768, 16384 },
2194 { 25, 16384, 8192 },
2195 { 20, 8192, 4096 },
2196 { 15, 4096, 2048 },
2197 { 10, 2048, 0 }
2198 };
2199
2200 int i, j, n, num_files;
2201 int num, pct;
2202 int size_used, size_max, size_chunk;
2203 int size_chunk_used, size_chunk_low, size_chunk_high;
2204
2205 num = (p0 == 0 ? 11 : p0);
2206 pct = (p1 == 0 ? 50 : p1);
2207
2208 tffs_mkdir("/north-america");
2209 tffs_mkdir("/north-america/usa");
2210
2211 size_max = pct * param.bytes_avail / 100;
2212 size_used = 0;
2213 num_files = sizeof(files) / sizeof(struct test_file_s);
2214 n = 0;
2215
2216 tw(tr(TR_FUNC, TrTestHigh, "pct = %d%% = %dk\n", pct, size_max/1024));
2217
2218 for (i = 0; i < DREC_CHUNKS; i++) {
2219 size_chunk = chunk[i].ratio * size_max / 100;
2220 size_chunk_low = chunk[i].low / 256 * fs.filesize_max / 256;
2221 size_chunk_high = chunk[i].high / 256 * fs.filesize_max / 256;
2222 tw(tr(TR_FUNC, TrTestHigh, "%4dk of [%d..%d]: ",
2223 size_chunk/1024, size_chunk_high, size_chunk_low));
2224 ttw(ttr(TTrTest, "%4dk of [%d..%d]: ",
2225 size_chunk/1024, size_chunk_high, size_chunk_low));
2226 size_chunk_used = 0;
2227 // If the total chunk size can guaranteeable be contained...
2228 if (size_chunk >= chunk[i].high) {
2229 for (j = 0; j < size_chunk / size_chunk_high; j++) {
2230 if (n >= num_files) {
2231 tw(tr(TR_FUNC, TrTestHigh, "j too big\n"));
2232 ttw(str(TTrTest, "j too big" NL));
2233 break;
2234 }
2235 files[n].size = size_chunk_low +
2236 rand() % (size_chunk_high - size_chunk_low);
2237 #ifdef WIN32
2238 /* due to limitation in MS Visual C */
2239 if( files[n].size > 65535 )
2240 files[n].size = 65535;
2241 #endif //WIN32
2242 files[n].data = (char *) tdata[TDATA_HUGE] + n;
2243 tw(tr(TR_NULL, TrTestHigh, "%d:%s ",
2244 files[n].size, &files[n].name[strlen(DREC_DIR)+1]));
2245 ttw(ttr(TTrTest, "%d:%s " NL,
2246 files[n].size, &files[n].name[strlen(DREC_DIR)+1]));
2247 size_chunk_used += files[n].size;
2248 n++;
2249 // FIXME: We should let all the unused bytes from this
2250 // chunk spill over into the next chunk so we use the
2251 // full percentage specified by p1.
2252 }
2253 size_used += size_chunk_used;
2254 }
2255 tw(tr(TR_NULL, TrTestHigh, "(%dk)\n", size_chunk_used/1024));
2256 ttw(ttr(TTrTest, "(%dk)" NL, size_chunk_used/1024));
2257 }
2258 tw(tr(TR_FUNC, TrTestHigh, "pct = %d%% = %dk\n",
2259 100 * size_used/param.bytes_avail, size_used/1024));
2260 ttw(ttr(TTrTest, "pct = %d%% = %dk" NL,
2261 100 * size_used/param.bytes_avail, size_used/1024));
2262
2263 for (j = 0; j < num; j++) {
2264 ttw(ttr(TTrTest, "drec: %d. write" NL, j+1));
2265 tw(tr(TR_FUNC, TrTestHigh, "drec: %d. write\n", j+1));
2266 for (i = 0; i < n; i++) {
2267 error = tffs_fwrite(files[i].name, files[i].data, files[i].size);
2268 expect(error, EFFS_OK);
2269 }
2270 }
2271
2272 for (i = 0; i < n; i++) {
2273 error = test_expect_file(files[i].name, files[i].data, files[i].size);
2274 if (error)
2275 return 1;
2276 error = tffs_remove(files[i].name);
2277 expect(error, EFFS_OK);
2278 }
2279 tffs_remove("/north-america/usa");
2280 tffs_remove("/north-america");
2281
2282 return 0;
2283 }
2284
2285 #else
2286
2287 // Test data reclaim. Use up to maximum half the total space available.
2288 // drec params: percentage of avail to use, numupdates. We must not have too
2289 // big files due to fragmentation problems
2290 int case_drec(int p0, int p1)
2291 {
2292 struct blabla_s {
2293 int ratio;
2294 int high;
2295 int low;
2296 };
2297
2298 struct blabla_s sizes[4] =
2299 {
2300 { 10, 65536, 32768 },
2301 { 40, 21800, 16384 },
2302 { 45, 16384, 2048 },
2303 { 5, 2048, 0 }
2304 };
2305
2306 static struct test_file_s files[] =
2307 {
2308 { DRECDIR "/alaska", 0, 0 },
2309 { DRECDIR "/arkansas", 0, 0 },
2310 { DRECDIR "/california", 0, 0 },
2311 { DRECDIR "/colorado", 0, 0 },
2312 { DRECDIR "/dakota", 0, 0 }, // north, south?
2313 { DRECDIR "/DistrictOfColumbia", 0, 0 }, // state?
2314 { DRECDIR "/florida", 0, 0 },
2315 { DRECDIR "/Georgia", 0, 0 },
2316 { DRECDIR "/hawaii", 0, 0 },
2317 { DRECDIR "/Idaho", 0, 0 },
2318 { DRECDIR "/Illinois", 0, 0 },
2319 { DRECDIR "/Iowa", 0, 0 },
2320 { DRECDIR "/kentucky", 0, 0 },
2321 { DRECDIR "/maine", 0, 0 },
2322 { DRECDIR "/Massachusettes", 0, 0 }, // spelling?
2323 { DRECDIR "/michigan", 0, 0 },
2324 { DRECDIR "/minnesota", 0, 0 },
2325 { DRECDIR "/mississippi", 0, 0 },
2326 { DRECDIR "/missouri", 0, 0 },
2327 { DRECDIR "/Montana", 0, 0 },
2328 { DRECDIR "/Nevada", 0, 0 },
2329 { DRECDIR "/NewHampshire", 0, 0 },
2330 { DRECDIR "/NewJersey", 0, 0 },
2331 { DRECDIR "/NewMexico", 0, 0 },
2332 { DRECDIR "/NewYork", 0, 0 }, // state?
2333 { DRECDIR "/north-carolina", 0, 0 },
2334 { DRECDIR "/ohio", 0, 0 },
2335 { DRECDIR "/oklahoma", 0, 0 },
2336 { DRECDIR "/Oregon", 0, 0 },
2337 { DRECDIR "/Pensylvania", 0, 0 },
2338 { DRECDIR "/RhodeIsland", 0, 0 }, // state?
2339 { DRECDIR "/south-carolina", 0, 0 },
2340 { DRECDIR "/Tennesee", 0, 0 },
2341 { DRECDIR "/texas", 0, 0 },
2342 { DRECDIR "/utah", 0, 0 },
2343 { DRECDIR "/Vermont", 0, 0 },
2344 { DRECDIR "/virginia", 0, 0 },
2345 { DRECDIR "/washington", 0, 0 },
2346 { DRECDIR "/Wyoming", 0, 0 },
2347
2348 { DRECDIR "/40", 0, 0 },
2349 { DRECDIR "/41", 0, 0 },
2350 { DRECDIR "/42", 0, 0 },
2351 { DRECDIR "/43", 0, 0 },
2352 { DRECDIR "/44", 0, 0 },
2353 { DRECDIR "/45", 0, 0 },
2354 { DRECDIR "/46", 0, 0 },
2355 { DRECDIR "/47", 0, 0 },
2356 { DRECDIR "/48", 0, 0 },
2357 { DRECDIR "/49", 0, 0 },
2358 { DRECDIR "/50", 0, 0 },
2359 { DRECDIR "/51", 0, 0 },
2360 { DRECDIR "/52", 0, 0 },
2361 };
2362
2363 int i, j, n, size, size_target;
2364 int num, percentage;
2365
2366 error = tffs_mkdir("/north-america");
2367
2368 percentage = (p1 == 0 ? 50 : p1);
2369 n = sizeof(files)/sizeof(struct test_file_s);
2370 size_target = percentage * param.bytes_avail / 100;
2371 size = 0;
2372 for (i = 0; i < 4; i++) {
2373 sizes[i].ratio = sizes[i].ratio * size_target / 100;
2374 sizes[i].high = sizes[i].high / 256 * fs.filesize_max / 256;
2375 sizes[i].low = sizes[i].low / 256 * fs.filesize_max / 256;
2376 size += sizes[i].ratio;
2377 }
2378 ttw(ttr(TTrTest, "%4dk in total" NL, size / 1024));
2379 tw(tr(TR_FUNC, TrTestHigh, "%4dk in total\n", size / 1024));
2380
2381 j = 0;
2382 size = size_target = 0;
2383 for (i = 0; i < 4; i++) {
2384 ttw(ttr(TTrTest, "%4dk: %d..%d = ",
2385 sizes[i].ratio / 1024, sizes[i].high, sizes[i].low));
2386 tw(tr(TR_FUNC, TrTestHigh, "%4dk: %d..%d = ",
2387 sizes[i].ratio / 1024, sizes[i].high, sizes[i].low));
2388 size_target += sizes[i].ratio;
2389 while (size < size_target) {
2390 files[j].data = (char *) tdata[TDATA_HUGE] + j;
2391 files[j].size = rand() % (sizes[i].high - sizes[i].low)
2392 + sizes[i].low;
2393 size += files[j].size;
2394 ttw(ttr(TTrTest, "%d:%s, ",
2395 files[j].size, &files[j].name[strlen(DRECDIR)+1]));
2396 tw(tr(TR_NULL, TrTestHigh, "%d:%s, ",
2397 files[j].size, &files[j].name[strlen(DRECDIR)+1]));
2398 j++;
2399 if (j >= n) {
2400 // TODO: better error handling?
2401 ttw(str(TTrTest, "j too big" NL));
2402 tw(tr(TR_FUNC, TrTestHigh, "j too big\n"));
2403 return -1;
2404 }
2405 }
2406 ttw(str(TTrTest, "" NL));
2407 tw(tr(TR_NULL, TrTestHigh, "\n"));
2408 }
2409 n = j;
2410
2411 error = tffs_mkdir(DRECDIR);
2412
2413 num = (p0 == 0 ? 5 : p0);
2414 for (j = 0; j < num; j++) {
2415 ttw(ttr(TTrTest, "drec: %d. write" NL, j+1));
2416 tw(tr(TR_FUNC, TrTestHigh, "drec: %d. write\n", j+1));
2417 for (i = 0; i < n; i++) {
2418 error = tffs_fwrite(files[i].name, files[i].data, files[i].size);
2419 expect(error, EFFS_OK);
2420 }
2421 // error = test_run("ri");
2422 // if (error) return 1;
2423 }
2424
2425 for (i = 0; i < n; i++) {
2426 error = test_expect_file(files[i].name, files[i].data, files[i].size);
2427 if (error) return 1;
2428 error = tffs_remove(files[i].name);
2429 expect(error, EFFS_OK);
2430 }
2431
2432 return 0;
2433 }
2434
2435 #endif
2436
2437 void save_block_ages(int *age)
2438 {
2439 int i;
2440 struct block_header_s *bhp;
2441
2442 for (i = 0; i < dev.numblocks; i++)
2443 {
2444 bhp = (struct block_header_s *) offset2addr(dev.binfo[i].offset);
2445 age[i] = bhp->age;
2446 tw(tr(TR_FUNC, TrTestHigh, "b,age(%d,%d)\n", i, age[i]));
2447 }
2448 }
2449
2450 // NOTE: This chech includes the inode block thus the inode block also needs
2451 // to be reclaimed before this functions returns ok
2452 int has_all_block_ages_changed(int *age)
2453 {
2454 int i;
2455 struct block_header_s *bhp;
2456
2457 for (i = 0; i < dev.numblocks; i++)
2458 {
2459 bhp = (struct block_header_s *) offset2addr(dev.binfo[i].offset);
2460 if (age[i] == bhp->age)
2461 return 0; // Age have not changed
2462 }
2463
2464 return 1;
2465 }
2466
2467 #define chunkalign(size) (((size) + fs.chunk_size_max-1) & ~(fs.chunk_size_max-1))
2468
2469 // Fill a data blocks with the maximum possible numbers of objects
2470 int fill_block_with_max_objects(void)
2471 {
2472 static int n = 1000; // Avoid white spaces in the filename
2473 int fsize, i, b, nobjects, bfree_space;
2474 char myname[] = "Fb_max-xxxx";
2475 char *mydata = (char *) tdata[TDATA_HUGE];
2476
2477 // Find an approximate objects size
2478 fsize = dev.blocksize / fs.block_files_max - sizeof(myname) - 1;
2479
2480 // Fill approximate 75% of a data block
2481 for (i = 0; i < fs.block_files_max / 4 * 3; i++, n++) {
2482 sprintf(myname, "/Fb_max-%4d", n);
2483 error = tffs_file_write(myname,(char*) mydata, fsize, FFS_O_CREATE);
2484 expect_ok(error);
2485 }
2486
2487 // Find the block we currently are filling with objects.
2488 error = tffs_xlstat(myname, &xstat);
2489 expect_ok(error);
2490
2491 b = xstat.block;
2492 bfree_space = dev.blocksize - bstat[b].used;
2493 tw(tr(TR_FUNC, TrTestHigh, "Block: %d, Free space:%dkB\n", b ,bfree_space>>10));
2494
2495 // Calculate the exact file size to fill a block with max number of files
2496 nobjects = fs.block_files_max - bstat[b].objects - fs.block_files_reserved;
2497
2498 fsize = bfree_space / nobjects - sizeof(myname);
2499 fsize &= ~dev.atomnotmask; // Avoid padding
2500 tw(tr(TR_FUNC, TrTestHigh, "Exact file size (%d)\n", fsize));
2501
2502 // Fill the rest of the block
2503 for (i = 0; i < nobjects; i++, n++) {
2504 sprintf(myname, "/Fb_max-%4d", n);
2505 error = tffs_file_write(myname, (char*) mydata, fsize, FFS_O_CREATE);
2506 expect_ok(error);
2507 }
2508
2509 tw(tr(TR_FUNC, TrTest, "Filled block %d done\n", b));
2510
2511 return EFFS_OK;
2512 }
2513
2514 // Stress test data reclaim and test wearleveling.
2515 // NOTE: This test can not run in less than 7 blocks
2516 int case_adrec(int nwrites, int nage_change)
2517 {
2518 int i, j, n, inodes_reserved, nobjects, bytes_max;
2519 int blocks, fsize;
2520 int dynamic_fsize = 1000;
2521 char static_name[] = "/Static-xxxx";
2522 char dynamic_name[] = "/Dynamic";
2523 int *age = (int*) smallbuf;
2524
2525 /* Flow:
2526 Format ffs.
2527 Fill N number of blocks with the maximum possible numbers of objects.
2528 Fill the rest of FFS except from a few kbytes. Use almost all the inodes.
2529 Trigger data reclaim by continue to update a file.
2530 Test if all flash blocks are used for wearleveling.
2531 */
2532 if (nwrites == 0) // Number of updates of the dynamic file (timeout)
2533 nwrites = 1000000;
2534 if (nage_change == 0) //
2535 nage_change = 2;
2536
2537 error = tffs_preformat(0xDEAD);
2538 expect(error, EFFS_OK);
2539 error = tffs_format("/ffs", 0x2BAD);
2540 expect(error, EFFS_OK);
2541
2542 // Major hack, disable journal and reserved space or else there will be
2543 // created new journals inside the blocks where we wish to have only
2544 // static objects
2545 fs.ijournal = 0;
2546 fs.reserved_space = 0;
2547
2548 // Minimum amount of inodes to fill FFS:
2549 ffs_query(Q_BYTES_FREE, &bytes_max);
2550 inodes_reserved = bytes_max / fs.chunk_size_max;
2551
2552 // We need to calculate the amount of inodes we can use and still
2553 // garanti that we have enough inodes to fill all data blocks with data.
2554 nobjects = fs.objects_max - inodes_reserved;
2555
2556 // How many blocks can we fill?
2557 blocks = nobjects / fs.block_files_max;
2558
2559 // Fill N number of data blocks with the maximum possible numbers of objects
2560 tw(tr(TR_FUNC, TrTest, "Fill %d blocks with max obj \n", blocks));
2561 n = 1000; // Avoid white spaces in the filename
2562 for (j = 0; j < blocks; j++) {
2563 error = fill_block_with_max_objects();
2564 expect_ok(error);
2565 }
2566
2567 // Restore journal and reserved space
2568 error = tffs_initialize();
2569 expect_ok(error);
2570
2571 // Fill the rest of FFS except from 3 times the dynamic size, futhermore
2572 // keep 4 objects 'free'
2573 ffs_query(Q_OBJECTS_FREE, &nobjects);
2574 tw(tr(TR_FUNC, TrTest, "Fill almost the rest of FFS\n"));
2575 n = 1000; // Avoid white spaces in the filename
2576 while (nobjects - 4 > 0) {
2577 ffs_query(Q_BYTES_FREE, &bytes_max);
2578 fsize = (bytes_max - dynamic_fsize * 3) / (nobjects - 4);
2579
2580 tw(tr(TR_FUNC, TrTestHigh, "space,obj,fsize (%dk,%d,%d)\n",
2581 bytes_max/1024, nobjects, fsize));
2582
2583 if (fsize > fs.chunk_size_max) {
2584 // Roundup to the closet complete chunk or else we use to many inodes
2585 fsize = chunkalign(fsize);
2586 tw(tr(TR_FUNC, TrTestHigh, "Chunkalign %d\n", fsize));
2587 }
2588 sprintf(static_name, "/Static-%4d", n++);
2589 error = tffs_file_write(static_name, (char*)tdata[TDATA_HUGE], fsize,
2590 FFS_O_CREATE);
2591 if (error == EFFS_NOSPACE) {
2592 tw(tr(TR_FUNC, TrTest, "WARNING: Fill failed with out of data space\n"));
2593 break;
2594 }
2595 else
2596 expect_ok(error);
2597
2598 ffs_query(Q_OBJECTS_FREE, &nobjects);
2599 }
2600 tw(tr(TR_FUNC, TrTest, "Fill done, all static objects are written\n"));
2601
2602 case_debug_help(2,0); // bstat trace
2603
2604 // Save the age of all blocks
2605 save_block_ages(age);
2606
2607 tw(tr(TR_FUNC, TrTest, "Update the dynamic file max %d times\n", nwrites));
2608
2609 for (i = 0, j = 0; i < nwrites; i++) {
2610 error = tffs_file_write(dynamic_name, (char*)tdata[TDATA_HUGE],
2611 dynamic_fsize,
2612 FFS_O_CREATE | FFS_O_TRUNC);
2613 expect_ok(error);
2614
2615 if (i%1000 == 0)
2616 tw(tr(TR_FUNC, TrTest, "Write number %d\n", i));
2617
2618 if (has_all_block_ages_changed(age) == 1) {
2619 tw(tr(TR_FUNC, TrTest,"All block ages have changed %d\n", j));
2620 j++;
2621 save_block_ages(age);
2622 }
2623
2624 if (j >= nage_change)
2625 break;
2626 }
2627
2628 test_statistics_print();
2629
2630 case_debug_help(2,0); // bstat trace
2631
2632 if (j >= nage_change)
2633 return 0;
2634
2635 return 1; //Error
2636 }
2637
2638 /******************************************************************************
2639 * Journalling and Recover Test Cases
2640 ******************************************************************************/
2641
2642 // Test journalling. Both for normal objects but also for journal file
2643 // itself! The following operations are tested for proper recovery: fcreate,
2644 // fupdate, relocate, delete, clean, fcontrol
2645 int case_journal(int p0, int p1)
2646 {
2647 struct test_file_s tf;
2648 struct ffs_state_s old, new;
2649 struct xstat_s stat_old, stat_new;
2650 int i, myspace = 0, mytmpspace, offset;
2651 char mytmpname[] = "/test-journal-tmp";
2652 int testflags[] = {
2653 JOURNAL_TEST_EMPTY, // object not committed
2654 JOURNAL_TEST_WRITING, // object not committed
2655 JOURNAL_TEST_READY, // object ok
2656 JOURNAL_TEST_COMMITTING, // object ok
2657 JOURNAL_TEST_COMMITTED, // object ok
2658 JOURNAL_TEST_DONE, // object ok
2659 };
2660
2661 tf.name = "/test-journal";
2662 tf.data = (char *) tdata[2];
2663 tf.size = sdata[2];
2664
2665 #if 1
2666 // NOTEME: Workaround, the below use of state_get can not handle if we
2667 // create a new journal while we run the test. Instead we create a
2668 // new journal before the test if we are close to full.
2669 if (fs.journal_pos >= fs.journal_size - (FFS_JOURNAL_MARGIN + 12) *
2670 sizeof(struct journal_s)) {
2671 tw(tr(TR_FUNC, TrTest, "Journal file (near) full!\n"));
2672 journal_create(fs.ijournal);
2673 }
2674
2675 // NOTEME: Workaround, this test is unabel to handle a
2676 // block_alloc(). The 'extra' amount of lost space caused by a
2677 // block_alloc() and its block header dos cheat the test to fail.
2678 offset = data_prealloc(1024);
2679 expect_gt(offset, 0);
2680 #endif
2681
2682 // init
2683 #if 1
2684 error = tffs_fwrite(mytmpname, tf.data, tf.size);
2685 expect(error, EFFS_OK);
2686 error = tffs_xlstat(mytmpname, &stat_old);
2687 expect(error, EFFS_OK);
2688 mytmpspace = stat_old.space;
2689 ttw(ttr(TTrTest, "file '%s' uses %d bytes" NL, mytmpname, mytmpspace));
2690 tw(tr(TR_FUNC, TrTestHigh, "file '%s' uses %d bytes\n", mytmpname, mytmpspace));
2691 #endif
2692 error = tffs_fcreate(tf.name, tf.data, tf.size);
2693 expect(error, EFFS_OK);
2694 error = tffs_xlstat(tf.name, &stat_old);
2695 expect(error, EFFS_OK);
2696 myspace = stat_old.space;
2697 error = tffs_fcreate("/dummy", 0, 0);
2698 expect(error, EFFS_OK);
2699 ttw(ttr(TTrTest, "file '%s' uses %d bytes" NL, tf.name, myspace));
2700 tw(tr(TR_FUNC, TrTestHigh, "file '%s' uses %d bytes\n", tf.name, myspace));
2701
2702 ttw(str(TTrTest, "fcreate:" NL));
2703 tw(tr(TR_BEGIN, TrTestHigh, "fcreate:\n"));
2704 stat_old.inode = 0;
2705 for (i = 0; i < sizeof(testflags)/sizeof(int); i++)
2706 {
2707 // clean up from operation being tested
2708 tffs_remove(tf.name);
2709
2710 ttw(ttr(TTrTest, "fs.testflags = 0x%x" NL, testflags[i]));
2711 tw(tr(TR_FUNC, TrTestHigh, "fs.testflags = 0x%x\n", testflags[i]));
2712 test_ffs_state_get(&old);
2713
2714 error = tffs_fcontrol("/dummy", OC_FS_TESTFLAGS, testflags[i]);
2715 expect(error, EFFS_OK);
2716
2717 error = tffs_fcreate(tf.name, tf.data, tf.size);
2718 expect(error, EFFS_OK);
2719
2720 error = tffs_initialize();
2721 expect(error, EFFS_OK);
2722
2723 test_ffs_state_get(&new);
2724
2725 tw(tr(TR_FUNC, TrTestHigh,
2726 "bytes_used new = %7d, old = %7d, diff = %d\n",
2727 new.bytes_used, old.bytes_used, new.bytes_used - old.bytes_used));
2728 tw(tr(TR_FUNC, TrTestHigh,
2729 "bytes_lost new = %7d, old = %7d, diff = %d\n",
2730 new.bytes_lost, old.bytes_lost,
2731 new.bytes_lost - old.bytes_lost));
2732
2733 expect_eq(new.bytes_used, old.bytes_used + myspace);
2734
2735 error = tffs_xlstat(tf.name, &stat_new);
2736 if (i < 2) {
2737 // object was not committed, so object must not exist
2738 expect(error, EFFS_NOTFOUND);
2739 expect_eq(new.objects_total, old.objects_total);
2740 expect_eq(new.bytes_lost, old.bytes_lost + myspace);
2741 }
2742 else {
2743 // object was committed, so object must exist
2744 expect(error, EFFS_OK);
2745 expect_ne(stat_old.inode, stat_new.inode);
2746 expect_eq(new.objects_total, old.objects_total + 1);
2747 expect_eq(new.bytes_lost, old.bytes_lost);
2748 }
2749 }
2750 ttw(str(TTrTest, "" NL));
2751 tw(tr(TR_END, TrTestHigh, ""));
2752
2753 ttw(str(TTrTest, "fupdate:" NL));
2754 tw(tr(TR_BEGIN, TrTestHigh, "fupdate:\n"));
2755 error = tffs_xlstat(tf.name, &stat_old);
2756 expect(error, EFFS_OK);
2757 for (i = 0; i < sizeof(testflags)/sizeof(int); i++)
2758 {
2759 ttw(ttr(TTrTest, "fs.testflags = 0x%x" NL, testflags[i]));
2760 tw(tr(TR_FUNC, TrTestHigh, "fs.testflags = 0x%x\n", testflags[i]));
2761 test_ffs_state_get(&old);
2762
2763 error = tffs_fcontrol("/dummy", OC_FS_TESTFLAGS, testflags[i]);
2764 expect(error, EFFS_OK);
2765
2766 error = tffs_fupdate(tf.name, tf.data, tf.size);
2767
2768 error = tffs_initialize();
2769 expect(error, EFFS_OK);
2770
2771 error = tffs_xlstat(tf.name, &stat_new);
2772 expect(error, EFFS_OK);
2773
2774 if (i < 2) {
2775 // object was not committed, so stats must be equal
2776 error = test_expect_data(&stat_new, &stat_old, sizeof(stat));
2777 if (error) return 1;
2778 }
2779 else {
2780 // object was committed, so inodes must not be equal
2781 expect_ne(stat_old.inode, stat_new.inode);
2782 }
2783
2784 test_ffs_state_get(&new);
2785
2786 tw(tr(TR_FUNC, TrTestHigh,
2787 "bytes_used new = %7d, old = %7d, diff = %d\n",
2788 new.bytes_used, old.bytes_used, new.bytes_used - old.bytes_used));
2789 tw(tr(TR_FUNC, TrTestHigh,
2790 "bytes_lost new = %7d, old = %7d, diff = %d\n",
2791 new.bytes_lost, old.bytes_lost,
2792 new.bytes_lost - old.bytes_lost));
2793
2794 expect_eq(new.objects_total, old.objects_total);
2795 expect_eq(new.bytes_used, old.bytes_used + myspace);
2796 expect_eq(new.bytes_lost, old.bytes_lost + myspace);
2797 }
2798 ttw(str(TTrTest, "" NL));
2799 tw(tr(TR_END, TrTestHigh, ""));
2800
2801 ttw(str(TTrTest, "rename:" NL));
2802 tw(tr(TR_BEGIN, TrTestHigh, "rename:\n"));
2803 error = tffs_fwrite(tf.name, tf.data, tf.size);
2804 expect(error, EFFS_OK);
2805 error = tffs_xlstat(tf.name, &stat_old);
2806 expect(error, EFFS_OK);
2807
2808 for (i = 0; i < sizeof(testflags)/sizeof(int); i++)
2809 {
2810 ttw(ttr(TTrTest, "fs.testflags = 0x%x" NL, testflags[i]));
2811 tw(tr(TR_FUNC, TrTestHigh, "fs.testflags = 0x%x\n", testflags[i]));
2812
2813 error = tffs_fwrite(mytmpname, tf.data, tf.size);
2814 expect_ok(error);
2815 error = tffs_fcontrol("/dummy", OC_FS_TESTFLAGS, testflags[i]);
2816 expect(error, EFFS_OK);
2817
2818 test_ffs_state_get(&old);
2819 error = tffs_rename(mytmpname, tf.name);
2820 expect_ok(error);
2821 error = tffs_initialize();
2822 expect(error, EFFS_OK);
2823 test_ffs_state_get(&new);
2824
2825 // tf.name MUST always exist
2826 error = tffs_xlstat(tf.name, &stat_new);
2827 expect(error, EFFS_OK);
2828
2829 tw(tr(TR_FUNC, TrTestHigh,
2830 "bytes_used new = %7d, old = %7d, diff = %d\n",
2831 new.bytes_used, old.bytes_used, new.bytes_used - old.bytes_used));
2832 tw(tr(TR_FUNC, TrTestHigh,
2833 "bytes_lost new = %7d, old = %7d, diff = %d\n",
2834 new.bytes_lost, old.bytes_lost,
2835 new.bytes_lost - old.bytes_lost));
2836
2837 error = tffs_xlstat(mytmpname, &stat_new);
2838 if (i < 2) {
2839 // Journal not READY, so test-journal-tmp must exist
2840 expect(error, EFFS_OK);
2841 expect_eq(new.objects_total, old.objects_total);
2842 expect_eq(new.bytes_used, old.bytes_used + myspace);
2843 expect_eq(new.bytes_lost, old.bytes_lost + myspace);
2844 }
2845 else {
2846 // Journal ready, so test-journal-tmp must not exist
2847 expect(error, EFFS_NOTFOUND);
2848 expect_eq(new.objects_total, old.objects_total - 1);
2849 expect_eq(new.bytes_used, old.bytes_used + myspace);
2850 expect_eq(new.bytes_lost, old.bytes_lost + myspace + mytmpspace);
2851 expect_ne(stat_old.inode, stat_new.inode);
2852 }
2853 }
2854
2855 ttw(str(TTrTest, "" NL));
2856 tw(tr(TR_END, TrTestHigh, ""));
2857
2858 // cleanup
2859 tffs_remove(tf.name);
2860 tffs_remove("/dummy");
2861
2862 return 0;
2863 }
2864
2865 // Test block recovery
2866 int case_brecover(int p0, int p1)
2867 {
2868 struct test_file_s tf;
2869 struct ffs_state_s old, new;
2870 int j, inodes_block_old = 0, inodes_block_new = 0;
2871 int i, error;
2872 uint16 free_blocks;
2873
2874 int testflags[] = {
2875 BLOCK_COMMIT_BEFORE,
2876 BLOCK_COMMIT_NO_VALID,
2877 BLOCK_COMMIT_OLD_FREE,
2878 BLOCK_COMMIT_AFTER
2879 };
2880
2881 int testflags_2[] = {
2882 BLOCK_RECLAIM_ALLOC,
2883 BLOCK_RECLAIM_CLEANING,
2884 BLOCK_RECLAIM_NO_CLEAN,
2885 };
2886
2887 tf.name = "/dummy4fctrl";
2888 tf.data = (char *) tdata[0];
2889 tf.size = sdata[0];
2890
2891 error = tffs_fwrite(tf.name, tf.data, tf.size);
2892 expect(error, EFFS_OK);
2893
2894 for (j = 0; j < 3; j++)
2895 {
2896 test_ffs_state_get(&old);
2897 //case_lsr(0, 0); // temp
2898
2899 error = ffs_query(Q_FS_INODES, (uint16 *) &inodes_block_old);
2900 expect(error, EFFS_OK);
2901
2902 // Trigger block_commit() to fail
2903 error = tffs_fcontrol(tf.name, OC_FS_TESTFLAGS, testflags[j]);
2904 expect(error, EFFS_OK);
2905
2906 inodes_reclaim();
2907
2908 error = tffs_initialize();
2909 expect(error, EFFS_OK);
2910
2911 error = ffs_query(Q_FS_INODES, (uint16 *) &inodes_block_new);
2912 expect(error, EFFS_OK);
2913
2914 ttw(ttr(TTrTest, "inodes block %d -> %d" NL,
2915 inodes_block_old, inodes_block_new));
2916 tw(tr(TR_FUNC, TrTestHigh, "inodes block %d -> %d\n",
2917 inodes_block_old, inodes_block_new));
2918
2919 switch (j) {
2920 case 0: expect(inodes_block_old, inodes_block_new); break;
2921 case 1: expect_ne(inodes_block_old, inodes_block_new); break;
2922 case 2: expect_ne(inodes_block_old, inodes_block_new); break;
2923 case 3: expect_ne(inodes_block_old, inodes_block_new); break;
2924 }
2925
2926 test_ffs_state_get(&new);
2927
2928 if (test_expect_objects(&old, &new))
2929 return 1;
2930
2931 }
2932
2933 // FIXME: The below test often fails (recalaim data block and create new
2934 // jouranl file makes it to fail)
2935 if (p0 == 9) {
2936 // Test BLOCK_RECLAIM Fill up all data blocks until only one is
2937 // free.
2938 tw(tr(TR_FUNC, TrTest, "Test Block reclaim\n"));
2939 ttw(ttr(TTrTest, "Test Block reclaim..." ));
2940 tffs_mkdir("/rand");
2941
2942 ffs_query(Q_BLOCKS_FREE, (uint16 *) &free_blocks);
2943 while (free_blocks > 1) {
2944 error = case_mk_rand_file("/rand", 10000, 0);
2945 if (error < 0) {
2946 tw(tr(TR_FUNC, TrTest, "ERROR: case_mk_rand_file failed\n"));
2947 return 1;
2948 }
2949 ffs_query(Q_BLOCKS_FREE, (uint16 *) &free_blocks);
2950 }
2951
2952 // case_mk_rand_file() will trigger a block_reclaim which will be
2953 // interruptet by the activated testflags. This will cause the FFS to
2954 // use the last free block! After a new init of ffs must it have
2955 // recovered.
2956
2957 ttw(ttr(TTrTest, "use last block..."));
2958 for (i = 0; i < sizeof(testflags_2)/sizeof(int); i++)
2959 {
2960 error = tffs_fcontrol(tf.name, OC_FS_TESTFLAGS, testflags_2[i]);
2961 expect(error, EFFS_OK);
2962
2963 do {
2964 case_cleanup(20000);
2965 error = case_mk_rand_file("/rand", 7000, 0);
2966 } while (error >= 0);
2967 expect(error, EFFS_NOSPACE);
2968
2969 ffs_query(Q_BLOCKS_FREE, (uint16 *) &free_blocks);
2970 if (free_blocks != 0) {
2971 tw(tr(TR_FUNC, TrTestHigh, "ERROR: We still have free blocks!\n"));
2972 return 1;
2973 }
2974
2975 error = tffs_initialize();// NOTE: error "WARNING: block_alloc failed" is okay.
2976 expect_ok(error);
2977
2978 ffs_query(Q_BLOCKS_FREE, (uint16 *) &free_blocks);
2979 if (free_blocks != 1) {
2980 tw(tr(TR_FUNC, TrTestHigh,
2981 "ERROR: Wrong number of blocks free: %d\n", free_blocks));
2982 return 1;
2983 }
2984 }
2985 // NOTE: We have to make this clenaup because we can not run if FFS is
2986 // near full
2987 case_cleanup(dev.blocksize);
2988
2989 // This test will make data_block_reclaim() to fail before it has
2990 // relocated all objects
2991 ttw(ttr(TTrTest, "interrupt block_reclaim()..."));
2992 for (i = 0; i < 4; i++)
2993 {
2994 error = tffs_fcontrol(tf.name, OC_FS_TESTFLAGS, BLOCK_RECOVER_OBJECTS);
2995 expect(error, EFFS_OK);
2996
2997 do {
2998 case_cleanup(20000);
2999 error = case_mk_rand_file("/rand", 7000, 0);
3000 } while (error >= 0);
3001 expect(error, EFFS_NOSPACE);
3002 // NOTE: error "WARNING: block_alloc failed" is okay.
3003 error = tffs_initialize();
3004 expect_ok(error);
3005 }
3006
3007 // Free up some space to the next test case.
3008 case_cleanup(dev.blocksize * 4);
3009
3010 ttw(ttr(TTrTest, "Done" NL));
3011 }
3012
3013 return 0;
3014 }
3015
3016
3017 /******************************************************************************
3018 * Specific/Special Test Cases
3019 ******************************************************************************/
3020
3021 // FIXME: Is this test case still valid after the new ffs_file_write() which
3022 // makes seveal chunks instead of on big?
3023
3024 // Test filling a block to the full block size. We write N files of
3025 // decreasing powers of two, starting at half the block size. At the end we
3026 // are guaranteed to have at least one completelt full block. Also, we know
3027 // we have used at least <block_size> bytes of storage.
3028 int case_bfull(int p0, int p1)
3029 {
3030 struct ffs_state_s old, new;
3031 char myname[40] = "/antarctica/ice-";
3032 int i, size, mysize, dirlen, space;
3033
3034 if (param.data_blocks <= 1) {
3035 ttw(ttr(TTrTest, "WARNING: Too few blocks to run. Skip test" NL));
3036 tw(tr(TR_FUNC, TrTest, "WARNING: Too few blocks to run. Skip test\n"));
3037 return 0;
3038 }
3039
3040 error = tffs_mkdir("/antarctica");
3041
3042 error = case_cleanup(param.block_size + 10000);
3043 expect_ok(error);
3044
3045 test_ffs_state_get(&old);
3046 dirlen = strlen(myname);
3047 space = 0;
3048
3049 size = param.block_size/2;
3050 for (i = 0; size > 0; i++) {
3051 mysize = size - param.atomsize;
3052 if (mysize < 0)
3053 mysize = size;
3054 sprintf(&myname[dirlen], "%d", size);
3055 error = tffs_fwrite(myname, (char *) tdata[TDATA_HUGE], mysize);
3056 expect(error, EFFS_OK);
3057 error = tffs_xlstat(myname, &xstat);
3058 expect(error, EFFS_OK);
3059 ttw(ttr(TTrTest, "%6d: %s" NL, xstat.space, myname));
3060 tw(tr(TR_FUNC, TrTestHigh, "%6d: %s\n", xstat.space, myname));
3061 space += xstat.space;
3062 size >>= 1;
3063 }
3064
3065 test_ffs_state_get(&new);
3066 // Check space used. We have used a little more than <space> because we
3067 // have probably allocated one or two new blocks, and that is
3068 // dev.atomsize more bytes per block.
3069 tw(tr(TR_FUNC, TrTestHigh,
3070 "old.free = %d, new.free = %d, space used in test = %d\n",
3071 old.bytes_free, new.bytes_free, space));
3072 expect_gt(old.bytes_free, new.bytes_free + space - 1);
3073
3074 // remove half of the files
3075 size = param.block_size/2;
3076 for (i = 0; size > 0; i++) {
3077 if (i & 1) {
3078 sprintf(&myname[dirlen], "%d", size);
3079 error = tffs_remove(myname);
3080 expect(error, EFFS_OK);
3081 }
3082 size >>= 1;
3083 }
3084
3085 return 0;
3086 }
3087
3088
3089 int case_list(int p0, int p1)
3090 {
3091 const struct testcase_s *p;
3092
3093 tw(tr(TR_FUNC, TrTestHigh, "bigbuf = 0x%X, %d\n", bigbuf, bigbuf_size));
3094 tw(tr(TR_FUNC, TrTestHigh, "smallbuf = 0x%X, %d\n", smallbuf, smallbuf_size));
3095
3096 ttw(ttr(TTrTest, "bigbuf = 0x%X, %d" NL, bigbuf, bigbuf_size));
3097 ttw(ttr(TTrTest, "smallbuf = 0x%X, %d" NL, smallbuf, smallbuf_size));
3098
3099 ttw(str(TTrTest, "Test Cases:" NL));
3100 for (p = testcase; p->name != 0; p++) {
3101 tw(tr(TR_FUNC, TrTest, "%8s: %s\n", p->name, p->comment));
3102 ttw(ttr(TTrTest, "%8s: %s" NL, p->name, p->comment));
3103 }
3104 return 0;
3105 }
3106
3107 // Stress test the target.
3108 // p0 is number of times to run. Default is 10.
3109 // p1 is number of milli-seconds to wait between each file update.
3110 int case_stress(int p0, int p1)
3111 {
3112 char myname[20];
3113 int i, mysize;
3114 char *mydata;
3115
3116 mydata = (char *) tdata[TDATA_HUGE];
3117 mysize = 9 * param.block_size / 16;
3118
3119 if (p0 == 0) p0 = 10;
3120 if (p1 == 0) p1 = 1000;
3121
3122 for (i = 0; i < p0; i++) {
3123 sprintf(myname, "/stress-%d", i);
3124 ttw(ttr(TTrTest, "/stress %d" NL, i));
3125 error = tffs_fcreate(myname, (char *)mydata, mysize);
3126 expect(error, EFFS_OK);
3127 tffs_delay(p1);
3128 error = tffs_remove(myname);
3129 expect(error, EFFS_OK);
3130 }
3131
3132 return 0;
3133 }
3134
3135 // Test ffs with many (small) files. p0 is number of times to write all the
3136 // files. Default is two e.g. one create and one update
3137 int case_mfiles(int p0, int p1)
3138 {
3139 char myname[40];
3140 int i, j;
3141
3142 static struct test_file_s tf[] =
3143 {
3144 { MFDIR "/MSCAP", 0, 6 },
3145 { MFDIR "/IMEI", 0, 8 },
3146 { MFDIR "/CLASS2", 0, 3 },
3147 { MFDIR "/CLASS3", 0, 2 },
3148 { MFDIR "/MSSUP", 0, 5 },
3149 { MFDIR "/MSSET", 0, 83 },
3150 { MFDIR "/UPN", 0, 165 },
3151 { MFDIR "/CTIM", 0, 92 },
3152 { MFDIR "/CCNT", 0, 52 },
3153 { MFDIR "/ECC", 0, 15 },
3154 { MFDIR "/ORG", 0, 2650 },
3155 { MFDIR "/BCCHINF", 0, 54 },
3156 { MFDIR "/CCP", 0, 7 },
3157 { MFDIR "/EXT1", 0, 13 },
3158 { MFDIR "/SIMLCK", 0, 62 },
3159 { MFDIR "/SIMLCKEXT",0, 172 },
3160 { MFDIR "/SECURITY", 0, 8 },
3161 { MFDIR "/MAIN", 0, 8 },
3162 { MFDIR "/SFK", 0, 8 },
3163 { MFDIR "/FAULT", 0, 8 },
3164 { MFDIR "/DEBUG", 0, 8 },
3165 { MFDIR "/POWER", 0, 8 },
3166 { MFDIR "/KEYB", 0, 64 },
3167 { MFDIR "/RADIO", 0, 8 },
3168 { MFDIR "/CGMI", 0, 20 },
3169 { MFDIR "/CGMM", 0, 20 },
3170 { MFDIR "/CGMR", 0, 20 },
3171 { MFDIR "/CGSN", 0, 20 },
3172 { MFDIR "/SMSPRFL", 0, 206 },
3173 { MFDIR "/PLMN", 0, 68 },
3174 { MFDIR "/ATRADIO", 0, 2700 },
3175 { MFDIR "/GROUP", 0, 1122 }
3176 };
3177
3178 error = tffs_preformat(0xDEAD);
3179 expect(error, EFFS_OK);
3180 error = tffs_format("/ffs/i2048o1024", 0x2BAD);
3181 expect(error, EFFS_OK);
3182 error = tffs_mkdir(MFDIR);
3183 expect(error, EFFS_OK);
3184
3185 if (p0 == 0) p0 = 2;
3186
3187 for (j = 0; j < p0; j++) {
3188 tw(tr(TR_FUNC, TrTestHigh, "mf: Writing many files, loop %d\n", j));
3189
3190 tw(tr(TR_FUNC, TrTestHigh, "SMS000..SMS099\n"));
3191 for (i = 0; i < 100; i++) {
3192 sprintf(myname, MFDIR "/SMS%03d", i);
3193 error = tffs_fwrite(myname, (char *) tdata[TDATA_HUGE], 176);
3194 expect(error, EFFS_OK);
3195 }
3196
3197 tw(tr(TR_FUNC, TrTestHigh, "LDN/LRN/LMN000..LDN/LRN/LMN009\n"));
3198 for (i = 0; i < 10; i++) {
3199 sprintf(myname, MFDIR "/LDN%03d", i);
3200 error = tffs_fwrite(myname, (char *) tdata[TDATA_HUGE], 31);
3201 expect(error, EFFS_OK);
3202 sprintf(myname, MFDIR "/LRN%03d", i);
3203 error = tffs_fwrite(myname, (char *) tdata[TDATA_HUGE], 32);
3204 expect(error, EFFS_OK);
3205 sprintf(myname, MFDIR "/LMN%03d", i);
3206 error = tffs_fwrite(myname, (char *) tdata[TDATA_HUGE], 30);
3207 expect(error, EFFS_OK);
3208 }
3209
3210 tw(tr(TR_FUNC, TrTestHigh, "ADN000..ADN149\n"));
3211 for (i = 0; i < 150; i++) {
3212 sprintf(myname, MFDIR "/ADN%03d", i);
3213 error = tffs_fwrite(myname, (char *) tdata[TDATA_HUGE], 43);
3214 expect(error, EFFS_OK);
3215 }
3216
3217 tw(tr(TR_FUNC, TrTestHigh, "PCM Files...\n"));
3218 for (i = 0; i < sizeof(tf)/sizeof(struct test_file_s); i++) {
3219 error = tffs_fwrite(tf[i].name,
3220 (char *) tdata[TDATA_HUGE], tf[i].size);
3221 expect(error, EFFS_OK);
3222 }
3223 }
3224
3225 error = tffs_exit();
3226 expect(error, EFFS_OK);
3227
3228 return 0;
3229 }
3230
3231
3232 int case_open(int p0, int p1)
3233 {
3234 fd_t fdi;
3235 char myname[20];
3236 int i, size;
3237
3238 #define RDONLY FFS_O_RDONLY
3239 #define WRONLY FFS_O_WRONLY
3240 #define APPEND FFS_O_APPEND
3241 #define CREATE FFS_O_CREATE
3242 #define EXCL FFS_O_EXCL
3243 #define TRUNC FFS_O_TRUNC
3244
3245 ffs_options_t non_exist_invalid[] = {
3246 ( 0 ),
3247 ( RDONLY | CREATE ),
3248 ( RDONLY | CREATE | EXCL ),
3249 ( RDONLY | CREATE | TRUNC ),
3250 ( RDONLY | CREATE | EXCL | TRUNC ),
3251 ( RDONLY | EXCL ),
3252 ( RDONLY | EXCL | TRUNC ),
3253 ( RDONLY | TRUNC ),
3254 ( EXCL ),
3255 ( EXCL | TRUNC ),
3256 ( TRUNC ),
3257 ( CREATE | TRUNC ),
3258 ( CREATE | EXCL | TRUNC ),
3259 ( CREATE | EXCL ),
3260 ( CREATE ),
3261 ( APPEND ),
3262 ( APPEND | EXCL ),
3263 ( APPEND | EXCL | TRUNC ),
3264 ( APPEND | TRUNC ),
3265 ( APPEND | CREATE | TRUNC ),
3266 ( APPEND | CREATE | EXCL ),
3267 ( APPEND | CREATE | EXCL ),
3268 ( APPEND | CREATE ),
3269 ( RDONLY | APPEND ),
3270 ( RDONLY | APPEND | EXCL ),
3271 ( RDONLY | APPEND | EXCL | TRUNC ),
3272 ( RDONLY | APPEND | TRUNC ),
3273 ( RDONLY | APPEND | CREATE | TRUNC ),
3274 ( RDONLY | APPEND | CREATE | EXCL | TRUNC ),
3275 ( RDONLY | APPEND | CREATE | EXCL ),
3276 ( RDONLY | APPEND | CREATE )
3277 };
3278
3279 ffs_options_t non_exist_notfound[] = {
3280 ( WRONLY | APPEND ),
3281 ( WRONLY | APPEND | EXCL ),
3282 ( WRONLY | APPEND | EXCL | TRUNC ),
3283 ( WRONLY | APPEND | TRUNC ),
3284 ( WRONLY ),
3285 ( WRONLY | EXCL ),
3286 ( WRONLY | EXCL | TRUNC ),
3287 ( WRONLY | TRUNC ),
3288 ( RDONLY | WRONLY ),
3289 ( RDONLY | WRONLY | EXCL ),
3290 ( RDONLY | WRONLY | EXCL | TRUNC ),
3291 ( RDONLY | WRONLY | TRUNC ),
3292 ( RDONLY | WRONLY | APPEND ),
3293 ( RDONLY | WRONLY | APPEND | EXCL ),
3294 ( RDONLY | WRONLY | APPEND | EXCL | TRUNC ),
3295 ( RDONLY | WRONLY | APPEND | TRUNC ),
3296 ( RDONLY )
3297 };
3298
3299 ffs_options_t non_exist_fd_offset[] = {
3300 ( WRONLY | APPEND | CREATE | TRUNC ),
3301 ( WRONLY | APPEND | CREATE | EXCL | TRUNC ),
3302 ( WRONLY | APPEND | CREATE | EXCL ),
3303 ( WRONLY | APPEND | CREATE ),
3304 ( WRONLY | CREATE | TRUNC ),
3305 ( WRONLY | CREATE | EXCL | TRUNC ),
3306 ( WRONLY | CREATE | EXCL ),
3307 ( WRONLY | CREATE ),
3308 ( RDONLY | WRONLY | CREATE | TRUNC ),
3309 ( RDONLY | WRONLY | CREATE | EXCL | TRUNC ),
3310 ( RDONLY | WRONLY | CREATE | EXCL ),
3311 ( RDONLY | WRONLY | CREATE ),
3312
3313 ( RDONLY | WRONLY | APPEND | CREATE | TRUNC ),
3314 ( RDONLY | WRONLY | APPEND | CREATE | EXCL | TRUNC ),
3315 ( RDONLY | WRONLY | APPEND | CREATE | EXCL ),
3316 ( RDONLY | WRONLY | APPEND | CREATE )
3317 };
3318
3319 ffs_options_t exist_invalid[] = {
3320 ( EXCL ),
3321 ( EXCL | TRUNC ),
3322 ( TRUNC ),
3323 ( CREATE | TRUNC ),
3324 ( CREATE | EXCL | TRUNC ),
3325 ( CREATE | EXCL ),
3326 ( CREATE ),
3327 ( APPEND ),
3328 ( APPEND | EXCL ),
3329 ( APPEND | EXCL | TRUNC ),
3330 ( APPEND | TRUNC ),
3331 ( APPEND | CREATE | TRUNC ),
3332 ( APPEND | CREATE | EXCL | TRUNC ),
3333 ( APPEND | CREATE | EXCL ),
3334 ( APPEND | CREATE ),
3335 ( RDONLY | APPEND ),
3336 ( RDONLY | APPEND | EXCL ),
3337 ( RDONLY | APPEND | EXCL | TRUNC ),
3338 ( RDONLY | APPEND | TRUNC ),
3339 ( RDONLY | APPEND | CREATE | TRUNC ),
3340 ( RDONLY | APPEND | CREATE | EXCL | TRUNC ),
3341 ( RDONLY | APPEND | CREATE | EXCL ),
3342 ( RDONLY | APPEND | CREATE ),
3343 ( RDONLY | EXCL ),
3344 ( RDONLY | EXCL | TRUNC ),
3345 ( RDONLY | TRUNC ),
3346 ( RDONLY | CREATE | TRUNC ),
3347 ( RDONLY | CREATE | EXCL | TRUNC ),
3348 ( RDONLY | CREATE | EXCL ),
3349 ( RDONLY | CREATE )
3350 };
3351
3352 ffs_options_t exist_exist[] = {
3353 ( WRONLY | APPEND | CREATE | EXCL | TRUNC ),
3354 ( WRONLY | APPEND | CREATE | EXCL ),
3355 ( WRONLY | CREATE | EXCL | TRUNC ),
3356 ( WRONLY | CREATE | EXCL ),
3357 ( RDONLY | WRONLY | CREATE | EXCL | TRUNC ),
3358 ( RDONLY | WRONLY | CREATE | EXCL ),
3359 ( RDONLY | WRONLY | APPEND | CREATE | EXCL | TRUNC ),
3360 ( RDONLY | WRONLY | APPEND | CREATE | EXCL )
3361 };
3362
3363 ffs_options_t exist_fd_offset[] = {
3364 ( WRONLY | APPEND ),
3365 ( WRONLY | APPEND | EXCL ),
3366 ( WRONLY | APPEND | EXCL | TRUNC ),
3367 ( WRONLY | APPEND | TRUNC ),
3368 ( WRONLY | APPEND | CREATE | TRUNC ),
3369 ( WRONLY | APPEND | CREATE ),
3370 ( RDONLY ),
3371 ( RDONLY | WRONLY ),
3372 ( RDONLY | WRONLY | EXCL ),
3373 ( RDONLY | WRONLY | EXCL | TRUNC ),
3374 ( RDONLY | WRONLY | TRUNC ),
3375 ( RDONLY | WRONLY | CREATE | TRUNC ),
3376 ( RDONLY | WRONLY | CREATE ),
3377 ( RDONLY | WRONLY | APPEND ),
3378 ( RDONLY | WRONLY | APPEND | EXCL ),
3379 ( RDONLY | WRONLY | APPEND | EXCL | TRUNC ),
3380 ( RDONLY | WRONLY | APPEND | TRUNC ),
3381 ( RDONLY | WRONLY | APPEND | CREATE | TRUNC ),
3382 ( RDONLY | WRONLY | APPEND | CREATE ),
3383 ( WRONLY ),
3384 ( WRONLY | EXCL ),
3385 ( WRONLY | EXCL | TRUNC ),
3386 ( WRONLY | TRUNC ),
3387 ( WRONLY | CREATE | TRUNC ),
3388 ( WRONLY | CREATE )
3389
3390 };
3391
3392 // Cleanup
3393 for (i = 0; i < 20; i++) {
3394 sprintf(myname, "/stream%d", i);
3395 tffs_remove(myname);
3396 }
3397
3398 tw(tr(TR_FUNC, TrTestHigh, "Various open options, file non-existing\n"));
3399 ttw(str(TTrTest, "Various open options, file non-existing" NL));
3400
3401 for (i = 0; i < sizeof(non_exist_invalid)/sizeof(ffs_options_t); i++) {
3402 sprintf(myname, "/stream%d", i - i/20 * 20);
3403 fdi = tffs_open(myname, non_exist_invalid[i]);
3404 expect(fdi, EFFS_INVALID);
3405 }
3406
3407 for (i = 0; i < sizeof(non_exist_notfound)/sizeof(ffs_options_t); i++) {
3408 sprintf(myname, "/stream%d", i - i/20 * 20);
3409 fdi = tffs_open(myname, non_exist_notfound[i]);
3410 expect(fdi, EFFS_NOTFOUND);
3411 }
3412
3413 for (i = 0; i < sizeof(non_exist_fd_offset)/sizeof(ffs_options_t); i++) {
3414 sprintf(myname, "/stream%d", i - i/20 * 20);
3415 fdi = tffs_open(myname, non_exist_fd_offset[i]);
3416 expect(fdi, FFS_FD_OFFSET);
3417 tffs_close(fdi);
3418 }
3419
3420 // Make FFS_FD_MAX number of files and write some data for later use
3421 tw(tr(TR_FUNC, TrTestHigh, "Create %d files with data \n", fs.fd_max));
3422 for (i = 0; i < fs.fd_max; i++) {
3423 sprintf(myname, "/stream%d", i);
3424 fdi = tffs_open(myname, FFS_O_WRONLY | FFS_O_CREATE | FFS_O_TRUNC);
3425 expect(fdi, i + FFS_FD_OFFSET);
3426 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], 31);
3427 expect(size, 31);
3428 }
3429
3430 // Try to open one more file, this will make a error!
3431 sprintf(myname, "/stream%d", i++);
3432 fdi = tffs_open(myname, FFS_O_WRONLY | FFS_O_CREATE | FFS_O_TRUNC);
3433 expect(fdi, EFFS_NUMFD);
3434
3435 for (i = 0; i < fs.fd_max; i++) {
3436 error = tffs_close(i + FFS_FD_OFFSET);
3437 expect(error, EFFS_OK);
3438 }
3439
3440 tw(tr(TR_FUNC, TrTestHigh,
3441 "Open same file multiple times in RDONLY mode\n"));
3442 for (i = 0; i < fs.fd_max; i++) {
3443 fdi = tffs_open("/stream2", FFS_O_RDONLY);
3444 expect(fdi, i + FFS_FD_OFFSET);
3445 }
3446
3447 for (i = 0; i < fs.fd_max; i++) {
3448 error = tffs_close(i + FFS_FD_OFFSET);
3449 expect(error, 0);
3450 }
3451
3452 tw(tr(TR_FUNC, TrTestHigh,
3453 "Open in WRONLY and try to open same file in WR or RD\n"));
3454 fdi = tffs_open("/stream1", FFS_O_WRONLY | FFS_O_APPEND);
3455 expect(fdi, FFS_FD_OFFSET);
3456 fdi = tffs_open("/stream1", FFS_O_WRONLY | FFS_O_APPEND);
3457 expect(fdi, EFFS_LOCKED);
3458 fdi = tffs_open("/stream1", FFS_O_RDONLY );
3459 expect(fdi, EFFS_LOCKED);
3460 error = tffs_close(FFS_FD_OFFSET);
3461 expect(error, 0);
3462
3463 tw(tr(TR_FUNC, TrTestHigh,
3464 "Open in READONLY and try to open same file WRONLY\n"));
3465 fdi = tffs_open("/stream1", FFS_O_RDONLY);
3466 expect(fdi, FFS_FD_OFFSET);
3467 fdi = tffs_open("/stream1", FFS_O_WRONLY | FFS_O_APPEND);
3468 expect(fdi, EFFS_LOCKED);
3469 error = tffs_close(FFS_FD_OFFSET);
3470 expect(error, 0);
3471
3472 tw(tr(TR_FUNC, TrTestHigh, "Various open options, file exists\n"));
3473 ttw(str(TTrTest, "Various open options, file exists" NL));
3474
3475 for (i = 0; i < sizeof(exist_invalid)/sizeof(ffs_options_t); i++) {
3476 sprintf(myname, "/stream%d", i - i/20 * 20);
3477 fdi = tffs_open(myname, exist_invalid[i]);
3478 expect(fdi, EFFS_INVALID);
3479 }
3480
3481 for (i = 0; i < sizeof(exist_exist)/sizeof(ffs_options_t); i++) {
3482 sprintf(myname, "/stream%d", i - i/20 * 20);
3483 fdi = tffs_open(myname, exist_exist[i]);
3484 expect(fdi, EFFS_EXISTS);
3485 tffs_close(fdi);
3486 }
3487
3488 for (i = 0; i < sizeof(exist_fd_offset)/sizeof(ffs_options_t); i++) {
3489 sprintf(myname, "/stream%d", i - i/10 * 10);
3490 fdi = tffs_open(myname, exist_fd_offset[i]);
3491 expect(fdi, FFS_FD_OFFSET);
3492 tffs_close(fdi);
3493 }
3494
3495 tw(tr(TR_FUNC, TrTestHigh, "Try to create a file with a dir name\n"));
3496 error = tffs_mkdir("/streams");
3497
3498 fdi = tffs_open("/streams", FFS_O_WRONLY | FFS_O_APPEND | FFS_O_CREATE);
3499 expect(fdi, EFFS_NOTAFILE);
3500
3501 tw(tr(TR_FUNC, TrTestHigh,
3502 "Try to use fwrite, fcreate, fread and remove on a open file\n"));
3503 fdi = tffs_open("/stream1", FFS_O_WRONLY | FFS_O_APPEND);
3504 expect(fdi, FFS_FD_OFFSET);
3505
3506 error = tffs_fcreate("/stream1", TDATA(2));
3507 expect(error, EFFS_EXISTS);
3508
3509 error = tffs_file_write("/stream1", (char *) tdata[TDATA_HUGE], 31,
3510 FFS_O_CREATE | FFS_O_TRUNC);
3511 expect(error, EFFS_LOCKED);
3512
3513 error = tffs_file_read("/stream1", bigbuf, 31);
3514 expect(error, EFFS_LOCKED);
3515
3516 error = tffs_remove("/stream1");
3517 expect(error, EFFS_LOCKED);
3518
3519 error = tffs_close(fdi);
3520 expect(error, 0);
3521
3522 #undef RDONLY
3523 #undef WRONLY
3524 #undef APPEND
3525 #undef CREATE
3526 #undef EXCL
3527 #undef TRUNC
3528
3529 return 0;
3530 }
3531
3532 int twrite_seek(fd_t fdi, offset_t offset, int whence)
3533 {
3534 int size;
3535
3536 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE] + 10, 100);
3537 if (test_expect(size, 100) < 0) return -1;
3538
3539 offset = tffs_seek(fdi, offset, whence);
3540 if (test_expect_ok(offset)) return -1;
3541
3542 return offset;
3543 }
3544
3545 int tseek_read(fd_t fdi, offset_t offset, int whence, offset_t src_offset)
3546 {
3547 int size;
3548
3549 offset = tffs_seek(fdi, offset, whence);
3550 if (test_expect_ok(offset)) return -1;
3551
3552 size = tffs_read(fdi, bigbuf, 100);
3553 if (test_expect(size, 100) < 0) return -1;
3554
3555 error = test_expect_data((char*)tdata[TDATA_HUGE] + src_offset, bigbuf, size);
3556 if (error) return error;
3557
3558 return offset;
3559 }
3560
3561 int tseek_write(fd_t fdi, offset_t offset, int whence, offset_t src_offset)
3562 {
3563 int size;
3564
3565 offset = tffs_seek(fdi, offset, whence);
3566 if (test_expect_ok(offset)) return -1;
3567
3568 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE] + src_offset + offset,
3569 fs.chunk_size_max / 2);
3570 if (test_expect(size, fs.chunk_size_max / 2) < 0) return -1;
3571
3572 return offset;
3573 }
3574
3575 int case_rw(int p0, int p1)
3576 {
3577 int i, size, offset;
3578 fd_t fdi;
3579
3580 const char *dirs[] = { "/streams", "/streams/rw", "/streams/rw/multi",
3581 "/streams/rw/multi/write" };
3582
3583 error = case_cleanup(5 * fs.chunk_size_max);
3584 expect_ok(error);
3585
3586 for (i = 0; i < sizeof(dirs)/sizeof(char *); i++) {
3587 error = tffs_mkdir(dirs[i]);
3588 }
3589
3590 tw(tr(TR_FUNC, TrTestHigh,
3591 "Test create and append of file, small and huge\n"));
3592 ttw(str(TTrTest, "Test create and append of file, small and huge" NL));
3593
3594 fdi = tffs_open("/streams/rw_test",
3595 FFS_O_WRONLY | FFS_O_CREATE | FFS_O_TRUNC);
3596 expect(fdi, FFS_FD_OFFSET);
3597 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], 31);
3598 expect(size, 31);
3599 error = tffs_close(fdi); expect(error, EFFS_OK);
3600
3601 fdi = tffs_open("/streams/rw_test", FFS_O_WRONLY | FFS_O_APPEND);
3602 expect(fdi, FFS_FD_OFFSET);
3603 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE] + 31, 2 * fs.chunk_size_max);
3604 expect(size, 2 * fs.chunk_size_max);
3605
3606 error = tffs_close(fdi); expect(error, EFFS_OK);
3607
3608 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
3609 ttw(ttr(TTrTest, "Validate data" NL));
3610 size = tffs_fread("/streams/rw_test", bigbuf, bigbuf_size);
3611 expect(size, 31 + 2 * fs.chunk_size_max);
3612 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, size);
3613 if (error) return 1;
3614
3615 // Provoke EFFS_FILETOOBIG
3616 error = tffs_fread("/streams/rw_test", bigbuf, size - 1);
3617 expect(error, EFFS_FILETOOBIG);
3618 error = tffs_fread("/streams/rw_test", bigbuf, size + 1);
3619 expect_ok(error);
3620 error = tffs_fread("/streams/rw_test", bigbuf, size);
3621 expect_ok(error);
3622
3623 // Update file from zero, middle (between to chunks) and end. File with
3624 // and without data in seghead. Update the first 100 bytes (new data
3625 // have a offset of 10 so the data not is identical with the existing
3626 // ones).
3627 fdi = tffs_open("/streams/rw_test", FFS_O_RDWR);
3628 expect(fdi, FFS_FD_OFFSET);
3629
3630 // write 100 bytes and seek to 50 bytes before end of the first chunk
3631 expect_ok(twrite_seek(fdi, fs.chunk_size_max - 50, FFS_SEEK_SET));
3632
3633 // write 100 bytes, seek to 50 bytes from the end of the file
3634 expect_ok(twrite_seek(fdi, -50, FFS_SEEK_END));
3635
3636 // Write 100 bytes. This will update the last 19 bytes of second chunk,
3637 // update and append data to the last chunk. Read the last data but
3638 // only 80 bytes because they are in the buffer
3639 expect_ok(twrite_seek(fdi, -80, FFS_SEEK_CUR));
3640
3641 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
3642 ttw(ttr(TTrTest, "Validate data" NL));
3643
3644 size = tffs_read(fdi, bigbuf, 100);
3645 expect(size, 80);
3646 error = test_expect_data((char *)tdata[TDATA_HUGE] + 10 + 20, bigbuf, size);
3647 if (error) return 1;
3648
3649 // Seek to start of file and read the first 100 bytes that was updated
3650 offset = tseek_read(fdi, 0, FFS_SEEK_SET, 10);
3651 expect_ok(offset);
3652
3653 // Read 100 bytes more to make sure that the old data still is valid
3654 offset = tseek_read(fdi, 0, FFS_SEEK_CUR, 100);
3655 expect_ok(offset);
3656
3657 // Seek to 50 bytes before end of the first chunk and read 100 bytes
3658 offset = tseek_read(fdi, fs.chunk_size_max - 50, FFS_SEEK_SET, 10);
3659 expect_ok(offset);
3660
3661 // Read 100 bytes more to make sure that the old data still is valid
3662 offset = tseek_read(fdi, 0, FFS_SEEK_CUR, offset + 100);
3663 expect_ok(offset);
3664
3665 // Read the last 100 bytes (updated data from 2 chunks)
3666 offset = tseek_read(fdi, -100, FFS_SEEK_END, 10);
3667 expect_ok(offset);
3668
3669 error = tffs_close(fdi);
3670 expect(error, EFFS_OK);
3671
3672 // Test where there is data in the seghead
3673 size = tffs_file_write("/streams/rw_test", (char*) tdata[TDATA_HUGE],
3674 2 * fs.chunk_size_max, FFS_O_CREATE | FFS_O_TRUNC);
3675
3676 fdi = tffs_open("/streams/rw_test", FFS_O_RDWR);
3677 expect(fdi, FFS_FD_OFFSET);
3678
3679 // Update the last half of the last chunk
3680 expect_ok(tseek_write(fdi, -fs.chunk_size_max / 2, FFS_SEEK_END, 10));
3681
3682 // Update the first half of the last chunk
3683 expect_ok(tseek_write(fdi, fs.chunk_size_max, FFS_SEEK_SET, 10));
3684
3685 // Update the first half of the seghead
3686 expect_ok(tseek_write(fdi, 0, FFS_SEEK_SET, 10));
3687
3688 // Update the last half of the seghead, but flush the buffer so it have
3689 // to write and read the previous updated data
3690 error = tffs_fdatasync(fdi);
3691 expect(error, EFFS_OK);
3692 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE] + 10 + fs.chunk_size_max/2,
3693 fs.chunk_size_max/2);
3694 expect(size, fs.chunk_size_max/2);
3695
3696 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
3697 ttw(ttr(TTrTest, "Validate data" NL));
3698 // Read the complete file, all the data should have a offset of 10 bytes now!
3699 offset = tffs_seek(fdi, 0, FFS_SEEK_SET);
3700 expect_ok(offset);
3701
3702 size = tffs_read(fdi, bigbuf, bigbuf_size);
3703 expect(size, 2 * fs.chunk_size_max);
3704 error = test_expect_data((char *)tdata[TDATA_HUGE] + 10, bigbuf, size);
3705 if (error) return 1;
3706
3707 error = tffs_close(fdi); expect(error, 0);
3708
3709 // Test bad fdi, write in RDONLY and read from WRONLY
3710 fdi = tffs_open("/streams/rw_test", FFS_O_RDONLY);
3711 expect(fdi, FFS_FD_OFFSET);
3712 size = tffs_read(fdi + 1, bigbuf, 31);
3713 expect(size, EFFS_BADFD);
3714 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], 31);
3715 tw(tr(TR_END, TrApi, "} %d\n", size)); // Avoid wrong indent
3716 expect(size, EFFS_INVALID);
3717 error = tffs_close(fdi); expect(error, 0);
3718
3719 fdi = tffs_open("/streams/rw_test", FFS_O_WRONLY | FFS_O_APPEND);
3720 expect(fdi, FFS_FD_OFFSET);
3721 size = tffs_read(fdi, bigbuf, 31);
3722 expect(size, EFFS_INVALID);
3723 error = tffs_close(fdi); expect(error, 0);
3724
3725 // Test for bad file name
3726 fdi = tffs_open("/stream&!'*", FFS_O_WRONLY | FFS_O_CREATE);
3727 expect(fdi, EFFS_BADNAME);
3728
3729 // Test for bad file name
3730 fdi = tffs_open("/%$#.+-_,Z19", FFS_O_WRONLY | FFS_O_CREATE);
3731 expect(fdi, FFS_FD_OFFSET);
3732 tffs_close(fdi);
3733
3734 return 0;
3735 }
3736
3737 int case_multi_open(int p0, int p1)
3738 {
3739 int i, j, size;
3740 fd_t fdi;
3741
3742 const char *dirs[] = { "/streams", "/streams/rw", "/streams/rw/multi",
3743 "/streams/rw/multi/write" };
3744
3745 const char *file[] = { "/x-rw", "/streams/rw/x-rw",
3746 "/streams/rw/multi/x-rw",
3747 "/streams/rw/multi/write/x-rw" };
3748
3749 error = case_cleanup(fs.fd_max * 50 * 10 + 5000);
3750 expect_ok(error);
3751
3752 for (i = 0; i < sizeof(dirs)/sizeof(char *); i++) {
3753 error = tffs_mkdir(dirs[i]);
3754 }
3755
3756 // Open multiply files from difference subdirectorys
3757 // and write/read X times to/from file (this also test truncate option)
3758 for (i = 0; i < fs.fd_max; i++) {
3759 fdi = tffs_open( file[i], FFS_O_WRONLY | FFS_O_APPEND |
3760 FFS_O_CREATE | FFS_O_TRUNC);
3761 expect(fdi, i + FFS_FD_OFFSET);
3762 }
3763
3764 for (i = 0; i < fs.fd_max; i++) {
3765 for (j = 0; j < 10; j++) {
3766 size = tffs_write(i + FFS_FD_OFFSET, (char *) tdata[TDATA_HUGE], 50);
3767 expect(size, 50);
3768 }
3769 }
3770 // Run test with p0 != 0 to stress the test
3771 if (p0) { error = test_run("irec;drec;"); if (error > 0) return 1; }
3772
3773 for (i = 0; i < fs.fd_max; i++) {
3774 error = tffs_close(i + FFS_FD_OFFSET);
3775 expect(error, 0);
3776 }
3777
3778 for (i = 0; i < fs.fd_max; i++) {
3779 fdi = tffs_open( file[i], FFS_O_RDONLY);
3780 expect(fdi, i + FFS_FD_OFFSET);
3781 }
3782
3783 if(p0) { error = test_run("irec;drec;"); if (error > 0) return 1; }
3784
3785 for (i = 0; i < fs.fd_max; i++) {
3786 for (j = 0; j < 10; j++) {
3787 size = tffs_read(i + FFS_FD_OFFSET, bigbuf, 50);
3788 expect(size, 50);
3789 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, size);
3790 if (error) return 1;
3791 }
3792 }
3793
3794 for (i = 0; i < fs.fd_max; i++) {
3795 error = tffs_close(i + FFS_FD_OFFSET);
3796 expect(error, 0);
3797 }
3798
3799 return 0;
3800 }
3801
3802
3803 int case_seek(int p0, int p1)
3804 {
3805 fd_t fdi;
3806 int size, offset;
3807
3808 error = case_cleanup(3 * fs.chunk_size_max);
3809 expect_ok(error);
3810
3811 // The seek function itself don't care if the file is open in read or
3812 // write mode.
3813 error = tffs_mkdir("/streams");
3814
3815 offset = tffs_seek(FFS_FD_OFFSET, 0, FFS_SEEK_SET);
3816 expect(offset, EFFS_BADFD);
3817
3818 /* Make file to seek on */
3819 fdi = tffs_open("/streams/seek", FFS_O_RDWR | FFS_O_CREATE | FFS_O_TRUNC);
3820 expect(fdi, FFS_FD_OFFSET);
3821 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], 3 * fs.chunk_size_max);
3822 expect(size, 3 * fs.chunk_size_max);
3823
3824 /* Simple test in RDWR without any writes. Test good and bad offset */
3825 tw(tr(TR_FUNC, TrTestHigh, "seek and RDWR \n"));
3826 ttw(str(TTrTest, "seek and RDWR" NL));
3827
3828 offset = ffs_seek(fdi, 0, 5);
3829 expect(offset, EFFS_INVALID );
3830
3831 tw(tr(TR_FUNC, TrTestHigh, "ffs_seek(FFS_SEEK_SET)\n"));
3832 ttw(str(TTrTest, "ffs_seek(FFS_SEEK_SET)" NL));
3833 offset = tffs_seek(fdi, -1, FFS_SEEK_SET);
3834 expect(offset, EFFS_INVALID);
3835 offset = tffs_seek(fdi, 3 * fs.chunk_size_max + 1, FFS_SEEK_SET);
3836 expect(offset, EFFS_INVALID);
3837
3838 offset = tffs_seek(fdi, 15, FFS_SEEK_SET);
3839 expect(offset, 15);
3840 size = tffs_read(fdi, bigbuf, fs.chunk_size_max/10);
3841 expect(size, fs.chunk_size_max/10);
3842 error = test_expect_data((char *)tdata[TDATA_HUGE] + 15,
3843 bigbuf, fs.chunk_size_max/10);
3844 if (error) return 1;
3845
3846 offset = tffs_seek(fdi, 0, FFS_SEEK_SET);
3847 expect(offset, 0);
3848
3849 size = tffs_read(fdi, bigbuf, bigbuf_size);
3850 error = test_expect_data((char*)tdata[TDATA_HUGE], bigbuf, size);
3851 if (error) return 1;
3852
3853 tw(tr(TR_FUNC, TrTestHigh, "ffs_seek(FFS_SEEK_CUR)\n"));
3854 ttw(str(TTrTest, "ffs_seek(FFS_SEEK_CUR)" NL));
3855 offset = tffs_seek(fdi, -(3 * fs.chunk_size_max + 1), FFS_SEEK_CUR);
3856 expect(offset, EFFS_INVALID);
3857 offset = tffs_seek(fdi, 1, FFS_SEEK_CUR);
3858 expect(offset, EFFS_INVALID);
3859 offset = tffs_seek(fdi, 0, FFS_SEEK_CUR);
3860 expect(offset, 3 * fs.chunk_size_max);
3861 offset = tffs_seek(fdi, -30, FFS_SEEK_CUR);
3862 expect(offset, 3 * fs.chunk_size_max - 30);
3863 offset = tffs_seek(fdi, 20, FFS_SEEK_CUR);
3864 expect(offset, 3 * fs.chunk_size_max - 30 + 20);
3865 size = tffs_read(fdi, bigbuf, 3 * fs.chunk_size_max);
3866 expect(size, 10);
3867 error = test_expect_data((char *)tdata[TDATA_HUGE] +
3868 (3 * fs.chunk_size_max - 10), bigbuf, 10);
3869 if (error) return 1;
3870
3871 tw(tr(TR_FUNC, TrTestHigh, "ffs_seek(FFS_SEEK_END)\n"));
3872 ttw(str(TTrTest, "ffs_seek(FFS_SEEK_END)" NL));
3873 offset = tffs_seek(fdi, -(3 * fs.chunk_size_max + 1), FFS_SEEK_END);
3874 expect(offset, EFFS_INVALID);
3875 offset = tffs_seek(fdi, 1, FFS_SEEK_END);
3876 expect(offset, EFFS_INVALID);
3877 offset = tffs_seek(fdi, 0, FFS_SEEK_END);
3878 expect(offset, 3 * fs.chunk_size_max);
3879 offset = tffs_seek(fdi, -15, FFS_SEEK_END);
3880 expect(offset, 3 * fs.chunk_size_max - 15);
3881 size = tffs_read(fdi, bigbuf, 20);
3882 expect(size, 15);
3883 error = test_expect_data((char *)tdata[TDATA_HUGE] +
3884 (3 * fs.chunk_size_max - 15), bigbuf, size);
3885 if (error) return 1;
3886
3887 error = tffs_close(fdi);
3888 expect(error, 0);
3889
3890 // Seek in write-mode with write in different places of the file. Test
3891 // on the dirty flag to se if the buffer is flushed ad the rigth time
3892 tw(tr(TR_FUNC, TrTestHigh, "seek and WRONLY \n"));
3893 ttw(str(TTrTest, "seek and WRONLY" NL));
3894
3895 fdi = tffs_open("/streams/seek", FFS_O_WRONLY);
3896
3897 // Write and seek in the last chunk, buffer must be dirty so long the fp
3898 // not have been moved away from that chunk!
3899
3900 offset = tffs_seek(fdi, -20, FFS_SEEK_END); expect_ok(offset);
3901 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 0);
3902 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE], 10); expect(size, 10);
3903 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3904
3905 offset = tffs_seek(fdi, -100, FFS_SEEK_CUR); expect_ok(offset);
3906 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3907 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE], 10); expect(size, 10);
3908 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3909
3910 offset = tffs_seek(fdi, -fs.chunk_size_max/4, FFS_SEEK_CUR); expect_ok(offset);
3911 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3912 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE], 10); expect(size, 10);
3913 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3914
3915 offset = tffs_seek(fdi, offset + 10, FFS_SEEK_SET); expect_ok(offset);
3916 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3917 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE], 10); expect(size, 10);
3918 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3919
3920 // Write and seek across several chunks, the buffer must be flushed after
3921 // leaving a 'dirty' chunk.
3922
3923 offset = tffs_seek(fdi, 10, FFS_SEEK_SET); expect_ok(offset);
3924 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 0);
3925 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE], 10); expect(size, 10);
3926 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3927
3928 offset = tffs_seek(fdi, -50, FFS_SEEK_END); expect_ok(offset);
3929 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 0);
3930 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE], 100); expect(size, 100);
3931 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3932
3933 offset = tffs_seek(fdi, -fs.chunk_size_max, FFS_SEEK_CUR); expect_ok(offset);
3934 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 0);
3935 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE], fs.chunk_size_max / 2);
3936 expect(size, fs.chunk_size_max / 2);
3937 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3938
3939 offset = tffs_seek(fdi, fs.chunk_size_max, FFS_SEEK_SET); expect_ok(offset);
3940 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 0);
3941 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE], 50); expect(size, 50);
3942 expect(fs.fd[fdi - FFS_FD_OFFSET].dirty, 1);
3943
3944 error = tffs_close(fdi);
3945 expect(error, 0);
3946
3947 return 0;
3948 }
3949
3950 // NOTEME: move to other place later
3951 int case_trunc(int p0, int p1)
3952 {
3953 fd_t fdi;
3954 int size, f_size, trunc_size, i;
3955 T_FFS_STAT stat;
3956 char fname[] = "/truncate/normal_file";
3957 char sname[] = "/truncate/stream_file";
3958 char imeifile[] = "/truncate/IMEI";
3959
3960 tffs_remove(fname);
3961 tffs_remove(sname);
3962
3963 error = case_cleanup(5 * fs.chunk_size_max);
3964 expect_ok(error);
3965
3966 tffs_mkdir("/truncate");
3967
3968 // Try to truncate a non existing file
3969 error = tffs_truncate(fname, 0);
3970 expect(error, EFFS_NOTFOUND);
3971
3972 // Try to truncate a directory
3973 error = tffs_truncate("/truncate", 0);
3974 expect(error, EFFS_NOTAFILE);
3975
3976 f_size = fs.chunk_size_max + 100;
3977 error = tffs_file_write(fname, (char *)tdata[TDATA_HUGE], f_size,
3978 FFS_O_TRUNC | FFS_O_WRONLY | FFS_O_CREATE);
3979 expect(error, EFFS_OK);
3980
3981 /* Test truncate from file size + 3 to file size - 3 */
3982 for (i = 0; i < 6; i++) {
3983 trunc_size = f_size + 3 - i;
3984 tw(tr(TR_FUNC, TrTestLow, "TRUNC file to size %d \n", trunc_size));
3985 error = tffs_truncate(fname, trunc_size);
3986 expect(error, EFFS_OK);
3987 error = tffs_lstat(fname, &stat);
3988 expect(error, EFFS_OK);
3989 expect_eq(stat.size, (f_size < trunc_size ? f_size : trunc_size));
3990 }
3991 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
3992 size = tffs_fread(fname, bigbuf, f_size);
3993 expect(size, trunc_size); // ++ because of the last -- in for()
3994 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, size);
3995 if (error) return 1;
3996
3997 tw(tr(TR_FUNC, TrTestLow, "Truncate file while it is open \n"));
3998 fdi = tffs_open(fname, FFS_O_WRONLY | FFS_O_APPEND);
3999 expect(fdi, FFS_FD_OFFSET);
4000 size = tffs_write(fdi, (char *) &tdata[TDATA_HUGE][trunc_size], 31);
4001 expect(size, 31);
4002 f_size = trunc_size + 31;
4003
4004 // Truncate to a size less than fp, this will fail
4005 error = tffs_ftruncate(fdi, f_size - 1);
4006 expect(error, EFFS_INVALID);
4007
4008 // Move fp otherwise it will not be possible to truncate
4009 error = ffs_seek(fdi, 0, FFS_SEEK_SET);
4010 expect(error, 0);
4011
4012 // Test truncate from file size + 3 to file size - 3
4013 for (i = 0; i < 6; i++) {
4014 trunc_size = f_size + 3 - i;
4015 tw(tr(TR_FUNC, TrTestLow, "TRUNC file to size %d \n", trunc_size));
4016 error = tffs_ftruncate(fdi, trunc_size);
4017 expect(error, EFFS_OK);
4018 error = tffs_lstat(fname, &stat);
4019 expect(error, EFFS_OK);
4020 expect_eq(stat.size, (f_size < trunc_size ? f_size : trunc_size));
4021 }
4022
4023 error = tffs_close(fdi);
4024 expect(error, EFFS_OK);
4025
4026 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
4027 size = tffs_fread(fname, bigbuf, f_size);
4028 expect(size, trunc_size);
4029 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, size);
4030 if (error) return 1;
4031
4032 // We need to test ftruncate with a size less than the first chunk
4033 fdi = tffs_open(fname, FFS_O_WRONLY);
4034 expect(fdi, FFS_FD_OFFSET);
4035
4036 error = tffs_ftruncate(fdi, 10);
4037 expect(error, EFFS_OK);
4038 error = tffs_ftruncate(fdi, 10); // same size twice to complicate it
4039 expect(error, EFFS_OK);
4040 error = tffs_lstat(fname, &stat);
4041 expect(error, EFFS_OK);
4042 expect_eq(stat.size, 10);
4043 error = tffs_ftruncate(fdi, 0);
4044 expect(error, EFFS_OK);
4045 error = tffs_ftruncate(fdi, 0);
4046 expect(error, EFFS_OK);
4047 error = tffs_lstat(fname, &stat);
4048 expect(error, EFFS_OK);
4049 expect_eq(stat.size, 0);
4050 error = tffs_close(fdi);
4051 expect(error, EFFS_OK);
4052
4053 // Stream file test, make 2 segments and test around the end of segment
4054 // 1 (file closed)
4055 fdi = tffs_open(sname, FFS_O_WRONLY | FFS_O_CREATE | FFS_O_TRUNC
4056 | FFS_O_APPEND);
4057 expect(fdi, FFS_FD_OFFSET);
4058
4059 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], fs.chunk_size_max + 1);
4060 expect(size, fs.chunk_size_max + 1);
4061 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], 31);
4062 expect(size, 31);
4063 error = tffs_close(fdi);
4064 expect(error, EFFS_OK);
4065
4066 f_size = fs.chunk_size_max + 1 + 31;
4067
4068 for (i = 0; i < 6; i++) {
4069 trunc_size = fs.chunk_size_max + 1 + 3 - i;
4070 tw(tr(TR_FUNC, TrTestLow, "TRUNC stream to size %d \n", trunc_size));
4071 error = tffs_truncate(sname, trunc_size);
4072 expect(error, EFFS_OK);
4073 error = tffs_lstat(sname, &stat);
4074 expect(error, EFFS_OK);
4075 expect_eq(stat.size, trunc_size);
4076 }
4077
4078 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
4079 size = tffs_fread(sname, bigbuf, f_size);
4080 expect(size, trunc_size);
4081 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, size);
4082 if (error) return 1;
4083
4084 // Try to enlarge the file.
4085 tw(tr(TR_FUNC, TrTestLow, "Try to enlarge the file \n"));
4086 error = tffs_truncate(sname, trunc_size + 1);
4087 expect(error, EFFS_OK);
4088 error = tffs_lstat(sname, &stat);
4089 expect(error, EFFS_OK);
4090 expect_eq(stat.size, trunc_size);
4091
4092 fdi = tffs_open(sname, FFS_O_WRONLY);
4093 expect(fdi, FFS_FD_OFFSET);
4094 error = tffs_ftruncate(fdi, trunc_size + 1);
4095 expect(error, EFFS_OK);
4096 error = tffs_lstat(sname, &stat);
4097 expect(error, EFFS_OK);
4098 expect_eq(stat.size, trunc_size);
4099 error = tffs_close(fdi);
4100 expect(error, EFFS_OK);
4101
4102 // Make a file with 2 segments + some data in the buffer, test around
4103 // end of segment 2 and the begining of segment 1 (file open).
4104 tw(tr(TR_FUNC, TrTestLow, "Truncate stream while it is open \n"));
4105 // Make segment 1
4106 fdi = tffs_open(sname, FFS_O_WRONLY | FFS_O_APPEND | FFS_O_TRUNC);
4107 expect(fdi, FFS_FD_OFFSET);
4108 f_size = tffs_write(fdi, (char *) tdata[TDATA_HUGE],
4109 2 * fs.chunk_size_max + 1);
4110 expect(f_size, 2 * fs.chunk_size_max + 1);
4111
4112 // Move fp otherwise it will not be possible to truncate
4113 error = ffs_seek(fdi, 0, FFS_SEEK_SET);
4114 expect(error, 0);
4115
4116 // Test around end of segment 2
4117 for (i = 0; i < 16; i++) {
4118 trunc_size = f_size + 3 - i;
4119 tw(tr(TR_FUNC, TrTestLow, "TRUNC stream to size %d \n",
4120 trunc_size));
4121 error = tffs_ftruncate(fdi, trunc_size);
4122 expect(error, EFFS_OK);
4123 error = tffs_lstat(sname, &stat); expect(error, EFFS_OK);
4124 expect_eq(stat.size, (f_size < trunc_size ? f_size : trunc_size));
4125 }
4126
4127 // Test around begining of segment 1
4128 for (i = 0; i < 6; i++) {
4129 trunc_size = 3 - i;
4130 tw(tr(TR_FUNC, TrTestLow, "TRUNC stream to size %d \n",
4131 trunc_size));
4132 error = tffs_ftruncate(fdi, trunc_size);
4133 if (trunc_size >= 0) {
4134 expect(error, EFFS_OK);
4135 }
4136 else {
4137 expect(error, EFFS_INVALID);
4138 break;
4139 }
4140 error = tffs_lstat(sname, &stat);
4141 expect(error, EFFS_OK);
4142 expect_eq(stat.size, (trunc_size));
4143 }
4144
4145 error = tffs_close(fdi);
4146 expect(error, EFFS_OK);
4147
4148 // Make, truncate and append to a file (use ffs_ftruncate())
4149 fdi = tffs_open(sname, FFS_O_WRONLY | FFS_O_CREATE | FFS_O_TRUNC
4150 | FFS_O_APPEND);
4151 expect(fdi, FFS_FD_OFFSET);
4152
4153 f_size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], fs.chunk_size_max);
4154 expect(f_size, fs.chunk_size_max );
4155
4156 tw(tr(TR_FUNC, TrTestLow, "write 1/2 buf size, trunc 1/4 buf size\n"));
4157 // write 1/2 buf size and truncate 1/4 buf size x times
4158 for (i = 0; i < 8; i++) {
4159 size = tffs_write(fdi, (char *) &tdata[TDATA_HUGE][f_size],
4160 fs.chunk_size_max / 2);
4161 if (size < 0) return size;
4162 f_size += size;
4163 size = tffs_seek(fdi, - fs.chunk_size_max / 4, FFS_SEEK_CUR);
4164 expect(size, f_size - fs.chunk_size_max / 4);
4165
4166 error = tffs_ftruncate(fdi, f_size - fs.chunk_size_max / 4);
4167 expect(error, EFFS_OK);
4168 f_size -= fs.chunk_size_max / 4;
4169
4170 error = tffs_lstat(sname, &stat);
4171 expect(error, EFFS_OK);
4172 expect_eq(stat.size, f_size);
4173 }
4174 error = tffs_close(fdi);
4175 expect(error, EFFS_OK);
4176
4177 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
4178 size = tffs_fread(sname, bigbuf, bigbuf_size);
4179 expect(size, f_size);
4180 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, f_size);
4181 if (error) return 1;
4182
4183 // backwards trunc 1/2 write 1/4 x times
4184 tw(tr(TR_FUNC, TrTestLow, "trunc 1/2 buf size, write 1/4 buf size\n"));
4185 fdi = tffs_open(sname, FFS_O_WRONLY | FFS_O_APPEND);
4186 expect(fdi, FFS_FD_OFFSET);
4187
4188 for (i = 0; i < 8; i++) {
4189 size = tffs_seek(fdi, - fs.chunk_size_max / 2, FFS_SEEK_CUR);
4190 expect(size, f_size - fs.chunk_size_max / 2);
4191
4192 error = tffs_ftruncate(fdi, f_size - fs.chunk_size_max / 2);
4193 expect(error, EFFS_OK);
4194 f_size -= fs.chunk_size_max / 2;
4195
4196 f_size += tffs_write(fdi, (char *) &tdata[TDATA_HUGE][f_size],
4197 fs.chunk_size_max / 4);
4198 error = tffs_lstat(sname, &stat);
4199 expect(error, EFFS_OK);
4200 expect_eq(stat.size, f_size);
4201 }
4202
4203 if (p0) { error = test_run("irec"); if (error > 0) return 1; }
4204
4205 error = tffs_close(fdi);
4206 expect(error, EFFS_OK);
4207
4208 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
4209 size = tffs_fread(sname, bigbuf, bigbuf_size);
4210 expect(size, f_size);
4211 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, f_size);
4212 if (error) return 1;
4213
4214 tw(tr(TR_FUNC, TrTestLow, "Make 2 segments, truncate to zero and append\n"));
4215 fdi = tffs_open(sname, FFS_O_WRONLY | FFS_O_CREATE | FFS_O_TRUNC
4216 | FFS_O_APPEND);
4217 expect(fdi, FFS_FD_OFFSET);
4218 size = tffs_write(fdi, (char*) tdata[TDATA_HUGE], 2 * fs.chunk_size_max + 10);
4219 expect(size, 2 * fs.chunk_size_max + 10);
4220
4221 // Move fp otherwise it will not be possible to truncate
4222 error = ffs_seek(fdi, 0, FFS_SEEK_SET);
4223 expect(error, 0);
4224
4225 error = tffs_ftruncate(fdi, 0);
4226 expect(error, EFFS_OK);
4227
4228 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], 150);
4229 expect(size, 150 );
4230
4231 error = tffs_close(fdi);
4232 expect(error, EFFS_OK);
4233
4234 if (p0) { error = test_run("irec;brec"); if (error > 0) return 1; }
4235
4236 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
4237 f_size = tffs_fread(sname, bigbuf, bigbuf_size);
4238 expect(f_size, size);
4239 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, f_size);
4240 if (error) return 1;
4241
4242 // Truncate through a symlink
4243 error = tffs_symlink("/str", sname);
4244 expect(error, EFFS_OK);
4245 error = tffs_truncate("/str", 100);
4246 expect(error, EFFS_OK);
4247 f_size = tffs_fread(sname, bigbuf, bigbuf_size);
4248 expect(f_size, 100);
4249 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, f_size);
4250 if (error) return 1;
4251 tffs_remove("/str");
4252
4253 // truncate a file which only have data in the stream buffer
4254 fdi = tffs_open(sname, FFS_O_WRONLY | FFS_O_CREATE | FFS_O_TRUNC
4255 | FFS_O_APPEND);
4256 expect(fdi, FFS_FD_OFFSET);
4257
4258 f_size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], 3);
4259 expect(f_size, 3);
4260
4261 // Move fp otherwise it will not be possible to truncate
4262 error = ffs_seek(fdi, 0, FFS_SEEK_SET);
4263 expect(error, 0);
4264
4265 tw(tr(TR_FUNC, TrTestLow, "Truncate in buffer\n"));
4266 for (i = 0; i < 4; i++) {
4267 error = tffs_ftruncate(fdi, --f_size);
4268 if (f_size >= 0) {
4269 expect(error, EFFS_OK);
4270
4271 error = tffs_lstat(sname, &stat);
4272 expect(error, EFFS_OK);
4273 expect_eq(stat.size, f_size);
4274 }
4275 else {
4276 expect(error, EFFS_INVALID);
4277 }
4278 }
4279 error = tffs_close(fdi);
4280 expect(error, EFFS_OK);
4281
4282 // Try to truncate a read-only file
4283 error = tffs_fwrite(imeifile, TDATA(0));
4284 // The file is maybe already created!
4285 error = tffs_fcontrol(imeifile, OC_FLAGS, OF_READONLY);
4286 error = tffs_truncate(imeifile, 0);
4287 expect(error, EFFS_ACCESS);
4288
4289 // Use ffs_truncate() on a file open in read mode
4290 fdi = tffs_open(fname, FFS_O_RDONLY);
4291 error = tffs_truncate(fname, 0);
4292 expect(error, EFFS_LOCKED);
4293 error = tffs_ftruncate(fdi, 0);
4294 expect(error, EFFS_INVALID);
4295 error = tffs_close(fdi);
4296 expect_ok(error);
4297
4298 return 0;
4299 }
4300
4301
4302 int case_append(int p0, int p1)
4303 {
4304 fd_t fdi;
4305 int size, fsize;
4306 char sname[] = "/stream/append1";
4307 char fname[] = "/stream/append2";
4308 int offset = 0;
4309
4310 if (param.data_blocks <= 1) {
4311 ttw(ttr(TTrTest, "WARNING: Too few blocks to run. Skip test" NL));
4312 tw(tr(TR_FUNC, TrTest, "WARNING: Too few blocks to run. Skip test\n"));
4313 return 0;
4314 }
4315
4316 case_cleanup(fs.chunk_size_max * 5);
4317
4318 tffs_mkdir("/stream");
4319
4320 fdi = tffs_open(sname, FFS_O_RDWR | FFS_O_APPEND | FFS_O_TRUNC |
4321 FFS_O_CREATE);
4322 expect(fdi, FFS_FD_OFFSET);
4323
4324 // Write data in buf and seek in the buf
4325 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE], fs.chunk_size_max / 2);
4326 expect_gt(size, 0);
4327
4328 offset = ffs_seek(fdi, -fs.chunk_size_max / 4, FFS_SEEK_CUR);
4329 expect_gt(offset, 0);
4330 expect(1, fs.fd[fdi - FFS_FD_OFFSET].dirty);
4331
4332 // Append data and seek in the buffer ad the end of the last chunk
4333 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE] +
4334 fs.fd[fdi - FFS_FD_OFFSET].size, fs.chunk_size_max);
4335 expect_gt(size, 0);
4336
4337 offset = ffs_seek(fdi, -fs.chunk_size_max / 4, FFS_SEEK_END);
4338 expect_gt(offset, 0);
4339
4340 size = tffs_read(fdi, bigbuf, fs.chunk_size_max / 4);
4341 expect(size, fs.chunk_size_max / 4);
4342 expect(1, fs.fd[fdi - FFS_FD_OFFSET].dirty);
4343
4344 // Append data and seek away from buf
4345 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE] +
4346 fs.fd[fdi - FFS_FD_OFFSET].size, fs.chunk_size_max * 1.5);
4347 expect_gt(size, 0);
4348
4349 offset = ffs_seek(fdi, -fs.chunk_size_max * 2, FFS_SEEK_END);
4350 expect_gt(offset, 0);
4351
4352 size = tffs_read(fdi, bigbuf, fs.chunk_size_max / 4);
4353 expect(size, fs.chunk_size_max / 4);
4354
4355 // Append again and validate the data
4356 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE] +
4357 fs.fd[fdi - FFS_FD_OFFSET].size, fs.chunk_size_max / 2);
4358 expect_gt(size, 0);
4359
4360 offset = ffs_seek(fdi, 0, FFS_SEEK_SET);
4361 expect(0, offset);
4362
4363 size = tffs_read(fdi, bigbuf, fs.fd[fdi - FFS_FD_OFFSET].size);
4364 expect_gt(size, 0);
4365
4366 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, size);
4367 if (error) return 1;
4368
4369 error = tffs_close(fdi);
4370 expect_ok(error);
4371
4372 // Append on a file with data in the seghead
4373 fsize = fs.chunk_size_max / 4;
4374 error = tffs_file_write(fname, (char *)tdata[TDATA_HUGE], fsize,
4375 FFS_O_TRUNC | FFS_O_WRONLY | FFS_O_CREATE);
4376 expect(error, EFFS_OK);
4377
4378 fdi = tffs_open(fname, FFS_O_RDWR | FFS_O_APPEND);
4379 expect(fdi, FFS_FD_OFFSET);
4380
4381 // Write data in buf and seek in the buf
4382 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE] + fsize,
4383 fs.chunk_size_max / 2);
4384 expect_gt(size, 0);
4385 fsize += size;
4386
4387 offset = ffs_seek(fdi, 0, FFS_SEEK_SET);
4388 expect(offset, 0);
4389 expect(1, fs.fd[fdi - FFS_FD_OFFSET].dirty);
4390
4391 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE] + fsize,
4392 fs.chunk_size_max / 2);
4393 expect_gt(size, 0);
4394 fsize += size;
4395
4396 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, fsize);
4397 if (error) return 1;
4398
4399 error = tffs_close(fdi);
4400 expect_ok(error);
4401
4402 return 0;
4403 }
4404
4405 int case_datasync(int p0, int p1)
4406 {
4407 fd_t fdi;
4408 int size, fsize, offset, i;
4409 char myname[] = "/datasync";
4410
4411 error = case_cleanup(4 * fs.chunk_size_max);
4412 expect_ok(error);
4413
4414 fdi = tffs_open(myname, FFS_O_RDWR | FFS_O_TRUNC | FFS_O_CREATE);
4415 expect(fdi, FFS_FD_OFFSET);
4416
4417 fsize = 0;
4418 for (i = 0; i < 10; i++) {
4419 // Write data in buf and use datasync
4420 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE] + fsize,
4421 fs.chunk_size_max / 3);
4422 expect_gt(size, 0);
4423 fsize += size;
4424
4425 error = tffs_fdatasync(fdi);
4426 expect(error, EFFS_OK);
4427 expect(0, fs.fd[fdi - FFS_FD_OFFSET].dirty);
4428 }
4429
4430 error = tffs_fdatasync(fdi + 1);
4431 expect(error, EFFS_BADFD);
4432
4433 // Seek to chunk 2, write data, seek, use fdatasync
4434 offset = tffs_seek(fdi, fs.chunk_size_max * 2.5, FFS_SEEK_SET);
4435 expect_gt(offset, 0);
4436
4437 size = tffs_write(fdi, (char *) tdata[TDATA_HUGE] + offset,
4438 fs.chunk_size_max / 4);
4439 expect_gt(size, 0);
4440
4441 offset = tffs_seek(fdi, -fs.chunk_size_max / 2, FFS_SEEK_CUR);
4442 expect_gt(offset, 0);
4443
4444 error = tffs_fdatasync(fdi);
4445 expect(error, EFFS_OK);
4446 expect(0, fs.fd[fdi - FFS_FD_OFFSET].dirty);
4447
4448 error = tffs_close(fdi);
4449 expect_ok(error);
4450
4451 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
4452 size = tffs_fread(myname, bigbuf, fsize);
4453 tw(tr(TR_FUNC, TrTestLow, "size: %d fsize: %d \n", size, fsize));
4454 expect(size, fsize);
4455 error = test_expect_data((char *)tdata[TDATA_HUGE], bigbuf, size);
4456 if (error) return 1;
4457
4458 return 0;
4459 }
4460
4461 int case_fw_flags(int p0, int p1)
4462 {
4463 int i;
4464 char myname[] = "/file_write";
4465
4466 tffs_remove(myname);
4467
4468 tw(tr(TR_FUNC, TrTestHigh, "Test ffs_file_write() flags, file non-exist\n"));
4469 for (i = 0; i < 0xFE; i++) {
4470 error = tffs_file_write(myname, 0,0, i);
4471
4472 if (is_open_option(i, FFS_O_CREATE)) {
4473 expect_ok(error);
4474 error = tffs_remove(myname);
4475 expect_ok(error);
4476 }
4477 else
4478 expect(EFFS_NOTFOUND, error);
4479 }
4480
4481 tw(tr(TR_FUNC, TrTestHigh, "Test ffs_file_write() flags, file exist\n"));
4482 error = tffs_file_write(myname, 0,0, FFS_O_CREATE);
4483 expect_ok(error);
4484
4485 for (i = 0; i < 0xFF; i++) {
4486 error = tffs_file_write(myname, 0,0, i);
4487
4488 if (is_open_option(i, FFS_O_EXCL) && is_open_option(i, FFS_O_CREATE)) {
4489 expect(error, EFFS_EXISTS);
4490 }
4491
4492 else
4493 expect_ok(error);
4494 }
4495
4496 return 0;
4497 }
4498
4499
4500 drv_Return_Type pcm_init(void);
4501
4502 int case_pcm(int p0, int p1)
4503 {
4504 UBYTE version;
4505 UBYTE imei[] = "12345678";
4506 int size;
4507
4508 ffs_mkdir("/pcm");
4509
4510 // Write imei file to ffs. Run init (this reads the imei file), remove
4511 // the file from flash, write the imei to flash from pcm and validate
4512 // data.
4513 error = ffs_file_write("/pcm/IMEI", &imei, sizeof(imei), FFS_O_CREATE);
4514 expect_ok(error);
4515
4516 error = pcm_init();
4517 expect(PCM_INITIALIZED, error);
4518
4519 error = pcm_ReadFile((UBYTE *) EF_IMEI_ID, SIZE_EF_IMEI, imei, &version);
4520 expect(PCM_OK, error);
4521
4522 error = ffs_remove("/pcm/IMEI");
4523 expect_ok(error);
4524
4525 error = pcm_WriteFile((UBYTE *) EF_IMEI_ID, SIZE_EF_IMEI, imei);
4526 expect(PCM_OK, error);
4527
4528 tw(tr(TR_FUNC, TrTestLow, "Validate data \n"));
4529 size = tffs_fread("/pcm/IMEI", bigbuf, bigbuf_size);
4530 expect(size, SIZE_EF_IMEI);
4531 error = test_expect_data(imei, bigbuf, size);
4532 if (error) return 1;
4533
4534 // TODO: test pcm_WriteRecord and pcm_ReadRecord.
4535 return 0;
4536 }
4537
4538
4539 int case_api_exceptions(int p0, int p1)
4540 {
4541 int i;
4542 T_FFS_DIR dir;
4543 T_FFS_FD myfdi;
4544 char myname[] = "/BAD";
4545 char mydir[] = "/tdir";
4546 char mylink[] = "/tlink";
4547 char mytfile[] = "/tfile";
4548 char buf[] = "foobar";
4549
4550 // Create test file and dir
4551 error = tffs_file_write(mytfile, buf, sizeof(buf), FFS_O_CREATE);
4552 expect_ok(error);
4553
4554 error = tffs_mkdir(mydir);
4555 if (error < 0 && error != EFFS_EXISTS) return 1;
4556
4557 error = tffs_symlink(mytfile, mylink);
4558 if (error < 0 && error != EFFS_EXISTS) return 1;
4559
4560 error = tffs_opendir(mydir, &dir);
4561 expect_ok(error);
4562
4563 myfdi = tffs_open("/tsream", FFS_O_WRONLY | FFS_O_CREATE | FFS_O_TRUNC);
4564 expect(myfdi, FFS_FD_OFFSET);
4565
4566 expect(tffs_write(myfdi, buf, sizeof(buf)), sizeof(buf));
4567
4568 // Test all API functions for pathname with null pointers.
4569 for (i = 0; i < 17; i++) {
4570 switch (i) {
4571 case 0: error = tffs_open(0, FFS_O_WRONLY | FFS_O_CREATE); break;
4572 case 1: error = tffs_truncate(0, 1); break;
4573 case 2: error = tffs_file_write(0, buf, sizeof(buf), FFS_O_CREATE); break;
4574 case 3: error = tffs_stat(0, &stat); break;
4575 case 4: error = tffs_lstat(0, &stat); break;
4576 case 5: error = tffs_xlstat(0, &xstat); break;
4577 case 6: error = tffs_remove(0); break;
4578 case 7: error = tffs_mkdir(0); break;
4579 case 8: error = tffs_opendir(0, &dir); break;
4580 case 9: error = tffs_symlink(0, myname); break;
4581 case 10: error = tffs_symlink("/link", 0); break;
4582 case 11: error = tffs_readlink(0, buf, sizeof(buf)); break;
4583 case 12: error = tffs_rename(0, "/test"); break;
4584 case 13: error = tffs_rename(mytfile, 0); break;
4585 case 14: error = tffs_file_write(0, buf, sizeof(buf), FFS_O_CREATE); break;
4586 case 15: error = tffs_file_read(0, buf, sizeof(buf)); break;
4587 case 16: error = tffs_fcontrol(0, OC_FLAGS, OF_READONLY); break;
4588 default: error = EFFS_INVALID;
4589 }
4590 expect(error, EFFS_BADNAME);
4591 }
4592
4593 // Test all API functions with buffer null pointers.
4594 for (i = 0; i < 13; i++) {
4595 switch (i) {
4596 case 0: error = tffs_file_write(myname, 0, 100, FFS_O_CREATE); break;
4597 case 1: error = tffs_file_read(mytfile, 0, 40); break;
4598 case 2: error = tffs_opendir(mydir, 0); break;
4599 case 3: error = ffs_readdir (0, buf, sizeof(buf)); break;
4600 case 4: error = ffs_readdir (&dir, 0, 100); break;
4601 case 5: error = tffs_readlink(mylink, 0, 1); break;
4602 case 6: error = tffs_stat(mytfile, 0); break;
4603 case 7: error = tffs_lstat(mylink, 0); break;
4604 case 8: error = tffs_xlstat(mylink, 0); break;
4605 case 9: error = tffs_query(Q_BYTES_FREE, 0); break;
4606 case 10: error = tffs_fstat(myfdi, 0); break;
4607 case 11: error = tffs_write(myfdi, 0, 10); break;
4608 case 12: error = tffs_read(myfdi, 0, 40); break;
4609 default: error = 0;
4610 }
4611 expect(error, EFFS_INVALID);
4612 }
4613 expect_ok(tffs_close(myfdi));
4614
4615 // Make sure we still can create empty files!
4616 error = tffs_file_write("/empty_file", 0, 0, FFS_O_CREATE);
4617 expect_ok(error);
4618
4619 // Open file for test of negative size
4620 myfdi = tffs_open(myname, FFS_O_RDWR | FFS_O_CREATE |
4621 FFS_O_TRUNC | FFS_O_APPEND);
4622 expect(myfdi, FFS_FD_OFFSET);
4623
4624 // Test all API 'size' parameters with a negative size
4625 for (i = 0; i < 11; i++) {
4626 switch (i) {
4627 // Try modify
4628 case 0: error = tffs_write(myfdi, buf, -1); break;
4629 case 1: error = tffs_truncate(myname, -1); break;
4630 case 2: error = tffs_fwrite(myname, buf, -2); break;
4631 case 3: error = tffs_fcreate(myname, buf, -3); break;
4632 case 4: error = tffs_fupdate(myname, buf, -56); break;
4633 case 5: error = tffs_file_write(myname, buf, -100, FFS_O_CREATE); break;
4634 // Try read
4635 case 6: error = tffs_fread(mytfile, bigbuf, -1); break;
4636 case 7: error = tffs_file_read(mytfile, bigbuf, -2); break;
4637 case 8: error = tffs_readdir(&dir, bigbuf, -3); break;
4638 case 9: error = tffs_readlink(mytfile, bigbuf, -4); break;
4639 case 10: error = tffs_read(myfdi, bigbuf, -5); break;
4640 default: error = 0;
4641 }
4642 expect(error, EFFS_INVALID);
4643 }
4644
4645 error = tffs_close(myfdi);
4646 expect_ok(error);
4647
4648 return EFFS_OK;
4649 }
4650
4651 int case_api_notformated(int p0, int p1)
4652 {
4653 int i;
4654 struct dir_s dir;
4655 fd_t fdi;
4656 char buf[20];
4657 uint16 q_out;
4658 struct xstat_s xstat;
4659
4660 error = tffs_preformat(0xDEAD);
4661 expect(error, EFFS_OK);
4662
4663 for(i=1; i<24; i++)
4664 {
4665 switch(i)
4666 {
4667 case 1: error = tffs_open("apinf", FFS_O_WRONLY); break;
4668 case 2: error = tffs_close(fdi); break;
4669 case 3: error = tffs_write(fdi, "abcdefghij", 10); break;
4670 case 4: error = tffs_read(fdi, buf, 2); break;
4671 case 5: error = tffs_seek(fdi, 0, FFS_SEEK_SET); break;
4672 case 6: error = tffs_truncate("apinf", 0); break;
4673 case 7: error = tffs_fdatasync(fdi); break;
4674 case 8: error = tffs_stat("apinf", &stat); break;
4675 case 9: error = tffs_fstat(fdi, &stat); break;
4676 case 10: error = tffs_lstat("apinf", &stat); break;
4677 case 11: error = tffs_xlstat("apinf", &xstat); break;
4678 case 12: error = ffs_remove("/apinf"); break;
4679 case 13: error = ffs_mkdir("/apinf"); break;
4680 case 14: error = ffs_opendir("/apinf", &dir); break;
4681 case 15: error = tffs_readdir(&dir, "/ffs", 21); break;
4682 case 16: error = tffs_symlink("/europe/imie", "imei"); break;
4683 case 17: error = ffs_rename("/Pluto", "/Dog"); break;
4684 case 18: error = ffs_file_write("/non-exist", "1234567890", 10, FFS_O_TRUNC); break;
4685 case 19: error = tffs_file_read("/stream1", buf, 10); break;
4686 case 20: error = tffs_fcreate("/", "apinf", 3); break;
4687 case 21: error = tffs_fupdate("/", "bar", 3); break;
4688 case 22: error = tffs_fwrite("/europe/imie","0123456789", 10); break;
4689 case 23: error = tffs_fcontrol("/apinf", OC_FLAGS, OF_READONLY); break;
4690 }
4691
4692 expect(error, EFFS_NOFORMAT);
4693
4694 }
4695 /* query must return information even if the device isnt formated */
4696 error = tffs_query(Q_FS_INODES, (uint16 *) &q_out);
4697 expect(error, EFFS_OK);
4698
4699 return 0;
4700 }
4701
4702 int case_query(int p0, int p1)
4703 {
4704 int i;
4705 uint32 ret_val32;
4706 uint16 ret_val16;
4707
4708 for(i=1; i<27; i++)
4709 {
4710 switch(i)
4711 {
4712 case 1: error = ffs_query(Q_TM_BUFADDR, (uint32 *)&ret_val32); break;
4713 case 2: error = ffs_query(Q_TM_BUFSIZE, (uint32 *)&ret_val32); break;
4714 case 3: error = ffs_query(Q_DEV_BASE, (uint32 *)&ret_val32); break;
4715 case 4: error = ffs_query(Q_FFS_API_VERSION, (uint16 *)&ret_val16); break;
4716 case 5: error = ffs_query(Q_FFS_DRV_VERSION, (uint16 *)&ret_val16); break;
4717 case 6: error = ffs_query(Q_FFS_REVISION, (uint16 *)&ret_val16); break;
4718 case 7: error = ffs_query(Q_FFS_FORMAT_WRITE, (uint16 *)&ret_val16); break;
4719 case 8: error = ffs_query(Q_FFS_FORMAT_READ, (uint16 *)&ret_val16); break;
4720 case 9: error = ffs_query(Q_FFS_LASTERROR, (uint16 *)&ret_val16); break;
4721 case 10: error = ffs_query(Q_FFS_TM_VERSION, (uint16 *)&ret_val16); break;
4722 case 11: error = ffs_query(Q_OBJECTS_MAX, (uint16 *)&ret_val16); break;
4723 case 12: error = ffs_query(Q_CHUNK_SIZE_MAX, (uint16 *)&ret_val16); break;
4724 case 13: error = ffs_query(Q_FD_BUF_SIZE, (uint32 *)&ret_val32); break;
4725 case 14: error = ffs_query(Q_FD_MAX, (uint16 *)&ret_val16); break;
4726 case 15: error = ffs_query(Q_DEV_MANUFACTURER, (uint16 *)&ret_val16); break;
4727 case 16: error = ffs_query(Q_DEV_DEVICE, (uint16 *)&ret_val16); break;
4728 case 17: error = ffs_query(Q_DEV_DRIVER, (uint16 *)&ret_val16); break;
4729 case 18: error = ffs_query(Q_LOST_HIGH, (uint16 *)&ret_val16); break;
4730 case 19: error = ffs_query(Q_FS_FLAGS, (uint16 *)&ret_val16); break;
4731 case 20: error = ffs_query(Q_FS_ROOT, (uint16 *)&ret_val16); break;
4732 case 21: error = ffs_query(Q_STATS_DRECLAIMS, (uint32 *)&ret_val32); break;
4733 case 22: error = ffs_query(Q_STATS_IRECLAIMS, (uint32 *)&ret_val32); break;
4734 case 23: error = ffs_query(Q_STATS_DATA_RECLAIMED, (uint32 *)&ret_val32); break;
4735 case 24: error = ffs_query(Q_STATS_INODES_RECLAIMED, (uint32 *)&ret_val32); break;
4736 case 25: error = ffs_query(Q_STATS_DATA_ALLOCATED, (uint32 *)&ret_val32); break;
4737 case 26: error = ffs_query(Q_REQUEST_ID_LAST, (uint32 *)&ret_val32); break;
4738 }
4739
4740 expect(error, EFFS_OK);
4741
4742 }
4743
4744 return 0;
4745 }
4746
4747 extern uint16 ffs_flash_device;
4748 extern uint16 ffs_flash_manufact;
4749
4750 int case_octrl(int p0, int p1)
4751 {
4752 int org_fsflags = fs.flags, org_manufact = ffs_flash_manufact;
4753 int org_device = ffs_flash_device, org_testf = fs.testflags;
4754
4755 error = object_control(0, OC_FS_FLAGS, 0x02);
4756 expect(error, EFFS_OK);
4757 error = object_control(0, OC_DEV_MANUFACT, 'T');
4758 expect(error, EFFS_OK);
4759 error = object_control(0, OC_DEV_DEVICE, 0x0F12);
4760 expect(error, EFFS_OK);
4761 error = object_control(0, OC_DEBUG_0, 0xAFFE);
4762 expect(error, EFFS_OK);
4763
4764 error = object_control(0, OC_DEBUG_LAST+1, 0);
4765 expect(error, EFFS_INVALID);
4766
4767 // Restore the values or else it mess up the other tests
4768 fs.flags = org_fsflags;
4769 fs.testflags = org_testf;
4770 ffs_flash_manufact = org_manufact;
4771 ffs_flash_device = org_device;
4772
4773 return 0;
4774 }
4775
4776 /***************************************************************************/
4777 /* Examples from RIV111 */
4778 /***************************************************************************/
4779 #if (TARGET == 0)
4780 int case_examples(int p0, int p1)
4781 {
4782 return 0;
4783 }
4784 #else
4785
4786 static req_id_t my_request_id;
4787 static int removed_greeting;
4788
4789 static void my_callback(T_FFS_FILE_CNF *confirm)
4790 {
4791 ttw(ttr(TTrTest, "my_callback(%d, %s, %d)" NL,
4792 confirm->error, confirm->path, confirm->request_id));
4793
4794 if (my_request_id != confirm->request_id) {
4795 ttw(ttr(TTrFatal, "error (%d, %d)" NL, confirm->request_id, my_request_id));
4796 return;
4797 }
4798
4799 if (confirm->error < 0) {
4800 ttw(ttr(TTrTest, "FATAL: my_callback error: %d" NL, confirm->error));
4801 }
4802 else {
4803 ttw(ttr(TTrTest, "Remove 'greeting'" NL));
4804 ffs_remove_nb("/greeting", 0);
4805 removed_greeting = 1;
4806 }
4807
4808 error = rvf_free_buf(confirm);
4809 if (error < 0)
4810 ttr(TTrFatal, "FFS FATAL: os_free() %d" NL, error);
4811 }
4812
4813 // Using Function Callback
4814 T_RV_RETURN my_fc_cnfpath = {0, (CALLBACK_FUNC)my_callback};
4815
4816 // Using Mail Callback
4817 #define MY_TASK_ID (MAX_RVF_TASKS - 1 - 1) // same as FFS_TEST_TASK_ID
4818 T_RV_RETURN my_mail_cnfpath = {MY_TASK_ID, 0};
4819
4820 int case_examples(int p0, int p1)
4821 {
4822 if(1) {
4823 // MMI Configuration scenario
4824 struct {
4825 int volume;
4826 uint32 ringtype;
4827 } mmi;
4828
4829 int temp;
4830 ttw(ttr(TTrTest, "Run MMI conf." NL));
4831 mmi.volume = temp = 3;
4832 mmi.ringtype = 1111;
4833 ffs_mkdir("/mmi");
4834 ffs_fwrite("/mmi/volume", &mmi.volume, sizeof(mmi.volume));
4835 error = ffs_file_read("/mmi/volume", &mmi.volume, sizeof(mmi.volume));
4836 if (error < 0)
4837 return 1;
4838 error = test_expect_data(&temp, &mmi.volume, sizeof(mmi.volume));
4839 if (error < 0)
4840 return 1;
4841 }
4842
4843 if(1) {
4844 // Using Mail Callback
4845 T_FFS_FILE_CNF *confirm_file;
4846 T_FFS_STREAM_CNF *confirm_stream;
4847 T_FFS_RET error;
4848 T_FFS_FD fdi;
4849 char hello[] = "Hello, world";
4850 T_FFS_SIZE size;
4851
4852 ttw(ttr(TTrTest, "Use Mail Callback" NL));
4853 if ((my_request_id = ffs_open_nb("/mmi/sound",
4854 FFS_O_WRONLY | FFS_O_CREATE |
4855 FFS_O_APPEND | FFS_O_TRUNC,
4856 &my_mail_cnfpath)) < 0)
4857 return my_request_id;
4858
4859 ttw(ttr(TTrTest, "Wait..." NL));
4860 rvf_wait(RVF_TASK_MBOX_0_EVT_MASK, 0);
4861
4862 ttw(ttr(TTrTest, "Get mail" NL));
4863 confirm_file = (T_FFS_FILE_CNF *) rvf_read_mbox(RVF_TASK_MBOX_0);
4864
4865 fdi = confirm_file->error;
4866 expect(FFS_FD_OFFSET, fdi);
4867 expect(my_request_id, confirm_file->request_id);
4868 expect(FFS_MESSAGE_OFFSET, confirm_file->header.msg_id);
4869 ttw(ttr(TTrTest, "Path '%s'" NL, confirm_file->path));
4870
4871 error = rvf_free_buf(confirm_file);
4872 if (error < 0)
4873 ttr(TTrFatal, "FFS FATAL: os_free() %d" NL, error);
4874
4875 if ((my_request_id = ffs_write_nb(fdi, hello, strlen(hello) ,
4876 &my_mail_cnfpath)) < 0)
4877 return my_request_id;
4878
4879 ttw(ttr(TTrTest, "Wait..." NL));
4880 rvf_wait(RVF_TASK_MBOX_0_EVT_MASK, 0);
4881
4882 ttw(ttr(TTrTest, "Get mail" NL));
4883 confirm_stream = (T_FFS_STREAM_CNF *) rvf_read_mbox(RVF_TASK_MBOX_0);
4884 size = confirm_stream->error;
4885 expect(strlen(hello), size);
4886 expect(my_request_id, confirm_stream->request_id);
4887 ttw(ttr(TTrTest, "fdi: %d" NL, confirm_stream->fdi));
4888 error = rvf_free_buf(confirm_stream);
4889 if (error < 0)
4890 ttr(TTrFatal, "FFS FATAL: os_free() %d" NL, error);
4891
4892 ttw(ttr(TTrTest, "Close file" NL));
4893 if ((error = ffs_close(fdi)) < 0)
4894 return error;
4895 }
4896
4897 if(1) {
4898 // Using Function Callback
4899 char hello[] = "Hello, world";
4900 T_FFS_STAT stat;
4901
4902 ttw(ttr(TTrTest, "Use Function Callback" NL));
4903
4904 if ((my_request_id = ffs_fcreate_nb("/greeting", hello, strlen(hello),
4905 &my_fc_cnfpath)) < 0)
4906 return my_request_id;
4907
4908 // Wait until my_callback() has removed the file "greeting"
4909 removed_greeting = 0;
4910 while (removed_greeting == 0)
4911 tffs_delay(10);
4912 // NOTE: This is importent because if the test brec runs immediately
4913 // after his one will the brec test fail because the "greeting" file
4914 // is removed inside the brec test which don't expect that files is
4915 // removed.
4916 }
4917
4918 return 0;
4919 }
4920 #endif
4921 /******************************************************************************
4922 * Non blocking test ffs_xx_nb()
4923 ******************************************************************************/
4924
4925 #if (TARGET == 0)
4926 int case_nonblock(int p0, int p1)
4927 {
4928 tw(tr(TR_FUNC, TrTest,
4929 "WARNING: This test can only run in target. Skip test\n"));
4930 return 0;
4931 }
4932
4933 #else
4934 extern int request_id_last; // from task.c
4935
4936 #define WAIT_GET_TEST ttw(ttr(TTrTest, "Wait..." NL)); rvf_wait(RVF_TASK_MBOX_0_EVT_MASK, 0); ttw(ttr(TTrTest, "Get mail" NL)); confirm_file = (T_FFS_FILE_CNF *) rvf_read_mbox(RVF_TASK_MBOX_0); ttw(ttr(TTrTest, "Path '%s'" NL, confirm_file->path)); expect(FFS_MESSAGE_OFFSET, confirm_file->header.msg_id); expect(my_id, confirm_file->request_id);
4937
4938 #define WAIT_GET_TEST_FDI ttw(ttr(TTrTest, "Wait..." NL)); rvf_wait(RVF_TASK_MBOX_0_EVT_MASK, 0); confirm_stream = (T_FFS_STREAM_CNF *) rvf_read_mbox(RVF_TASK_MBOX_0); ttw(ttr(TTrTest, "Get mail" NL)); ttw(ttr(TTrTest, "fdi: %d" NL, confirm_stream->fdi)); expect(FFS_MESSAGE_OFFSET, confirm_stream->header.msg_id); expect(FFS_FD_OFFSET, confirm_stream->fdi); expect(my_id, confirm_stream->request_id);
4939
4940 #define FREE error = rvf_free_buf(confirm_file); if (error < 0) { ttr(TTrFatal, "FFS FATAL: os_free() %d" NL, error); return 1;}
4941
4942 #define FREE_FDI error = rvf_free_buf(confirm_stream); if (error < 0) { ttr(TTrFatal, "FFS FATAL: os_free() %d" NL, error); return 1;}
4943
4944 int case_nonblock(int p0, int p1)
4945 {
4946 T_FFS_FILE_CNF *confirm_file;
4947 T_FFS_STREAM_CNF *confirm_stream;
4948 T_FFS_RET error;
4949 T_FFS_FD fdi;
4950 char hello[] = "Hello, world";
4951 char myfname[] = "/non_block/file";
4952 char mysname[] = "/non_block/stream";
4953 T_FFS_SIZE size;
4954 int i;
4955 T_FFS_REQ_ID my_id;
4956 // mail confirm
4957 T_RV_RETURN cnfpath = {MY_TASK_ID, 0};
4958
4959 // Test wrap arround in request_id_get(). Set id to 5 from max, the following
4960 // function calls will trigger the wrap arround
4961 request_id_last = 0x7FFFFFFF - 5;
4962 ttw(ttr(TTrTest, "request :%d" NL, request_id_last));
4963
4964 // preformat, format, mkdir
4965 // open, write, ftruncate, close, remove,
4966 // file_write, symlink, rename, truncate, fcontrol
4967
4968 my_id = ffs_preformat_nb(0xDEAD, &cnfpath);
4969 expect_gt(my_id, -1);
4970 WAIT_GET_TEST; FREE;
4971
4972 my_id = ffs_format_nb("/ffs", 0x2BAD, &cnfpath);
4973 expect_gt(my_id, -1);
4974 WAIT_GET_TEST; FREE;
4975
4976 my_id = ffs_mkdir_nb("/non_block", &cnfpath);
4977 expect_gt(my_id, -1);
4978 WAIT_GET_TEST; FREE;
4979
4980 if ((my_id = ffs_open_nb(mysname, FFS_O_WRONLY | FFS_O_CREATE,
4981 &cnfpath)) < 0)
4982 return my_id;
4983 WAIT_GET_TEST;
4984 fdi = confirm_file->error;
4985 expect(FFS_FD_OFFSET, fdi);
4986 FREE;
4987
4988 if ((my_id = ffs_write_nb(fdi, hello, strlen(hello), &cnfpath)) < 0)
4989 return my_id;
4990 WAIT_GET_TEST_FDI; FREE_FDI;
4991
4992 expect(1, fs.fd[fdi - FFS_FD_OFFSET].dirty);
4993 if ((my_id = ffs_fdatasync_nb(fdi, &cnfpath)) < 0)
4994 return my_id;
4995 WAIT_GET_TEST_FDI; FREE_FDI;
4996 expect(0, fs.fd[fdi - FFS_FD_OFFSET].dirty);
4997
4998 if ((my_id = ffs_seek_nb(fdi, 2, FFS_SEEK_SET, &cnfpath)) < 0)
4999 return my_id;
5000 WAIT_GET_TEST_FDI; FREE_FDI;
5001 expect(2, fs.fd[fdi - FFS_FD_OFFSET].fp);
5002
5003 if ((my_id = ffs_ftruncate_nb(fdi, 3, &cnfpath)) < 0)
5004 return my_id;
5005 WAIT_GET_TEST_FDI;
5006 expect(EFFS_OK, confirm_stream->error);
5007 FREE_FDI;
5008
5009 if ((my_id = ffs_close_nb(fdi, &cnfpath)) < 0)
5010 return my_id;
5011 WAIT_GET_TEST_FDI;
5012 expect(EFFS_OK, confirm_stream->error);
5013 FREE_FDI;
5014
5015 if ((my_id = ffs_remove_nb(mysname, &cnfpath)) < 0)
5016 return my_id;
5017 WAIT_GET_TEST;
5018 expect(EFFS_OK, confirm_file->error);
5019 FREE;
5020
5021 if ((my_id = ffs_file_write_nb(myfname, hello, strlen(hello),
5022 FFS_O_WRONLY | FFS_O_CREATE, &cnfpath)) < 0)
5023 return my_id;
5024 WAIT_GET_TEST;
5025 expect(EFFS_OK, confirm_file->error);
5026 FREE;
5027
5028 if ((my_id = ffs_symlink_nb("/nb_file", myfname, &cnfpath)) < 0)
5029 return my_id;
5030 WAIT_GET_TEST;
5031 expect(EFFS_OK, confirm_file->error);
5032 FREE;
5033
5034 if ((my_id = ffs_rename_nb("/nb_file", "/nbf", &cnfpath)) < 0)
5035 return my_id;
5036 WAIT_GET_TEST;
5037 expect(EFFS_OK, confirm_file->error);
5038 FREE;
5039
5040 if ((my_id = ffs_truncate_nb(myfname, 0, &cnfpath)) < 0)
5041 return my_id;
5042 WAIT_GET_TEST;
5043 expect(EFFS_OK, confirm_file->error);
5044 FREE;
5045
5046 if ((my_id = ffs_fcontrol_nb(myfname, OC_FLAGS, OF_READONLY, &cnfpath)) < 0)
5047 return my_id;
5048 WAIT_GET_TEST;
5049 expect(EFFS_OK, confirm_file->error);
5050 FREE;
5051
5052 ttw(ttr(TTrTest, "request :%d" NL, request_id_last));
5053
5054 // This should make a EFFS_MEMORY fail
5055 for(i = 0; i < 10000; i++) {
5056 my_id = ffs_fwrite_nb("/mem_test", TDATA(0), 0);
5057 if (my_id < 0)
5058 break;
5059 }
5060 ttw(ttr(TTrTest, "write nr: %d my_id: %d" NL, i, my_id));
5061 expect(my_id, EFFS_MEMORY);
5062 // All memory has been used so wait a while (let some of the writes finish)
5063 tffs_delay(50);
5064 // Make a call to a blocking function will make it possible for ffs to
5065 // finish all the non blocking writes.
5066 error = ffs_fwrite("/mem_test", TDATA(0));
5067 expect_ok(error);
5068
5069 return 0;
5070 }
5071 #endif
5072
5073 /******************************************************************************
5074 * Blocking test
5075 ******************************************************************************/
5076
5077 #if (TARGET == 0)
5078 int case_blocking(int p0, int p1)
5079 {
5080 tw(tr(TR_FUNC, TrTest,
5081 "WARNING: This test can only run in target. Skip test\n"));
5082 return 0;
5083 }
5084 #else
5085
5086 #include "ffs/board/task.h"
5087 void my_mb_callback(T_RVF_MB_ID id)
5088 { // Dummy callback function
5089 }
5090
5091
5092 // Test that a FFS API blocking call that not is able to send a mail to FFS fails as expected and don't lock the system (it must unlock and delete the mutex).
5093 int case_blocking(int p0, int p1)
5094 {
5095 UINT32 mem_unused;
5096 T_RVF_MB_STATUS status;
5097 char *buf;
5098 extern T_OS_MB_ID ffs_mb_id;
5099
5100 // Later when all the memory is used the memory bank will be marked as
5101 // RED by Riviera. The RED flag is erased when the buffer is freed but
5102 // only if a callback function is applied.
5103 rvf_set_callback_func(ffs_mb_id, my_mb_callback); // Ignore error
5104
5105 mem_unused = rvf_get_mb_unused_mem(ffs_mb_id);
5106 ttw(ttr(TTrTest,"Unused memory: %d" NL, mem_unused));
5107
5108 // When all the memory is used the blocking functions will not be able
5109 // to allocate memory to the FFS mail thus it must fail. After the
5110 // buffer has been freed the block call must again be fully functional.
5111 if ((buf = (char *) target_malloc(mem_unused)) == 0)
5112 return EFFS_MEMORY;
5113
5114 error = tffs_fwrite("/blocking_test", (char*)tdata[TDATA_HUGE], 20);
5115
5116 target_free(buf);
5117
5118 expect(error, EFFS_MEMORY);
5119
5120 error = tffs_fwrite("/blocking_test", (char*)tdata[TDATA_HUGE], 20);
5121 expect_ok(error);
5122
5123 return EFFS_OK;
5124 }
5125 #endif
5126
5127 /******************************************************************************
5128 * Test erase_suspend and erase_resume
5129 ******************************************************************************
5130 * Test erase suspend and resume by making a call to ffs_begin() followed by
5131 * an ffs_read() while we are erasing. We trigger the erase by sending
5132 * several write mails to FFS. We let FFS get the CPU for less time than an
5133 * erase, which will guarantee that we interrupt the erase when we call
5134 * ffs_begin().
5135 ******************************************************************************/
5136 #if (TARGET == 1)
5137 static int write_mails_left;
5138
5139 static void driver_callback(T_FFS_FILE_CNF *confirm)
5140 {
5141 if (confirm->error < 0) {
5142 ttw(ttr(TTrTest, "FATAL: driver_callback error: %d" NL, confirm->error));
5143 }
5144
5145 error = rvf_free_buf(confirm);
5146 if (error < 0)
5147 ttr(TTrFatal, "FFS FATAL: os_free() %d" NL, error);
5148
5149 write_mails_left--;
5150 }
5151 #endif
5152
5153 #if (TARGET == 0)
5154 int case_erase_suspend(int p0, int p1)
5155 {
5156 tw(tr(TR_FUNC, TrTestHigh, "Erase_suspend test not supported on PC sim\n"));
5157 return 1;
5158 }
5159 #else
5160 // TEST erase_suspend and erase_resume
5161 int case_erase_suspend(int p0, int p1)
5162 {
5163 int i, j, error, nsuspend;
5164 char myname[] = "/ffs/b4"; // Run test with only 4 block.
5165 char myfname[] = "/driver_test";
5166 int file_size = 1024;
5167
5168 T_RV_RETURN cnfpath = {0, (CALLBACK_FUNC)driver_callback};
5169
5170 ttw(ttr(TTrTest, "Erase suspend-resume test" NL));
5171
5172 // To minimize the numbers of required writes before we are guarantied an
5173 // erase we run this test in only 4 blocks
5174 error = tffs_preformat(0xDEAD);
5175 expect(error, EFFS_OK);
5176 error = tffs_format(myname, 0x2BAD);
5177 expect(error, EFFS_OK);
5178
5179 if (p0 == 0) p0 = 1;
5180 for (j = 0; j < p0; j++) {
5181 write_mails_left = 0;
5182 // Write 2 blocks of data (must trigger an erase)
5183 for (i = 0; i < 2 * (dev.blocksize/file_size); i++)
5184 {
5185 // Non blockin write
5186 if ((error = ffs_file_write_nb(myfname, (char *) tdata[TDATA_HUGE],
5187 file_size,
5188 FFS_O_WRONLY | FFS_O_CREATE | FFS_O_TRUNC,
5189 &cnfpath)) < 0)
5190 return error;
5191 write_mails_left++;
5192 }
5193 ttw(ttr(TTrTest, "Finish write calls(%d)" NL, i));
5194
5195 // NOTE only tested with AMD MB driver
5196 switch (dev.driver) {
5197 case FFS_DRIVER_AMD:
5198 case FFS_DRIVER_SST:
5199 case FFS_DRIVER_INTEL:
5200 case FFS_DRIVER_AMD_SB:
5201 ttw(ttr(TTrTest, "Start read test" NL));
5202 nsuspend = 0;
5203 do {
5204 // NOTEME: We can also test if we get the CPU and not is delay
5205 // by the erase
5206
5207 // This will change state to ERASE_SUSPEND if we are in
5208 // ERASE mode (only valid for MB driver)
5209 error = ffs_begin();
5210 expect(error, EFFS_OK);
5211
5212 if (dev.state == DEV_ERASE_SUSPEND) {
5213 nsuspend++;
5214 // Read... Note it will also resume the erase (MB driver).
5215 error = tffs_file_read(myfname, bigbuf, file_size/4);
5216 expect(error, EFFS_FILETOOBIG);
5217 }
5218 rvf_delay(5); // Let FFS get the CPU for some time
5219 } while (write_mails_left > 0);
5220
5221 if (nsuspend <= 0) {
5222 ttw(ttr(TTrTest, "Arg no erase suspend" NL));
5223 return 1;
5224 }
5225 ttw(ttr(TTrTest, "Erase suspended %d times" NL, nsuspend));
5226 break;
5227
5228 case FFS_DRIVER_SST_SB:
5229 case FFS_DRIVER_INTEL_SB:
5230 default:
5231 ttw(ttr(TTrTest, "Driver not supported by this test" NL));
5232 // Do not return before the cnfpath resource not is used anymore
5233 while (write_mails_left > 0) rvf_delay(100);
5234 return 1;
5235 }
5236 }
5237 return 0;
5238 }
5239 #endif // TARGET == 0
5240
5241 /******************************************************************************
5242 * The 3 below testh_.. function is helper function used for the low level
5243 * driver tests.
5244 ******************************************************************************/
5245 #if (TARGET == 1)
5246
5247 uint32 int_disable(void);
5248 void int_enable(uint32 tmp);
5249
5250 // Only start an erase, don't wait until it is finish
5251 // Note to use this function with SB drivers require a multi bank flash device
5252 int testh_nb_erase(uint8 block)
5253 {
5254 volatile char *flash = dev.base;
5255 volatile char *addr;
5256 uint32 cpsr;
5257
5258 ttw(ttr(TTrTest, "e(%d)" NL, block));
5259
5260 addr = block2addr(block);
5261 cpsr = int_disable();
5262 switch (dev.driver) {
5263 case FFS_DRIVER_AMD:
5264 case FFS_DRIVER_AMD_SB:
5265 case FFS_DRIVER_AMD_PSEUDO_SB:
5266 flash[0xAAAA] = 0xAA; // AMD unlock cycle 1
5267 flash[0x5555] = 0x55; // AMD unlock cycle 2
5268 flash[0xAAAA] = 0x80;
5269 flash[0xAAAA] = 0xAA; // AMD unlock cycle 1
5270 flash[0x5555] = 0x55; // AMD unlock cycle 2
5271 *addr = 0x30; // AMD erase sector command
5272 int_enable(cpsr);
5273 break;
5274 default:
5275 return -1; // Not yet supported
5276 }
5277 return 0;
5278 }
5279
5280 // Note to use this function with SB drivers require a multi bank flash device
5281 int testh_erase_suspend(uint8 block)
5282 {
5283 volatile char *flash = dev.base;
5284 volatile char *addr;
5285
5286 ttw(ttr(TTrTest, "esusp(%d)" NL, block));
5287
5288 addr = block2addr(block);
5289 switch (dev.driver) {
5290 case FFS_DRIVER_AMD:
5291 case FFS_DRIVER_AMD_SB:
5292 case FFS_DRIVER_AMD_PSEUDO_SB:
5293 *addr = 0xB0; // AMD erase suspend command
5294 break;
5295 default:
5296 return -1; // Not yet supported
5297 }
5298 return 0;
5299 }
5300
5301 int testh_get_free_block(void)
5302 {
5303 uint8 b;
5304
5305 for (b = 0; b < dev.numblocks; b++)
5306 {
5307 if (is_block(b, BF_IS_FREE) || is_block(b, BF_IS_EMPTY))
5308 return b;
5309 }
5310 return -1;
5311 }
5312 #endif // TARGET == 1
5313
5314 // NOTEME: The below test have not been used in this FFS version
5315 // NOTEME: We don't have an AMD init function
5316 /******************************************************************************
5317 * Test drv.init (low level driver test)
5318 ******************************************************************************
5319 * Test drv.init by start an erase, call drv.init and make sure that the
5320 * erase is finish after we return from drv.init.
5321 *
5322 * Test drv.init by start erase, suspend the erase and make sure that the
5323 * erase is resumed and finish before we return from drv.init. The test is
5324 * valid for both AMD SB and MB driver
5325 ******************************************************************************/
5326
5327 #if (TARGET == 0)
5328 int case_drv_init(int p0, int p1)
5329 {
5330 tw(tr(TR_FUNC, TrTestHigh, "drv_init test not supported on PC sim\n"));
5331 return 1;
5332 }
5333 #else
5334 int case_drv_init(int p0, int p1)
5335 {
5336 volatile char *flash = dev.base;
5337 volatile char *addr;
5338 char myname[] = "/drv_init";
5339 uint32 cpsr;
5340 uint8 block;
5341 int free_b, i, j, command;
5342 enum command { ERASE, WRITE, READ };
5343 char action[6][3] =
5344 {
5345 { WRITE, READ, ERASE },
5346 { WRITE, ERASE, READ },
5347 { READ, WRITE, ERASE },
5348 { READ, ERASE, WRITE },
5349 { ERASE, READ, WRITE },
5350 { ERASE, WRITE, READ }
5351 };
5352
5353 ttw(ttr(TTrTest, "Test ffsdrv.init()" NL));
5354
5355 // Test drv.init 6 times there the flash is in erase mode and 6 times
5356 // there the flash is in erase suspend mode.
5357 for (i = 0; i < 12; i++)
5358 {
5359 free_b = testh_get_free_block();
5360 expect_ok(free_b);
5361
5362 fs.debug[0] = 0;
5363 fs.debug[1] = 0;
5364
5365 // Start erase, run driver init.
5366 error = testh_nb_erase(free_b);
5367 expect_ok(error);
5368 rvf_delay(20); // Make sure that the erase has started
5369
5370 if (i > 6) {
5371 error = testh_erase_suspend(free_b);
5372 expect_ok(error);
5373 rvf_delay(20); // Make sure that the erase is suspended
5374 }
5375
5376 error = ffsdrv.init(); // Call ffsdrv_init() instead? (inc. autodetect)
5377 expect_ok(error);
5378
5379 if (i > 6) {
5380 expect(fs.debug[0], 1); // Test that the erase was resumed
5381 expect_gt(fs.debug[1], 0); // Test what we did wait for erase to finish
5382 }
5383 else {
5384 expect(fs.debug[0], 0); // Erase not resumed
5385 expect_gt(fs.debug[1], 0); // We did wait for an erase to finish
5386 }
5387
5388 // To test that any combination of erase, read or write can be
5389 // performed after drv.init do we make the test below. The first
5390 // complete drv.init test will be like this: Start an erase, call
5391 // drv.init(), write a file, read the file and erase a free block.
5392 for (j = 0; j < 3; j++) {
5393 if (i < 6)
5394 command = action[i][j];
5395 else
5396 command = action[i-6][j];
5397 switch (command) {
5398 case ERASE:
5399 free_b = testh_get_free_block();
5400 expect_ok(free_b);
5401 ffsdrv.erase(free_b);
5402 break;
5403 case READ:
5404 error = test_expect_file(myname, TDATA(2));
5405 if (error) return 1;
5406 break;
5407 case WRITE:
5408 error = tffs_fwrite(myname, TDATA(2));
5409 expect(error, EFFS_OK);
5410 break;
5411 default:
5412 return 1;
5413 }
5414 }
5415 }
5416 return 0;
5417 }
5418 #endif
5419
5420 uint32 get_cpsr(void);
5421
5422 // Test disable and enable of interrupt (Note only valid for MB drivers)
5423 int case_interrupt(int p0, int p1)
5424 {
5425 extern void int_enable(uint32 tmp);
5426 extern uint32 int_disable(void);
5427 uint32 cpsr;
5428 int i;
5429 uint32 xcpsr;
5430
5431 switch (dev.driver) {
5432 case FFS_DRIVER_AMD:
5433 case FFS_DRIVER_INTEL:
5434 break;
5435 default:
5436 return 1; // Not supported
5437 }
5438
5439 p0 *= 100;
5440
5441 ttw(ttr(TTrTest, "Test interrupt code" NL));
5442
5443 for (i = 0; i < p0; i++) {
5444 cpsr = int_disable();
5445 xcpsr = get_cpsr();
5446 if ((xcpsr & 0xC0) != 0xC0) {
5447 ttw(ttr(TTrTest, "FATAL int not disablet!" NL));
5448 return 1;
5449 }
5450 int_enable(cpsr);
5451 xcpsr = get_cpsr();
5452 if ((xcpsr & 0xC0) != 0) {
5453 ttw(ttr(TTrTest, "FATAL int not enablet!" NL));
5454 return 1;
5455 }
5456 if ((i % 1000) == 0)
5457 ttw(ttr(TTrTest, "loop (%d)" NL, i));
5458 }
5459
5460 return 0;
5461 }
5462
5463
5464 /******************************************************************************
5465 * Benchmarking
5466 ******************************************************************************/
5467 // Below define only for PC compile
5468 #if (TARGET == 0)
5469 #define UINT32 int
5470 #define tffs_timer_begin() 0
5471 #define tffs_timer_end(time) 5000 - time
5472 #endif
5473
5474 const struct results_s old_results[] =
5475 {
5476 { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 },
5477 { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }
5478 };
5479
5480 // NOTEME: Bencmark_results() not finished.
5481 int bencmark_results(int manufact, int driver, int sample,
5482 struct results_s *meas_res)
5483 {
5484 switch(manufact) {
5485 case MANUFACT_AMD:
5486 switch(driver) {
5487 // case FFS_DRIVER_AMD:
5488 case FFS_DRIVER_AMD_SB:
5489 default: return 0;
5490 }
5491 case MANUFACT_FUJITSU:
5492 switch(driver) {
5493 // case FFS_DRIVER_AMD:
5494 case FFS_DRIVER_AMD_SB:
5495 default: return 0;
5496 }
5497 default: return 0;
5498 }
5499 }
5500
5501 // Benchmark write and read operation. Measure write throughput as kBps in
5502 // idle and loopback mode. Both for an empty ffs where we don't have to
5503 // reclaim blocks and for a full ffs where block reclaims/erasures affect
5504 // the overall throughput. For SB flash driver as well as for DB driver. For
5505 // files of size:16, 256, 4096, bytes.
5506 int case_bm_stream_rw(int p0, int p1)
5507 {
5508 #if (TARGET == 1)
5509 UINT32 time_begin, elapsed, bytes_max;
5510 char myname[] = "/Benchmark0";
5511 int i, j, k, nm_files, file_size;
5512 int size, write_size[] = {16, 256, 4096};
5513 fd_t fdi;
5514
5515 //ttw(ttr(TTrTest, "RVF_POOL_0_SIZE: %d" NL, RVF_POOL_0_SIZE));
5516
5517 // 1 block is used for inodes, 1 have to be free, 1/4 is used for
5518 // journal and 1/2 is used as a margin
5519
5520 // FIXME: changed to 4.75 because the SW TCS2.1 dos make some files and
5521 // because of this is there not enough data space left to this test.
5522 // NOTEME: Instead of hardwire this value then query it
5523 bytes_max = (dev.numblocks - 4.75) * dev.blocksize;
5524 file_size = bytes_max / dev.numblocks;
5525 ttw(ttr(TTrTest, "bytes_max: %d(- 2.75 block)" NL, bytes_max));
5526
5527 // Format, measure write of 16B.
5528 // Format, measure write of 256B.
5529 // Format, measure write of 4096B.
5530
5531 for (j = 0; j < 3; j++) {
5532 ttw(ttr(TTrTest, "Format, measure write of %dB" NL, write_size[j]));
5533
5534 error = tffs_preformat(0xDEAD);
5535 expect(error, EFFS_OK);
5536 error = tffs_format("/foo", 0x2BAD);
5537 expect(error, EFFS_OK);
5538
5539 time_begin = tffs_timer_begin();
5540 for (i = 0; i < dev.numblocks; i++) {
5541 myname[10] = i + (i <= 9 ? '0': 'A' -10);
5542
5543 fdi = tffs_open(myname, FFS_O_WRONLY | FFS_O_CREATE);
5544 expect(fdi, FFS_FD_OFFSET);
5545
5546 size = 0;
5547 do {
5548 size += error = tffs_write(fdi, (char*)tdata[TDATA_HUGE],
5549 write_size[j]);
5550 if (error < 0) {
5551 if (error == EFFS_NOSPACE) {
5552 ttw(ttr(TTrTest, "Aarg out of space" NL));
5553 bytes_max = size;
5554 break;
5555 }
5556 else
5557 expect_ok(error);
5558 }
5559 } while (size < file_size);
5560
5561 error = tffs_close(fdi);
5562 expect_ok(error);
5563 }
5564
5565 elapsed = tffs_timer_end(time_begin);
5566 // FFS has used the CPU for a long time so let the trace task get
5567 // the CPU for a while
5568 tffs_delay(50);
5569 ttw(ttr(TTrTest, "Write %dBytes %d times in %dms, %dkBps" NL NL,
5570 write_size[j], file_size/write_size[j], elapsed,
5571 (bytes_max * 1000) / (elapsed * 1024)));
5572 }
5573
5574
5575 ttw(ttr(TTrTest, "Rm all files, measure write of 4kB" NL));
5576 for (i = 0; i < dev.numblocks; i++) {
5577 myname[10] = i + (i <= 9 ? '0': 'A' -10);
5578 error = tffs_remove(myname);
5579 expect_ok(error);
5580 }
5581
5582 time_begin = tffs_timer_begin();
5583 for (i = 0; i < dev.numblocks; i++) {
5584 myname[10] = i + (i <= 9 ? '0': 'A' -10);
5585 fdi = tffs_open(myname, FFS_O_WRONLY | FFS_O_CREATE);
5586 expect(fdi, FFS_FD_OFFSET);
5587
5588 size = 0;
5589 do {
5590 size += tffs_write(fdi, (char*)tdata[TDATA_HUGE], write_size[2]);
5591 } while (size < file_size);
5592
5593 error = tffs_close(fdi);
5594 expect_ok(error);
5595 }
5596
5597 elapsed = tffs_timer_end(time_begin);
5598 tffs_delay(50);
5599 ttw(ttr(TTrTest, "Write %dBytes %d times in %dms, %dkBps" NL NL,
5600 write_size[2], file_size/write_size[2], elapsed,
5601 (bytes_max * 1000) / (elapsed * 1024)));
5602
5603
5604 // Read all files with 16B at a time
5605 for (j = 0; j < 3; j++) {
5606 ttw(ttr(TTrTest, "Read all files with %dB at a time" NL, write_size[j]));
5607 time_begin = tffs_timer_begin();
5608
5609 for (i = 0; i < dev.numblocks; i++) {
5610 myname[10] = i + (i <= 9 ? '0': 'A' -10);
5611 fdi = tffs_open(myname, FFS_O_RDONLY);
5612 expect(fdi, FFS_FD_OFFSET);
5613 size = 0;
5614 do {
5615 size += tffs_read(fdi, bigbuf, write_size[j]);
5616 } while (size < file_size);
5617
5618 error = tffs_close(fdi);
5619 expect(error, 0);
5620 }
5621 elapsed = tffs_timer_end(time_begin);
5622
5623 ttw(ttr(TTrTest, "Read %d files of %dkB in %dms, %dkBps," NL NL,
5624 dev.numblocks, size/1024, elapsed,
5625 (dev.numblocks * size * 1000) / (elapsed * 1024)));
5626 }
5627
5628
5629 // Measure write and read of one big file
5630 ttw(ttr(TTrTest, "Format, Write and read one big file" NL));
5631
5632 error = tffs_preformat(0xDEAD);
5633 expect(error, EFFS_OK);
5634 error = tffs_format("/foo", 0x2BAD);
5635 expect(error, EFFS_OK);
5636
5637 time_begin = tffs_timer_begin();
5638
5639 fdi = tffs_open(myname, FFS_O_WRONLY | FFS_O_CREATE);
5640
5641 size = 0;
5642 do {
5643 size += tffs_write(fdi, bigbuf, write_size[2]);
5644 } while (size < bytes_max);
5645
5646 error = tffs_close(fdi);
5647 expect(error, 0);
5648
5649 elapsed = tffs_timer_end(time_begin);
5650
5651 ttw(ttr(TTrTest, "Write %dkB in %dms, %dkBps" NL,
5652 size/1024 , elapsed, (size * 1000) / (elapsed * 1024)));
5653
5654
5655 time_begin = tffs_timer_begin();
5656
5657 // read one big file (reading one big file take more time than reading
5658 // many files of the same total size because there is a lot more inode
5659 // to look through)
5660 fdi = tffs_open(myname, FFS_O_RDONLY);
5661 expect(fdi, FFS_FD_OFFSET);
5662 size = 0;
5663 do {
5664 size += tffs_read(fdi, bigbuf, write_size[2]);
5665 } while (size < bytes_max);
5666
5667 error = tffs_close(fdi);
5668 expect(error, 0);
5669
5670 elapsed = tffs_timer_end(time_begin);
5671
5672 ttw(ttr(TTrTest, "Read %dkB in %dms, %dkBps," NL,
5673 size/1024, elapsed, (size * 1000) / (elapsed * 1024)));
5674
5675 // Make a data base of old measurements her? compare with old data and
5676 // make a warning if the performance has decreased?
5677
5678 return 0;
5679 #else
5680 return 1;
5681 #endif
5682 }
5683
5684 // Benchmark ffs_file_write() and ffs_file_read()
5685 int case_bm_file_rw(int p0, int p1)
5686 {
5687 UINT32 time_begin, elapsed;
5688 char myname[] = "/Benchmark";
5689 int i, j;
5690
5691 struct test_info_s {
5692 int fsize;
5693 int fnumber;
5694 };
5695
5696 struct test_info_s tinfo[] = {
5697 { 16, 500 },
5698 { 256, 200 },
5699 { 4096, 200 },
5700 {32768, 10 }
5701 };
5702
5703 for (j = 0; j < 4; j++) {
5704 ttw(ttr(TTrTest, "Format, measure write of %dB" NL, tinfo[j].fsize));
5705
5706 error = tffs_preformat(0xDEAD);
5707 expect(error, EFFS_OK);
5708 error = tffs_format("/foo", 0x2BAD);
5709 expect(error, EFFS_OK);
5710
5711 time_begin = tffs_timer_begin();
5712 for (i = 0; i < tinfo[j].fnumber; i++) {
5713 error = tffs_file_write(myname, (char*)tdata[TDATA_HUGE],
5714 tinfo[j].fsize, FFS_O_CREATE);
5715 expect_ok(error);
5716 }
5717 elapsed = tffs_timer_end(time_begin);
5718
5719 // FFS has used the CPU for a long time so let the trace task get
5720 // the CPU for a while
5721 tffs_delay(50);
5722 ttw(ttr(TTrTest, "Write %dBytes %d times in %dms, %dkBps" NL NL,
5723 tinfo[j].fsize, i, elapsed,
5724 (i * tinfo[j].fsize * 1000) / (elapsed * 1024)));
5725
5726 time_begin = tffs_timer_begin();
5727 for (i = 0; i < tinfo[j].fnumber; i++) {
5728 error = tffs_file_read(myname, bigbuf, tinfo[j].fsize);
5729 expect_ok(error);
5730 }
5731 elapsed = tffs_timer_end(time_begin);
5732
5733 ttw(ttr(TTrTest, "Read %dBytes %d times in %dms, %dkBps" NL NL,
5734 tinfo[j].fsize, i, elapsed,
5735 (i * tinfo[j].fsize * 1000) / (elapsed * 1024)));
5736 }
5737 tffs_delay(3000);
5738
5739 return 0;
5740 }
5741
5742 // We banchmark lookups by using ffs_stat()
5743 int case_bm_lookup(int p0, int p1)
5744 {
5745 //format, update file x times, measure lookup with ffs_stat()
5746 //format, make x difference files, measure lookup with ffs_stat()
5747 //format, make a file in max directory depth, measure lookup with ffs_stat()
5748 //format, make one file, measure lookup with ffs_stat() (no traverse)
5749 //format, make x dirs with x files, measure lookup of last file in last dir
5750 #if (TARGET == 1)
5751 UINT32 time_begin, elapsed;
5752 int i, j;
5753 char myname[] = "/benchmark";
5754 char mydir[100] = "";
5755 char depth[] = "/depth99";
5756 char mypath[100] = "";
5757
5758 //format, update file 100 times, measure lookup with ffs_stat()
5759 error = tffs_preformat(0xDEAD);
5760 expect(error, EFFS_OK);
5761 error = tffs_format("/foo", 0x2BAD);
5762 expect(error, EFFS_OK);
5763
5764 for (i = 0; i < 100; i++) {
5765 error = tffs_file_write(myname, 0,0, FFS_O_CREATE);
5766 expect_ok(error);
5767 }
5768
5769 time_begin = tffs_timer_begin();
5770 for (i = 0; i < 100; i++) {
5771 error = tffs_stat(myname, &stat);
5772 expect_ok(error);
5773 }
5774 elapsed = tffs_timer_end(time_begin);
5775
5776 ttw(ttr(TTrTest, "Lookup through 100 deleted objects 100 times in %dms(%d obj/s)" NL, elapsed, 100 * 1000 / elapsed));
5777
5778 //format, make 100 difference files, measure lookup with ffs_stat()
5779 error = tffs_preformat(0xDEAD);
5780 expect(error, EFFS_OK);
5781 error = tffs_format("/foo", 0x2BAD);
5782 expect(error, EFFS_OK);
5783
5784 for (i = 0; i < 100; i++) {
5785 sprintf(myname, "/benchm%d", i);
5786 error = tffs_file_write(myname, 0,0, FFS_O_CREATE);
5787 expect_ok(error);
5788 }
5789
5790 time_begin = tffs_timer_begin();
5791 for (i = 0; i < 100; i++) {
5792 error = tffs_stat(myname, &stat);
5793 expect_ok(error);
5794 }
5795 elapsed = tffs_timer_end(time_begin);
5796
5797 ttw(ttr(TTrTest, "Lookup through 100 difference objects 100 times in %dms(%d obj/s)"
5798 NL, elapsed, 100 * 1000 / elapsed));
5799
5800 //format, make a file in max directory depth, measure lookup with ffs_stat()
5801 error = tffs_preformat(0xDEAD);
5802 expect(error, EFFS_OK);
5803 error = tffs_format("/foo", 0x2BAD);
5804 expect(error, EFFS_OK);
5805
5806 for (i = 0; i < fs.path_depth_max - 1; i++) {
5807 sprintf(depth, "/depth%d", i + 1);
5808 strcat(mydir, depth);
5809 error = tffs_mkdir(mydir);
5810 expect_ok(error);
5811 }
5812
5813 strcat(mydir, myname);
5814 error = tffs_file_write(mydir, 0,0, FFS_O_CREATE);
5815 expect_ok(error);
5816
5817 time_begin = tffs_timer_begin();
5818 for (i = 0; i < 1000; i++) {
5819 error = tffs_stat(mydir, &stat);
5820 expect_ok(error);
5821 }
5822 elapsed = tffs_timer_end(time_begin);
5823
5824 ttw(ttr(TTrTest, "Lookup througt %d directorys 1000 times in %dms(%d obj/s)"
5825 NL, fs.path_depth_max, elapsed, 1000 * 1000 / elapsed));
5826
5827 //format, make x files in each directory, measure lookup with ffs_stat()
5828 error = tffs_preformat(0xDEAD);
5829 expect(error, EFFS_OK);
5830 error = tffs_format("/foo", 0x2BAD);
5831 expect(error, EFFS_OK);
5832
5833 mydir[0] = 0; // Reset/empty buffer
5834
5835 for (i = 0; i < fs.path_depth_max - 1; i++) {
5836 sprintf(depth, "/depth%d", i + 1);
5837 strcat(mydir, depth);
5838 error = tffs_mkdir(mydir);
5839 expect_ok(error);
5840
5841 for (j = 0; j < 5; j++){
5842 sprintf(myname, "/benchm%d", j);
5843 strcpy(mypath, mydir);
5844 strcat(mypath, myname);
5845 error = tffs_file_write(mypath, 0,0, FFS_O_CREATE);
5846 expect_ok(error);
5847 }
5848 }
5849
5850 time_begin = tffs_timer_begin();
5851 for (i = 0; i < 100; i++) {
5852 error = tffs_stat(mypath, &stat);
5853 expect_ok(error);
5854 }
5855 elapsed = tffs_timer_end(time_begin);
5856
5857 ttw(ttr(TTrTest, "Lookup througt %d directorys with %d files in each times in %dms(%d obj/s)"
5858 NL, fs.path_depth_max, j, elapsed, 100 * 1000 / elapsed));
5859
5860 //format, make one file, measure lookup with ffs_stat() (no traverse)
5861 error = tffs_preformat(0xDEAD);
5862 expect(error, EFFS_OK);
5863 error = tffs_format("/foo", 0x2BAD);
5864 expect(error, EFFS_OK);
5865
5866 error = tffs_file_write(myname, 0,0, FFS_O_CREATE);
5867 expect_ok(error);
5868
5869 time_begin = tffs_timer_begin();
5870 for (i = 0; i < 10000; i++) {
5871 error = tffs_stat(myname, &stat);
5872 expect_ok(error);
5873 }
5874 elapsed = tffs_timer_end(time_begin);
5875
5876 ttw(ttr(TTrTest, "Lookup one file 10000 times in %dms(%d obj/s)"
5877 NL, elapsed, 1000 * 10000 / elapsed));
5878
5879 //format, make x dirs with x files, measure lookup of last file in last dir
5880 error = tffs_preformat(0xDEAD);
5881 expect(error, EFFS_OK);
5882 error = tffs_format("/foo", 0x2BAD);
5883 expect(error, EFFS_OK);
5884 return 0;
5885 #else
5886 return 1;
5887 #endif
5888 }
5889
5890 int case_bm_seek(int p0, int p1)
5891 {
5892 UINT32 time_begin, elapsed;
5893 int i, pos, fsize = 20000;
5894 char myname[] = "/benchmark";
5895 fd_t fdi;
5896
5897 //format, update file 100 times, measure lookup with ffs_stat()
5898 error = tffs_preformat(0xDEAD);
5899 expect(error, EFFS_OK);
5900 error = tffs_format("/foo", 0x2BAD);
5901 expect(error, EFFS_OK);
5902
5903 // Create test file
5904 error = tffs_file_write(myname, (char*)tdata[TDATA_HUGE], fsize, FFS_O_CREATE);
5905 expect_ok(error);
5906
5907 fdi = tffs_open(myname, FFS_O_RDWR);
5908 expect(fdi, FFS_FD_OFFSET);
5909
5910 tw(tr(TR_FUNC, TrTest, "Run read reference\n"));
5911 ttw(ttr(TTrTest, "Run read reference" NL));
5912
5913 // Reference
5914 time_begin = tffs_timer_begin();
5915 for (i = 0; i < fsize/4; i++) {
5916 error = tffs_read(fdi, bigbuf, 4);
5917 expect_ok(error);
5918 }
5919 elapsed = tffs_timer_end(time_begin);
5920
5921 tw(tr(TR_FUNC, TrTest, "Read %d entries of 4 bytes (bytes/s %d)\n",
5922 fsize/4, (fsize * 1000 / elapsed)));
5923
5924 ttw(ttr(TTrTest, "Read %d entries of 4 bytes in %dms (bytes/s %d)" NL,
5925 fsize/4, elapsed, (fsize * 1000 / elapsed)));
5926
5927 tw(tr(TR_FUNC, TrTest, "Benchmark seek\n"));
5928 ttw(ttr(TTrTest, "Benchmark seek" NL));
5929
5930 error = tffs_seek(fdi, 0, FFS_SEEK_SET);
5931 expect_ok(error);
5932 pos = 0;
5933
5934 // Benchmark seek
5935 time_begin = tffs_timer_begin();
5936 for (i = 0; i < fsize/4; i++) {
5937 error = tffs_seek(fdi, pos, FFS_SEEK_SET);
5938 expect_ok(error);
5939 pos += 4;
5940 error = tffs_read(fdi, bigbuf, 4);
5941 expect_ok(error);
5942 }
5943 elapsed = tffs_timer_end(time_begin);
5944
5945 tw(tr(TR_FUNC, TrTest, "Seek and Read %d entries of 4 bytes (bytes/s %d)\n",
5946 fsize/4, (fsize * 1000 / elapsed)));
5947
5948 ttw(ttr(TTrTest, "Seek and Read %d entries of 4 bytes in %dms (bytes/s %d)" NL,
5949 fsize/4, elapsed, (fsize * 1000 / elapsed)));
5950
5951 error = tffs_close(fdi);
5952 expect_ok(error);
5953
5954 return EFFS_OK;
5955 }
5956
5957 // Benchmark the blocking 'system'. Make n number of calls to ffs_format()
5958 // with bad magic.
5959 int case_bm_blocking(int p0, int p1)
5960 {
5961 int i;
5962 int time_begin, elapsed;
5963
5964 time_begin = tffs_timer_begin();
5965 for (i = 0; i < 10000; i++)
5966 tffs_format("/ffs", 0);
5967 elapsed = tffs_timer_end(time_begin);
5968
5969 ttw(ttr(TTrTest, "Made %d blocking calls in %dms (ms/call %d)" NL,
5970 i, elapsed, elapsed / i));
5971
5972 // Make sure that it did fail as we expected
5973 error = tffs_format("/ffs", 0);
5974 expect(error, EFFS_INVALID);
5975
5976 return EFFS_OK;
5977 }
5978 /******************************************************************************
5979 * Core test case
5980 ******************************************************************************/
5981
5982 int case_seekfile(int p0, int p1)
5983 {
5984 // NOTEME: this is only a visual test add some expect() to finish it.
5985 char *name;
5986 iref_t i, dir, i_out;
5987 int length, flength;
5988 int segment_offset;
5989 fd_t fdi;
5990
5991 error = tffs_fwrite("/File_seg", (char*)tdata[TDATA_HUGE], 50);
5992 expect(error, EFFS_OK);
5993
5994 if ((i = object_lookup("/File_seg", &name, &dir)) < 0)
5995 return i;
5996
5997 for (length = 52; length > 48; length--) {
5998 flength = segfile_seek(i, length, &i_out, &segment_offset);
5999 tw(tr(TR_FUNC, TrTest,
6000 "*** Length %d, Flength %d, Seg_offset %2d, i %d ***\n",
6001 length, flength, segment_offset, i_out));
6002 }
6003
6004 fdi = tffs_open("/Stream_seg", FFS_O_WRONLY | FFS_O_CREATE);
6005 tffs_write(fdi, (char*)tdata[TDATA_HUGE], 55);
6006 tffs_close(fdi);
6007
6008 fdi = tffs_open("/Stream_seg", FFS_O_WRONLY | FFS_O_APPEND);
6009 tffs_write(fdi, (char*)tdata[TDATA_HUGE], 50);
6010 tffs_close(fdi);
6011
6012 fdi = tffs_open("/Stream_seg", FFS_O_WRONLY | FFS_O_APPEND);
6013 tffs_write(fdi, (char*)tdata[TDATA_HUGE], 45);
6014 tffs_close(fdi);
6015
6016 if ((i = object_lookup("/Stream_seg", &name, &dir)) < 0)
6017 return i;
6018
6019 for (length = 153; length > 148; length--) {
6020 flength = segfile_seek(i, length, &i_out, &segment_offset);
6021 tw(tr(TR_FUNC, TrTest,
6022 "*** Length %d, Flength %d, Seg_offset %2d, i %d ***\n",
6023 length, flength, segment_offset, i_out));
6024 }
6025
6026 for (length = 107; length > 103; length--) {
6027 flength = segfile_seek(i, length, &i_out, &segment_offset);
6028 tw(tr(TR_FUNC, TrTest,
6029 "*** Length %d, Flength %d, Seg_offset %2d, i %d ***\n",
6030 length, flength, segment_offset, i_out));
6031 }
6032
6033 for (length = 58; length > 53; length--) {
6034 flength = segfile_seek(i, length, &i_out, &segment_offset);
6035 tw(tr(TR_FUNC, TrTest,
6036 "*** Length %d, Flength %d, Seg_offset %2d, i %d ***\n",
6037 length, flength, segment_offset, i_out));
6038 }
6039
6040 return 0;
6041 }
6042
6043 void display_fs_params(void)
6044 {
6045 int numdatablocks;
6046
6047 numdatablocks = dev.numblocks - fs.blocks_free_min - 1;
6048
6049 // Note expect only one free block!
6050 tw(tr(TR_FUNC, TrTest, "Data blocks %d, size %dkB\n",
6051 numdatablocks, dev.blocksize / 1024));
6052 tw(tr(TR_FUNC, TrTest, " User data: %dkB\n",
6053 (numdatablocks * dev.blocksize - fs.reserved_space) / 1024));
6054 tw(tr(TR_FUNC, TrTest, " reserved_space: %dkB\n", fs.reserved_space/ 1024));
6055 tw(tr(TR_FUNC, TrTest, " inodes_max: %d\n", fs.inodes_max));
6056 tw(tr(TR_FUNC, TrTest, " journal_size: %d\n", fs.journal_size));
6057 tw(tr(TR_FUNC, TrTest, " objects_max: %d\n", fs.objects_max));
6058 tw(tr(TR_FUNC, TrTest, " block_files_max: %d\n", fs.block_files_max));
6059 tw(tr(TR_FUNC, TrTest, " chunk_size: %d\n\n", fs.chunk_size_max));
6060 }
6061
6062 // NOTE only visual test
6063 int case_param_init(int p0, int p1)
6064 {
6065 char myname[] = "/ffs/xx/xx/xx";
6066 int numblocks[] = { 8, 3, 7, 15, 31, 63, 96, 127, 61, 127};
6067 int i;
6068
6069 for (i = 0; i < sizeof(numblocks) / sizeof(int); i++)
6070 {
6071 dev.numblocks = numblocks[i];
6072
6073 if (i == 0)
6074 dev.blocksize = 1024 * 8;
6075 else if (i > 0 && i < 8)
6076 dev.blocksize = 1024 * 64;
6077 else
6078 dev.blocksize = 1024 * 128;
6079
6080 fs_params_init(myname);
6081 display_fs_params();
6082
6083 // TODO add some checks, test format("/ffs/bx/ox") etc.
6084 }
6085
6086 error = tffs_initialize(); // Use the 'real' blocksize
6087 expect(error, EFFS_OK);
6088
6089 // Test param input from the format string
6090 error = tffs_preformat(0xDEAD);
6091 expect(error, EFFS_OK);
6092 tffs_format("/ffs/i4000/o3000", 0x2BAD);
6093 expect(error, EFFS_OK);
6094
6095 display_fs_params();
6096
6097 // Test small conf.
6098 tw(tr(TR_FUNC, TrTest, "Test small conf\n"));
6099 error = tffs_preformat(0xDEAD);
6100 expect(error, EFFS_OK);
6101 tffs_format("/ffs/b3/j8/r24/f100/o100", 0x2BAD);
6102 expect(error, EFFS_OK);
6103 display_fs_params();
6104
6105 return 0;
6106 }
6107
6108 /******************************************************************************
6109 * Trace mask
6110 ******************************************************************************/
6111
6112 // This is not a test case but a way to change the trace mask on the fly
6113 int case_trace_mask(int p0, int p1)
6114 {
6115 unsigned int temp;
6116
6117 // Because p0 and p1 only is of the type int and trace_mask is a
6118 // unsigned int is p0 bit 1-28 and p1 is bit 29-32
6119 temp = p1 << 28;
6120 temp |= p0;
6121
6122 #if (TARGET == 0)
6123 tr_init(temp, 2, NULL );
6124 tw(tr(TR_FUNC, TrAll, "trace mask = %x\n", temp));
6125 #else
6126 ttr_init(temp);
6127 ttw(ttr(TTrTest, "trase mask = %x" NL, temp));
6128 #endif
6129 return 0;
6130 }
6131
6132 /******************************************************************************
6133 * Helper function
6134 ******************************************************************************/
6135
6136 #define CLEANUP_MIN_SIZE 1
6137
6138 int ignore_file(char *pathname)
6139 {
6140 int i;
6141 const char *ignore[] = {
6142 "/.journal", "/.ffs-stats", "/.ffs-bstat.old","/dummy4fctrl" ,
6143 "/truncate/IMEI", "/truncate/stream_file", "/gsm/com/rfcap",
6144 "/var/dbg/dar" };
6145 const char readonly[] = "/READONLY";
6146
6147 for (i = 0; i < sizeof(ignore)/sizeof(char *); i++) {
6148 if (ffs_strcmp(pathname, ignore[i]) == 0) {
6149 return 0;
6150 }
6151 }
6152
6153 // Special case, we have one directory where we don't delete any objects
6154 if (memcmp(pathname, readonly, sizeof(readonly) - 1) == 0)
6155 return 0;
6156
6157 return 1;
6158 }
6159
6160 // Remove files in FFS until there is min_space left. Ignore all files that
6161 // are listed in the ignore string above.
6162 int case_cleanup(int min_space)
6163 {
6164 int free, i, n, error;
6165 struct object_s *plist, *plist_start;
6166 char *pnames, *pnames_start;
6167
6168 tw(tr(TR_FUNC, TrTestLow, "case_cleanup(%d)?\n", min_space));
6169
6170 ffs_query(Q_BYTES_FREE, (uint32 *) &free);
6171 if (free > min_space) {
6172 tw(tr(TR_FUNC, TrTestLow, "Total data free: %d\n", free));
6173 return free; // Nothing to do
6174 }
6175
6176 plist = plist_start = (struct object_s *) bigbuf;
6177 pnames = pnames_start = bigbuf + bigbuf_size;
6178 n = case_dir_list("/", &plist, &pnames);
6179 plist = plist_start;
6180
6181 for (i = 0; i < n; i++, plist++)
6182 {
6183 ffs_query(Q_BYTES_FREE, (uint32 *) &free);
6184 if (free > min_space) {
6185 tw(tr(TR_FUNC, TrTestLow, "Total data free: %d\n", free));
6186 return free;
6187 }
6188
6189 if (plist->stat.size > CLEANUP_MIN_SIZE && ignore_file(plist->name)) {
6190 if ((error = ffs_remove(plist->name)) < 0)
6191 return error;
6192 }
6193 }
6194 return EFFS_NOSPACE;
6195 }
6196
6197 // Make a random file. Random filename length, random filename, random filesize.
6198 // Note it is up to the caller that the directory exists.
6199 int case_mk_rand_file(char *dirname, int max_size, int min_size)
6200 {
6201 int size, diff, filenamelen, pathlen = 0, i, error;
6202 char pathname[(FFS_FILENAME_MAX + 1) * FFS_PATH_DEPTH_MAX];
6203 // All valid filename chars
6204 char tabel[] = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"
6205 "abcdefghijklmnopqrtsuvwyz_-.,+%$#";
6206
6207 diff = max_size - min_size;
6208 if (diff < 0)
6209 return EFFS_INVALID;
6210 else if (diff == 0)
6211 size = max_size;
6212 else
6213 size = (rand() % diff) + min_size;
6214
6215 filenamelen = (rand() % (FFS_FILENAME_MAX - 1)) + 1; // We need min one char
6216 tw(tr(TR_FUNC, TrTestLow, "size:%5d, filenamelen:%3d,",size, filenamelen));
6217
6218 if (dirname != 0) {
6219 strcpy(pathname, dirname);
6220 pathlen = strlen(pathname);
6221 }
6222
6223 pathname[pathlen++] = '/';
6224
6225 for (i = 0; i < filenamelen; pathlen++, i++) {
6226 pathname[pathlen] = tabel[rand() % strlen(tabel)];
6227 }
6228 pathname[pathlen] = 0; // Zero terminate
6229
6230 tw(tr(TR_FUNC, TrTestLow, "pathname:'%s'\n", pathname, size));
6231
6232 error = tffs_file_write(pathname, (char *) tdata[TDATA_HUGE], size,
6233 FFS_O_CREATE | FFS_O_TRUNC);
6234 return error;
6235
6236 }
6237
6238 // Various helper functions for debug
6239 int case_debug_help(int p0, int p1)
6240 {
6241 int tr_mask;
6242
6243 switch(p0) {
6244 case 1:
6245 {
6246 // Write rfcap file
6247 char rfcap_data[] = { 0, 0xf, 0x41, 0x10, 0, 0, 0, 0, 0x50, 0,
6248 0, 0xa5, 0x5, 0, 0xc0, 0 };
6249
6250 ttw(ttr(TTrTest, "Write rfcap file..."));
6251 tffs_mkdir("/gsm"); // ignore error
6252 tffs_mkdir("/gsm/com");
6253 error = tffs_file_write("/gsm/com/rfcap", rfcap_data,
6254 sizeof(rfcap_data), FFS_O_CREATE | FFS_O_TRUNC);
6255 expect_ok(error);
6256 ttw(ttr(TTrTest, "done" NL));
6257 }
6258 break;
6259 case 2:
6260 tr_mask = tr_query(INT_MAX);
6261 #if (TARGET == 0)
6262 tr_init(TrBstat, 2, NULL );
6263 tr_bstat();
6264 tr_init(tr_mask, 2, NULL );
6265 #else
6266 ttr_init(TTrBstat);
6267 tr_bstat();
6268 ttr_init(tr_mask);
6269 #endif
6270 break;
6271
6272 case 3:
6273 // Make mem dump to screen
6274 #if (TARGET == 1)
6275 rvf_dump_mem();
6276 break;
6277 #else
6278 tw(tr(TR_FUNC, TrTestLow, "Not supported on PC\n"));
6279 return 1;
6280 #endif
6281
6282 case 4:
6283 #if (TARGET == 1)
6284 rvf_dump_tasks(); break;
6285 #else
6286 tw(tr(TR_FUNC, TrTestLow, "Not supported on PC\n"));
6287 return 1;
6288 #endif
6289 case 5:
6290 #if (TARGET == 1)
6291 rvf_dump_pool(); break;
6292 #else
6293 tw(tr(TR_FUNC, TrTestLow, "Not supported on PC\n"));
6294 return 1;
6295 #endif
6296 default:
6297 tw(tr(TR_FUNC, TrTest, "Available helper functions:\n"));
6298 tw(tr(TR_FUNC, TrTest, " 1: Write rfcap file\n"));
6299 tw(tr(TR_FUNC, TrTest, " 2: Display bstat trace\n"));
6300
6301 ttw(ttr(TTrTest, "Available helper functions:" NL));
6302 ttw(ttr(TTrTest, " 1: Write rfcap file" NL));
6303 ttw(ttr(TTrTest, " 2: Display bstat trace" NL));
6304 ttw(ttr(TTrTest, " 3: Memory dump to screen" NL));
6305 ttw(ttr(TTrTest, " 4: Tasks dump to screen" NL));
6306 ttw(ttr(TTrTest, " 5: Pool dump to screen" NL));
6307 return 0;
6308 }
6309
6310 return 0;
6311 }
6312
6313 /******************************************************************************
6314 * Hexdumping
6315 ******************************************************************************/
6316 #if 0
6317 void hexdump(const char *buf, int size, unsigned int address, int unitsize)
6318 {
6319 char string[(8+1+1) + (1+16+1+1) + (3*16) + 1];
6320 int n, i;
6321 char *s;
6322
6323 while (size > 0)
6324 {
6325 s = string;
6326 s += sprintf(s, "%8x: ", address); // print offset
6327
6328 n = (size > 16 ? 16 : size);
6329
6330 // print the textual representation
6331 for (i = 0; i < n; i++) {
6332 if (buf[i] >= ' ' && buf[i] < 127)
6333 *s++ = buf[i];
6334 else
6335 *s++ = '.';
6336 }
6337 // pad textual representation with spaces
6338 for (i = 0; i < 16 - n; i++) {
6339 *s++ = ' ';
6340 }
6341 *s++ = ' ';
6342
6343 // print hexedecimal representation
6344 for (i = 0; i < n; i += unitsize) {
6345 switch (unitsize) {
6346 case 1: s += sprintf(s, "%02x ", *(unsigned char *) (buf+i)); break;
6347 case 2: s += sprintf(s, "%04x ", *(unsigned short *) (buf+i)); break;
6348 case 4: s += sprintf(s, "%08x ", *(unsigned int *) (buf+i)); break;
6349 }
6350 }
6351 buf += 16;
6352 address += 16;
6353 size -= 16;
6354 puts(string);
6355 }
6356 }
6357 #endif
6358
6359 /******************************************************************************
6360 * Test Cases
6361 ******************************************************************************/
6362
6363 const struct testcase_s testcase[] =
6364 {
6365 // Collective test cases
6366 { "all", PC , case_all, "All" },
6367 { "alot", PC , case_alot, "Almost all" },
6368 { "tall", IT , case_tall, "Target all" },
6369 { "aall", PC , case_aall, "Agressive all" },
6370 { "list", PC|IT , case_list, "List all testcases" },
6371 { "lsr", PC|IT , case_lsr, "Recursively list all objects" },
6372 { "test", PC|IT , case_test, "Ad hoc test" },
6373 { "rand", PC|IT , case_rand, "Random tests (NOT FINISHED)" },
6374 { "fail", PC|IT , case_fail, "Fail (just fail)" },
6375 { "okay", PC|IT , case_okay, "Okay (just succeed)" },
6376
6377 // Atomic/small test cases
6378 { "s", PC|IT , case_status, "Status: ffs_init(), lsr" },
6379 { "r", PC|IT , case_reset, "preformat(), format(), init(), exit()" },
6380 { "i", PC|IT , case_init, "ffs_init()" },
6381 { "x", PC|IT , case_exit, "ffs_exit()" },
6382 { "p", PC|IT , case_only_preformat, "ffs_preformat()" },
6383 { "f", PC|IT , case_only_format, "ffs_format()" },
6384
6385 // Populate test cases
6386 { "world", PC|IT , case_world, "Make world" },
6387 { "eu", PC|IT , case_europe, "Make europe" },
6388 { "dk", PC|IT , case_denmark, "Make denmark" },
6389
6390 // Special test cases
6391 { "mf", PC|IT , case_mfiles, "Test FFS with many (small) files" },
6392 { "threeb", PC|IT , case_threeb, "Test ffs in only three blocks" },
6393 { "twob", PC|IT , case_twob, "Test ffs in only two blocks" },
6394 { "pcm", PC|IT , case_pcm, "Test PCM interface" },
6395 { "stress", IT , case_stress, "Stress test erase/write" },
6396 { "find", PC|IT|RND, case_find, "Recursively traverse all objects" },
6397 { "nb", IT , case_nonblock, "Test non block functions" },
6398 { "bf", IT|RND , case_blocking, "Test blocking function" },
6399 { "cust", PC , case_customer, "Test customer typical use"},
6400 { "esus", IT , case_erase_suspend, "Test erase suspend and resume" },
6401
6402 // Driver test cases
6403 { "int", IT , case_interrupt, "Test INT disable and enable" },
6404
6405 // Benchmark test cases
6406 { "bmsrw", IT , case_bm_stream_rw, "Benchmark stream read and writes" },
6407 { "bmfrw", IT , case_bm_file_rw, "Benchmark file read and writes" },
6408 { "bmlu", IT , case_bm_lookup, "Benchmark lookups" },
6409 { "bmsk", PC|IT , case_bm_seek, "Benchmark ffs_seek()" },
6410 { "bmbl", PC|IT , case_bm_blocking, "Benchmark blocking 'system'" },
6411
6412 // Normal test cases
6413 { "format", PC , case_format, "Test ffs_pre/format()" },
6414 { "lu", PC|IT , case_lookup, "Test object_lookup()" },
6415 { "fc", PC|IT , case_fcontrol, "Test fcontrol() and read-only" },
6416 { "root", PC|IT|RND, case_root, "Test root inode is non-modifiable" },
6417 { "dirs", PC|IT|RND, case_dirs, "Test Directories" },
6418 { "frd", PC|IT|RND, case_fread, "Test fread() with varying sizes/names" },
6419 { "bfull", PC|IT , case_bfull, "Test filling a block completely" },
6420 { "ffull", PC|IT , case_ffull, "Test filling ffs completely" },
6421 { "bigf", PC , case_bigfile, "Test big files" },
6422 { "stat", PC|IT|RND, case_stat, "Test ffs_stat()" },
6423 { "rm", PC|IT , case_remove, "Test ffs_remove()" },
6424 { "ren", PC|IT|RND, case_rename, "Test ffs_rename()" },
6425 { "renext", PC|IT , case_rename_extended, "Extended test of ffs_rename()" },
6426
6427 { "irec", PC|IT , case_irec, "Test inodes reclaim" },
6428 { "drec", PC , case_drec, "Test data reclaim" },
6429 { "adrec", PC , case_adrec, "Test xxxx reclaim" },
6430
6431 { "jnl", PC|IT , case_journal, "Test journalling" },
6432 { "brec", PC|IT|RND, case_brecover, "Test block recovery" },
6433
6434 { "ssym", PC|IT|RND, case_ssym, "Test Simple Symlinks" },
6435 { "fsym", PC|IT , case_fsym, "Test Full Symlinks (NOT IMPLEMENTED)" },
6436 { "ri", PC|IT|RND, case_reinit, "Test re-ffs_init()" },
6437 { "open", PC|IT|RND, case_open, "Test open flags" },
6438 { "rw", PC|IT|RND, case_rw, "Test read and write files" },
6439 { "mopen", PC|IT|RND, case_multi_open, "Test open of multiply files" },
6440 { "seek", PC|IT|RND, case_seek, "Test seek on files " },
6441 { "trunc", PC|IT|RND, case_trunc, "Test ffs_truncate() " },
6442 { "append", PC|IT|RND, case_append, "Test append on a open file" },
6443 { "dsync", PC|IT|RND, case_datasync, "Test ffs_fdatasync" },
6444 { "ex", IT|RND, case_examples, "Test examples from RIV111 " },
6445 { "fwflags",PC|IT|RND, case_fw_flags, "Test ffs_file_write() flags" },
6446 { "pcm" ,PC|IT|RND, case_pcm, "Test pcm interface" },
6447 { "apiexc" ,PC|IT, case_api_exceptions, "Test API exceptions" },
6448 { "ninit" ,PC|IT, case_api_notformated, "Test API on a nformated flash"},
6449 { "query" ,PC|IT|RND, case_query, "Test query function" },
6450 { "octrl" ,PC|IT, case_octrl, "Test of object control" },
6451
6452 // Core test cases
6453 { "seekf", PC, case_seekfile, "Test core segfile_seek" },
6454 { "param", PC, case_param_init,"Test core param_init" },
6455
6456 // Helper test case
6457 { "tr", PC|IT , case_trace_mask, "Change trace mask on the fly " },
6458 { "dh", PC|IT , case_debug_help, "Difference help functions " },
6459 { 0, 0, 0, 0 }
6460 };
6461
6462 #if (TARGET == 1)
6463 // Note like the interrupt functions in drv.c this asm needs to be places ad
6464 // the buttom of the file or else it crach!
6465 uint32 get_cpsr(void)
6466 {
6467 asm(" .state16");
6468 asm(" adr A2, get_cpsr32");
6469 asm(" bx A2");
6470 asm(" .state32");
6471 asm("get_cpsr32");
6472 asm(" mrs A1,cpsr ; get current CPSR");
6473 }
6474 #else
6475 uint32 get_cpsr(void){return 0;}
6476 #endif
6477
6478
6479
6480
6481
6482
6483
6484
6485
6486
6487
6488