-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsimplefs_aux.c
743 lines (699 loc) · 30.5 KB
/
simplefs_aux.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
#include "simplefs_aux.h"
// searches for a file or a directory given a DirectoryHandle and the element name
SearchResult *SimpleFS_search(DirectoryHandle *d, const char *name) {
//we check that we have been given valid data
if (d == NULL || strlen(name) == 0) {
return NULL;
}
// we will save our results and all our intermediate variables in here
SearchResult *result = (SearchResult *) malloc(sizeof(SearchResult));
result->result = FAILED;
result->dir_block_in_disk = MISSING;
result->element = NULL;
result->last_visited_dir_block = NULL;
result->type = MISSING;
//now we check if the file is already present in the given directory
int i, res;
//this variable is used to stop the search if we have already found a matching filename
int searching = 1;
//we know how many entries has a directory but we don't know how many entries per block we have so we calculate it
int FDB_max_elements = (BLOCK_SIZE - sizeof(BlockHeader) - sizeof(FileControlBlock) - sizeof(int)) / sizeof(int);
int Dir_Block_max_elements = (BLOCK_SIZE - sizeof(BlockHeader)) / sizeof(int);
// in here we will save the read FirstFileBlock/FirstDirectoryBlock to check if has a matching file name
FirstFileBlock *element = malloc(sizeof(FirstFileBlock));
//we save the number of entries in the directory
int dir_entries = d->dcb->num_entries;
// we search for all the files in the current directory block
for (i = 0; i < dir_entries && i < FDB_max_elements && searching; i++) {
memset(element, 0, BLOCK_SIZE);
//we get the file
res = DiskDriver_readBlock(d->sfs->disk, element, d->dcb->file_blocks[i]);
if (res == FAILED) {
free(element);
free(result);
CHECK_ERR(res == FAILED, "the directory contains a free block in the entry list");
}
//we check if the file has the same name of the file we want to create
searching = strncmp(element->fcb.name, name, strlen(name));
}
if (searching == 0) {
result->result = SUCCESS;
result->dir_block_in_disk = MISSING;
result->type = element->fcb.is_dir;
result->last_visited_dir_block = NULL;
result->element = element;
result->pos_in_block = i - 1;
return result;
}
//now we search in the other directory blocks which compose the directory list
int next_block = d->dcb->header.next_block;
int dir_block_in_disk;
//we save the block in disk of the current directory block for future use
//we adjust the number of remaining entries
dir_entries -= FDB_max_elements;
//we start the search in the rest of the chained list
DirectoryBlock *dir = (DirectoryBlock *) malloc(sizeof(DirectoryBlock));
while (next_block != MISSING) {
memset(dir, 0, sizeof(DirectoryBlock));
//we get the next directory block;
dir_block_in_disk = next_block;
// we clean our directory before getting the new one
memset(dir, 0, sizeof(DirectoryBlock));
res = DiskDriver_readBlock(d->sfs->disk, dir, next_block);
if (res == FAILED) {
free(element);
free(dir);
free(result);
CHECK_ERR(res == FAILED, "the directory contains a free block in the entry chain");
}
//we search it
for (i = 0; i < dir_entries && i < Dir_Block_max_elements && searching; i++) {
//we get the file
res = DiskDriver_readBlock(d->sfs->disk, element, dir->file_blocks[i]);
if (res == FAILED) {
free(element);
free(dir);
free(result);
CHECK_ERR(res == FAILED, "the directory contains a free block in the entry list");
}
//we check if the file has the same name of the file we want to create
searching = strncmp(element->fcb.name, name, strlen(name));
}
if (searching == 0) {
result->type = element->fcb.is_dir;
result->dir_block_in_disk = dir_block_in_disk;
result->element = element;
result->last_visited_dir_block = dir;
result->result = SUCCESS;
result->pos_in_block = i;
}
//now we need to change block and to adjust the number of remaining entries to check
dir_entries -= Dir_Block_max_elements;
next_block = dir->header.next_block;
}
if (element != NULL) free(element);
if (dir != NULL) free(dir);
return result;
}
void Dir_addentry(DirectoryHandle *d, int file_block) {
//we check that we have been given valid data
if (d == NULL || file_block <= 0 || file_block >= d->sfs->disk->header->num_blocks) {
return;
}
//we know how many entries has a directory but we don't know how many entries per block we have so we calculate it
int FDB_max_elements = (BLOCK_SIZE - sizeof(BlockHeader) - sizeof(FileControlBlock) - sizeof(int)) / sizeof(int);
int Dir_Block_max_elements = (BLOCK_SIZE - sizeof(BlockHeader)) / sizeof(int);
int res;
//we use the current block, pos_in_block and pos_in_dir to determine the position where we
//must write the new file in the directory
//but it could be uninitialized!
DirectoryBlock *dir = (DirectoryBlock *) malloc(sizeof(DirectoryBlock));
//we determine if the current block is uninitialized
if (d->current_block->next_block != MISSING) {
while (d->current_block->next_block != MISSING) {
res = DiskDriver_readBlock(d->sfs->disk, dir, d->current_block->next_block);
CHECK_ERR(res == FAILED, "can't read the directory block");
memcpy(d->current_block, &(dir->header), sizeof(BlockHeader));
}
} else {
//we check if we must load the current directory block
if (d->current_block->block_in_file != 0) {
res = DiskDriver_readBlock(d->sfs->disk, dir, d->current_block->block_in_disk);
CHECK_ERR(res == FAILED, "can't read the directory block");
}
}
d->pos_in_dir = d->dcb->num_entries;
if (d->current_block->block_in_file == 0) {
d->pos_in_block = d->pos_in_dir - 1;
} else {
d->pos_in_block =
(d->pos_in_dir - FDB_max_elements - 1) - Dir_Block_max_elements * (d->current_block->block_in_file - 1);
}
//we get the index of the block in which we must update the data
int dir_block = d->current_block->block_in_file;
int dir_block_displacement;
if (d->dcb->num_entries == 0) {
dir_block_displacement = 0;
} else {
dir_block_displacement = d->pos_in_block + 1;
}
if (dir_block == 0) {
if (dir_block_displacement < FDB_max_elements) {
//in this case we can add the file directly in the FDB
d->dcb->file_blocks[dir_block_displacement] = file_block;
} else {
//in this case we must allocate a directory block
dir_block_displacement = 0;
int new_dir_block = DiskDriver_getFreeBlock(d->sfs->disk, 0);
CHECK_ERR(new_dir_block == FAILED, "full disk on directory update");
dir->header.block_in_file = d->dcb->header.block_in_file + 1;
dir->header.next_block = MISSING;
dir->header.block_in_disk = new_dir_block;
dir->header.previous_block = d->dcb->header.block_in_disk;
//we set the chain to MISSING
memset(dir->file_blocks, MISSING, sizeof(int) * Dir_Block_max_elements);
dir->file_blocks[dir_block_displacement] = file_block;
d->dcb->header.next_block = new_dir_block;
//we update the current_block
memcpy(d->current_block, &(dir->header), sizeof(BlockHeader));
res = DiskDriver_writeBlock(d->sfs->disk, dir, new_dir_block);
CHECK_ERR(res == FAILED, "can't write the 2nd directory block");
}
free(dir);
} else {
//we save the current block in disk because it could change
int current_block = d->current_block->block_in_disk;
// now we must check if in the last directory_block we have room for one more file
if (dir_block_displacement < Dir_Block_max_elements) {
//in this case we can add the file directly in the DB
dir->file_blocks[dir_block_displacement] = file_block;
} else {
int new_dir_block = DiskDriver_getFreeBlock(d->sfs->disk, 0);
CHECK_ERR(new_dir_block == FAILED, "full disk on directory update");
dir_block_displacement = 0;
DirectoryBlock *new_dir = (DirectoryBlock *) malloc(sizeof(DirectoryBlock));
new_dir->header.block_in_file = dir->header.block_in_file + 1;
new_dir->header.block_in_disk = new_dir_block;
new_dir->header.next_block = MISSING;
new_dir->header.previous_block = d->current_block->block_in_disk;
//we set the chain to MISSING
memset(new_dir->file_blocks, MISSING, sizeof(int) * Dir_Block_max_elements);
new_dir->file_blocks[dir_block_displacement] = file_block;
dir->header.next_block = new_dir_block;
//we update the current_block
memcpy(d->current_block, &(new_dir->header), sizeof(BlockHeader));
//we write the new dir_block
res = DiskDriver_writeBlock(d->sfs->disk, new_dir, new_dir_block);
free(new_dir);
if (res == FAILED) {
free(dir);
CHECK_ERR(res == FAILED, "can't write the new dir_block on disk");
}
}
//we write the updated dir_block
res = DiskDriver_writeBlock(d->sfs->disk, dir, current_block);
free(dir);
CHECK_ERR(res == FAILED, "can't write the updated dir_block on disk");
}
//we update the metedata into the FDB
d->dcb->num_entries++;
//we update the metadata into the DirectoryHandle
d->pos_in_dir++;
d->pos_in_block = dir_block_displacement;
//we write the updated FirstDirectoryBlock
res = DiskDriver_writeBlock(d->sfs->disk, d->dcb, d->dcb->header.block_in_disk);
CHECK_ERR(res == FAILED, "can't write the updated FirstDirectoryBlock on disk");
}
//remves one file, including the FCB, all the index block and all the indexes
int SimpleFS_removeFile(SimpleFS *sfs, int file) {
int i, res, stop = 0;
int FFB_max_entries =
(BLOCK_SIZE - sizeof(FileControlBlock) - sizeof(BlockHeader) - sizeof(int) - sizeof(int)) / sizeof(int);
int IB_max_entries = (BLOCK_SIZE - sizeof(BlockHeader)) / sizeof(int);
int next_IndexBlock;
//we load our FFB
FirstFileBlock *ffb = (FirstFileBlock *) malloc(sizeof(FirstFileBlock));
res = DiskDriver_readBlock(sfs->disk, ffb, file);
if (res == FAILED) {
free(ffb);
CHECK_ERR(res == FAILED, "can't read the FFB to remove");
}
//firstly we deallocate the blocks indexed in our FileControlBlock
for (i = 0; i < FFB_max_entries && !stop; i++) {
if (ffb->blocks[i] != MISSING) {
res = DiskDriver_freeBlock(sfs->disk, ffb->blocks[i]);
CHECK_ERR(res == FAILED, "can't deallocate a data block indexed in the ffb");
} else {
stop = 1;
}
}
//now we need to deallocate the ffb
res = DiskDriver_freeBlock(sfs->disk, file);
CHECK_ERR(res == FAILED, "can't deallocate the first file block")
//now before destroying our ffb in memory we check if there are other IndexBlocks
if (ffb->next_IndexBlock == MISSING) {
free(ffb);
return SUCCESS;
}
//if we get here we need to deallocate the other IndexBlocks
//we load the first IndexBlock
Index *index = (Index *) malloc(
sizeof(Index));
next_IndexBlock = ffb->next_IndexBlock;
//now we can deallocate the ffb
free(ffb);
//now we deallocate all the data blocks in the index chain
while (next_IndexBlock != MISSING) {
//if possible we load the next index block
memset(index, 0, sizeof(Index));
res = DiskDriver_readBlock(sfs->disk, index,
next_IndexBlock);
//block in file dovrebbe gia essere inizializzata
CHECK_ERR(res == FAILED, "can't load the next index block");
//we deallocate all the data block in this index block
for (i = 0; i < IB_max_entries && !stop; i++) {
if (index->indexes[i] != MISSING) {
res = DiskDriver_freeBlock(sfs->disk, index->indexes[i]);
CHECK_ERR(res == FAILED, "can't deallocate a block in the index chain");
} else {
stop = 1;
}
}
//now we deallocate the current index block
res = DiskDriver_freeBlock(sfs->disk,
index->header.block_in_disk);
CHECK_ERR(res == FAILED, "can't deallocate the current index block");
//we get the next index block
next_IndexBlock = index->header.next_block;
}
free(index);
return SUCCESS;
}
int SimpleFS_removeFileOnDir(DirectoryHandle *dh, void *element, int pos_in_block) {
// reordering the list of file after removing one file
int i, res;
int FDB_max_elements = (BLOCK_SIZE - sizeof(BlockHeader) - sizeof(FileControlBlock) - sizeof(int)) / sizeof(int);
int Dir_Block_max_elements = (BLOCK_SIZE - sizeof(BlockHeader)) / sizeof(int);
DirectoryHandle *dir = dh;
DirectoryBlock *db = (DirectoryBlock *) malloc(sizeof(DirectoryBlock));
DirectoryBlock *db1 = (DirectoryBlock *) malloc(sizeof(DirectoryBlock));
FirstFileBlock *file = (FirstFileBlock *) element;
// removing the file
SimpleFS_removeFile(dh->sfs, file->header.block_in_disk);
// if the file is in the FDB
if (dh->current_block->block_in_file == 0) {
//if the file is in the FDB
if (dir->dcb->num_entries >= FDB_max_elements) {
memset(db, 0, sizeof(DirectoryBlock));
int app = dir->dcb->header.next_block;
while (app != MISSING) {
res = DiskDriver_readBlock(dh->sfs->disk, db, app);
CHECK_ERR(res == FAILED, "Error on read of DirectoryBlock in simpleFS_remove");
app = db->header.next_block;
}
// search the last item in file_block of directory block
for (i = 0; i < Dir_Block_max_elements && db->file_blocks[i] != MISSING; i++);
dir->dcb->file_blocks[pos_in_block] = db->file_blocks[i - 1];
db->file_blocks[i - 1] = MISSING;
// We write the updates on file
res = DiskDriver_writeBlock(dh->sfs->disk, db, db->header.block_in_disk);
if (res == FAILED) return FAILED;
} else {
// search the last item in file_block of directory block
for (i = pos_in_block; i < dir->dcb->num_entries && dir->dcb->file_blocks[i] != MISSING; i++);
if (i - 1 != 0) dir->dcb->file_blocks[pos_in_block] = dir->dcb->file_blocks[i - 1];
dir->dcb->file_blocks[i - 1] = MISSING;
}
} else {
// if the file is in the Directory blocks
//we get the block in which the fie is stored
memset(db, 0, sizeof(DirectoryBlock));
res = DiskDriver_readBlock(dh->sfs->disk, db, dir->current_block->block_in_disk);
CHECK_ERR(res == FAILED, "Error on read of DirectoryBlock in simpleFS_remove");
//we get the last block with entries
memset(db1, 0, sizeof(DirectoryBlock));
int app = dir->current_block->next_block;
while (app != MISSING) {
res = DiskDriver_readBlock(dir->sfs->disk, db1, app);
CHECK_ERR(res == FAILED, "Error on read of DirectoryBlock in simpleFS_remove");
app = db1->header.next_block;
}
// we get the last occupied entry
for (i = 0; i < Dir_Block_max_elements && db1->file_blocks[i] != MISSING; i++);
db->file_blocks[pos_in_block] = db1->file_blocks[i - 1];
db1->file_blocks[i - 1] = MISSING;
if (i - 1 >= 0) {
// We write the updates on file
res = DiskDriver_writeBlock(dh->sfs->disk, db1, db1->header.block_in_disk);
CHECK_ERR(res == FAILED, "Error on write of DirectoryBlock in simpleFS_remove");
} else {
// we free the last block used if there was olny the entry we have moved
res = DiskDriver_freeBlock(dir->sfs->disk, db1->header.block_in_disk);
CHECK_ERR(res == FAILED, "Error on read of DirectoryBlock in simpleFS_remove");
//we get the new last block and update its header
res = DiskDriver_readBlock(dir->sfs->disk, db1, db1->header.previous_block);
CHECK_ERR(res == FAILED, "Error on read of DirectoryBlock in simpleFS_remove");
db1->header.next_block = MISSING;
res = DiskDriver_writeBlock(dh->sfs->disk, db1, db1->header.block_in_disk);
CHECK_ERR(res == FAILED, "Error on write of DirectoryBlock in simpleFS_remove");
}
//we write the updates on file
res = DiskDriver_writeBlock(dh->sfs->disk, db, db->header.block_in_disk);
if (res == FAILED) return FAILED;
}
free(db1);
free(db);
// we update our DCB metedata
dir->dcb->num_entries--;
// We write the updates on file
res = DiskDriver_writeBlock(dh->sfs->disk, dir->dcb, dir->dcb->header.block_in_disk);
if (res == FAILED) return FAILED;
return SUCCESS;
}
// Removes files and directories recursively
int SimpleFS_remove_rec(DirectoryHandle *handle) {
// Check if passed parameters are valid
if (handle == NULL) return FAILED;
int FDB_max_elements = (BLOCK_SIZE - sizeof(BlockHeader) - sizeof(FileControlBlock) - sizeof(int)) / sizeof(int);
int Dir_Block_max_elements = (BLOCK_SIZE - sizeof(BlockHeader)) / sizeof(int);
int i, res;
// Allocating needed structures
DirectoryBlock *db = (DirectoryBlock *) malloc(sizeof(DirectoryBlock));
FirstDirectoryBlock *fh = (FirstDirectoryBlock *) malloc(sizeof(FirstDirectoryBlock));
DirectoryHandle *hand = (DirectoryHandle *) malloc(sizeof(DirectoryBlock));
//we go to the end of the directory and traverse it backwards removing its content
while (handle->current_block->next_block != MISSING) {
memset(db, 0, sizeof(DirectoryBlock));
res = DiskDriver_readBlock(handle->sfs->disk, db, handle->current_block->next_block);
CHECK_ERR(res == FAILED, "Error while traversing the directory")
memcpy(handle->current_block, &(db->header), sizeof(BlockHeader));
}
//we traverse the directory backward removing its content
while (handle->current_block->block_in_file > 0) {
// we remove the file in the current directory block
for (i = 0; i < Dir_Block_max_elements && db->file_blocks[i] != MISSING; i++) {
memset(fh, 0, sizeof(FirstDirectoryBlock));
// we read the FDB/FFB to check whether the element is a file or a directory
res = DiskDriver_readBlock(handle->sfs->disk, fh, db->file_blocks[i]);
CHECK_ERR(res == FAILED, "error reading file in dbs in remove_rec")
if (fh->fcb.is_dir == FILE) {
// if it's a file we remove it
res = SimpleFS_removeFile(handle->sfs, db->file_blocks[i]);
CHECK_ERR(res == FAILED, "error removing file in dbs in remove_rec")
} else {
//if it's a directory we create a handle and call SimpleFS_remove_rec in it
memset(hand, 0, sizeof(DirectoryHandle));
hand->sfs = handle->sfs;
hand->dcb = fh;
hand->directory = handle->dcb;
hand->current_block = malloc(sizeof(BlockHeader));
memcpy(hand->current_block, &(fh->header), sizeof(BlockHeader));
hand->pos_in_block = 0;
hand->pos_in_dir = 0;
// we check if the directory it's empty and in that case we deallocate it directly
if (hand->dcb->num_entries == 0) {
res = DiskDriver_freeBlock(handle->sfs->disk, hand->dcb->header.block_in_disk);
CHECK_ERR(res == FAILED, "Failed to free block in simplefs_remove_rec");
} else SimpleFS_remove_rec(hand);
}
}
// we deallocate the current Directory block
res = DiskDriver_freeBlock(handle->sfs->disk, handle->current_block->block_in_disk);
CHECK_ERR(res == FAILED, "Failed to free block in simplefs_remove_rec");
// we go to the previous directory block
memset(db, 0, sizeof(DirectoryBlock));
if (handle->current_block->block_in_file > 1) {
res = DiskDriver_readBlock(handle->sfs->disk, db, handle->current_block->previous_block);
CHECK_ERR(res == FAILED, "Error while traversing the directory")
memcpy(handle->current_block, &(db->header), sizeof(BlockHeader));
} else {
memcpy(handle->current_block, &(handle->dcb->header), sizeof(BlockHeader));
}
}
// we repeat the operations done in the directory block in the FDB
for (i = 0; i < FDB_max_elements && handle->dcb->file_blocks[i] != MISSING; i++) {
memset(fh, 0, sizeof(FirstDirectoryBlock));
// we read the FDB/FFB to check whether the element is a file or a directory
res = DiskDriver_readBlock(handle->sfs->disk, fh, handle->dcb->file_blocks[i]);
CHECK_ERR(res == FAILED, "error reading file in dbs in remove_rec")
if (fh->fcb.is_dir == FILE) {
// if it's a file we remove it
res = SimpleFS_removeFile(handle->sfs, handle->dcb->file_blocks[i]);
CHECK_ERR(res == FAILED, "error removing file in dbs in remove_rec")
} else {
//if it's a directory we create a handle and call SimpleFS_remove_rec in it
memset(hand, 0, sizeof(DirectoryHandle));
hand->sfs = handle->sfs;
hand->dcb = fh;
hand->directory = handle->dcb;
hand->current_block = malloc(sizeof(BlockHeader));
memcpy(hand->current_block, &(fh->header), sizeof(BlockHeader));
hand->pos_in_block = 0;
hand->pos_in_dir = 0;
// we check if the directory it's empty and in that case we deallocate it directly
if (hand->dcb->num_entries == 0) {
res = DiskDriver_freeBlock(handle->sfs->disk, hand->dcb->header.block_in_disk);
CHECK_ERR(res == FAILED, "Failed to free block in simplefs_remove_rec");
} else SimpleFS_remove_rec(hand);
free(hand->current_block);
}
}
// now we deallocate the FDB which is now empty
res = DiskDriver_freeBlock(handle->sfs->disk, handle->dcb->header.block_in_disk);
CHECK_ERR(res == FAILED, "Failed to free block in simplefs_remove_rec");
//we cleanup all the variables allocated
free(db);
free(fh);
free(hand);
return SUCCESS;
}
// we add a block in the index list of the file returning FAILED/SUCCESS
int SimpleFS_addIndex(FileHandle *f, int block) {
//we check that we have been given valid data
if (f == NULL || block <= 0 || block >= f->sfs->disk->header->num_blocks) {
return FAILED;
}
int i, res;
int FFB_max_entries =
(BLOCK_SIZE - sizeof(FileControlBlock) - sizeof(BlockHeader) - sizeof(int) - sizeof(int)) / sizeof(int);
int IB_max_entries = (BLOCK_SIZE - sizeof(BlockHeader)) / sizeof(int);
//we use he current_index block to determine where we must put the new block
if (f->fcb->num_entries - FFB_max_entries <= 0) {
//we search for a free spot in the FFB
for (i = 0; i < FFB_max_entries; i++) {
if (f->fcb->blocks[i] == MISSING) {
f->fcb->blocks[i] = block;
f->fcb->num_entries++;
res = DiskDriver_writeBlock(f->sfs->disk, f->fcb, f->fcb->header.block_in_disk);
CHECK_ERR(res == FAILED, "can't write the update to disk");
return SUCCESS;
}
}
// if we get in here we need to allocate the 2nd index block
int new_index = DiskDriver_getFreeBlock(f->sfs->disk, 0);
CHECK_ERR(new_index == FAILED, "can't get a new index block");
Index *new_index_block = (Index *) malloc(
sizeof(Index));
new_index_block->header.block_in_disk = new_index;
new_index_block->header.next_block = MISSING;
new_index_block->header.previous_block = f->current_index_block->block_in_disk;
new_index_block->header.block_in_file = f->current_index_block->block_in_file +
1;
new_index_block->indexes[0] = block;
for (i = 1; i < IB_max_entries; i++) {
new_index_block->indexes[i] = MISSING;
}
//we update our current index block
memcpy(f->current_index_block, &(new_index_block->header), sizeof(BlockHeader));
//now we write the new index block to disk
res = DiskDriver_writeBlock(f->sfs->disk, new_index_block, new_index);
free(new_index_block);
CHECK_ERR(res == FAILED, "can't write the 2nd index block");
//we update the FFB
f->fcb->next_IndexBlock = new_index;
f->fcb->num_entries++;
res = DiskDriver_writeBlock(f->sfs->disk, f->fcb, f->fcb->header.block_in_disk);
CHECK_ERR(res == FAILED, "can't write the update to disk");
return SUCCESS;
}
Index *index = (Index *) malloc(
sizeof(Index));
//if we get here we use the index block in current_index_block to add the index
res = DiskDriver_readBlock(f->sfs->disk, index, f->current_index_block->block_in_disk);
//block_in_file gia popolata in teoria
CHECK_ERR(res == FAILED, "can't load the current index block from disk");
for (i = 0; i < IB_max_entries; i++) {
if (index->indexes[i] == MISSING) {
index->indexes[i] = block;
f->fcb->num_entries++;
res = DiskDriver_writeBlock(f->sfs->disk, index,
index->header.block_in_disk);
free(index);
CHECK_ERR(res == FAILED, "can't write the update to the current index block on disk");
res = DiskDriver_writeBlock(f->sfs->disk, f->fcb, f->fcb->header.block_in_disk);
CHECK_ERR(res == FAILED, "can't write the update to disk");
return SUCCESS;
}
}
//if we get here we need to allocate a new index block
int new_index = DiskDriver_getFreeBlock(f->sfs->disk, 0);
CHECK_ERR(new_index == FAILED, "can't get a new index block");
Index *new_index_block = (Index *) malloc(
sizeof(Index));
new_index_block->header.block_in_disk = new_index;
new_index_block->header.next_block = MISSING;
new_index_block->header.previous_block = f->current_index_block->block_in_disk;
new_index_block->header.block_in_file = f->current_index_block->block_in_file +
1;
new_index_block->indexes[0] = block;
for (i = 1; i < IB_max_entries; i++) {
new_index_block->indexes[i] = MISSING;
}
//we update our current index block
memcpy(f->current_index_block, &(new_index_block->header), sizeof(BlockHeader));
//now we write the new index block to disk
res = DiskDriver_writeBlock(f->sfs->disk, new_index_block, new_index);
free(new_index_block);
CHECK_ERR(res == FAILED, "can't write the next index block");
//we update the actual index block
index->header.next_block = new_index;
res = DiskDriver_writeBlock(f->sfs->disk, index, f->current_index_block->previous_block);
free(index);
CHECK_ERR(res == FAILED, "can't write the actual index block");
// we update the FFB
f->fcb->num_entries++;
res = DiskDriver_writeBlock(f->sfs->disk, f->fcb, f->fcb->header.block_in_disk);
CHECK_ERR(res == FAILED, "can't write the update to disk");
return SUCCESS;
}
//given the block num in the file it returns the block in disk/FAILED
int SimpleFS_getIndex(FileHandle *f, int block_in_file) {
//we check that we have been given valid data
if (f == NULL || block_in_file < 0) {
return FAILED;
}
int i, res, index_num, block_in_disk;
int FFB_max_entries =
(BLOCK_SIZE - sizeof(FileControlBlock) - sizeof(BlockHeader) - sizeof(int) - sizeof(int)) / sizeof(int);
int IB_max_entries = (BLOCK_SIZE - sizeof(BlockHeader)) / sizeof(int);
//firstly we need to get the number of the index which contains the block we need
if (block_in_file < FFB_max_entries) {
index_num = 0;
} else {
index_num = ((block_in_file - FFB_max_entries) / IB_max_entries) + 1;
}
if (index_num == 0) {
//we search in the FFB and we get the first data block if we seek at the fcb
if (block_in_file == 0) {
block_in_file = 1;
}
block_in_disk = f->fcb->blocks[block_in_file - 1];
} else {
//if we get here we need to search in the other index blocks
Index *index = (Index *) malloc(sizeof(Index));
if (f->current_index_block->block_in_file < index_num) {
//we search for another index block after the current block
for (i = 0; i < index_num; i++) {
//we check if we have another index block
if (f->current_index_block->next_block == MISSING) {
free(index);
return FAILED;
}
//we read the index
res = DiskDriver_readBlock(f->sfs->disk, index, f->current_index_block->next_block);
if (res == FAILED) {
free(index);
CHECK_ERR(res == FAILED, "can't read the next index block");
}
//we update our current index block
memcpy(f->current_index_block, &(index->header), sizeof(BlockHeader));
}
} else {
if (f->current_index_block->block_in_file > index_num) {
//we search for another index block before the current block
for (i = 0; i < index_num; i++) {
//we check if we have another index block
if (f->current_index_block->previous_block == MISSING) {
free(index);
return FAILED;
}
//we read the index
res = DiskDriver_readBlock(f->sfs->disk, index, f->current_index_block->previous_block);
if (res == FAILED) {
free(index);
CHECK_ERR(res == FAILED, "can't read the next index block");
}
//we update our current index block
memcpy(f->current_index_block, &(index->header), sizeof(BlockHeader));
}
} else {
//if we get here we need to read only the current index block
res = DiskDriver_readBlock(f->sfs->disk, index, f->current_index_block->block_in_disk);
if (res == FAILED) {
free(index);
CHECK_ERR(res == FAILED, "can't read the next index block");
}
}
}
//now we need to get the index from the found block
block_in_disk = index->indexes[(block_in_file - FFB_max_entries) -
IB_max_entries * (f->current_index_block->block_in_file - 1) - 1];
free(index);
}
return block_in_disk;
}
// this function clears the index list starting from the provided block in file
void SimpleFS_clearIndexes(FileHandle *f, int block_in_file) {
//we check that we have been given valid data
if (f == NULL || block_in_file <= 0) {
return;
}
int i, res, stop = 0;
int displacement;
int FFB_max_entries =
(BLOCK_SIZE - sizeof(FileControlBlock) - sizeof(BlockHeader) - sizeof(int) - sizeof(int)) / sizeof(int);
int IB_max_entries = (BLOCK_SIZE - sizeof(BlockHeader)) / sizeof(int);
//we seek the block to adjust the current_index_block
res = SimpleFS_getIndex(f, block_in_file);
CHECK_ERR(res == FAILED, "can't clear find the first block to clear from indexes");
Index *indexblock = (Index *) malloc(
sizeof(Index));
//c'è la read dovrebbe essere apposto block_in_file
//if we have the index in the FFB we start clearing from there
if (f->fcb->num_entries - FFB_max_entries <= 0) {
displacement = block_in_file - 1;
for (i = displacement; i < FFB_max_entries && !stop; i++) {
if (f->fcb->blocks[i] == MISSING) {
stop = 1;
} else {
res = DiskDriver_freeBlock(f->sfs->disk, f->fcb->blocks[i]);
CHECK_ERR(res == FAILED, "can't free a block from disk");
f->fcb->blocks[i] = MISSING;
}
}
} else {
//we load the current index block
res = DiskDriver_readBlock(f->sfs->disk, indexblock, f->current_index_block->block_in_disk);
CHECK_ERR(res == FAILED, "can't load the current index block");
displacement =
block_in_file - FFB_max_entries - IB_max_entries * (f->current_index_block->block_in_file - 1) - 1;
for (i = displacement; i < IB_max_entries && !stop; i++) {
if (indexblock->indexes[i] == MISSING) {
stop = 1;
} else {
res = DiskDriver_freeBlock(f->sfs->disk, indexblock->indexes[i]);
CHECK_ERR(res == FAILED, "can't free a block from disk");
indexblock->indexes[i] = MISSING;
}
}
//we will remove any index blocks after the current block
indexblock->header.next_block = MISSING;
//we write the update on disk
res = DiskDriver_writeBlock(f->sfs->disk, indexblock,
indexblock->header.block_in_disk);
CHECK_ERR(res == FAILED, "can't write the index block on disk");
}
//now we update the number of entries
if (f->fcb->num_entries - FFB_max_entries <= 0) {
f->fcb->num_entries = displacement - 1;
} else {
f->fcb->num_entries = f->current_index_block->block_in_file * IB_max_entries + displacement - 1;
}
//now we clear the eventual next index block which aren't used
while (f->current_index_block->next_block != MISSING) {
res = DiskDriver_readBlock(f->sfs->disk, indexblock, f->current_index_block->next_block);
CHECK_ERR(res == FAILED, "can't load the current index block");
for (i = 0; i < IB_max_entries; i++) {
if (indexblock->indexes[i] != MISSING) {
res = DiskDriver_freeBlock(f->sfs->disk, indexblock->indexes[i]);
CHECK_ERR(res == FAILED, "can't free a block from disk");
}
}
res = DiskDriver_freeBlock(f->sfs->disk,
indexblock->header.block_in_disk);
CHECK_ERR(res == FAILED, "can't deallocate the index block")
//we advance our current_index_block
memcpy(f->current_index_block, &(indexblock->header), sizeof(BlockHeader));
}
free(indexblock);
}