1 /*
2  * COPYRIGHT:        See COPYRIGHT.TXT
3  * PROJECT:          Ext2 File System Driver for WinNT/2K/XP
4  * FILE:             indirect.c
5  * PROGRAMMER:       Matt Wu <mattwu@163.com>
6  * HOMEPAGE:         http://www.ext2fsd.com
7  * UPDATE HISTORY:
8  */
9 
10 /* INCLUDES *****************************************************************/
11 
12 #include "ext2fs.h"
13 
14 /* GLOBALS *****************************************************************/
15 
16 extern PEXT2_GLOBAL Ext2Global;
17 
18 /* DEFINITIONS *************************************************************/
19 
20 #ifdef ALLOC_PRAGMA
21 #endif
22 
23 
24 NTSTATUS
Ext2ExpandLast(IN PEXT2_IRP_CONTEXT IrpContext,IN PEXT2_VCB Vcb,IN PEXT2_MCB Mcb,IN ULONG Base,IN ULONG Layer,IN PULONG * Data,IN PULONG Hint,IN PULONG Block,IN OUT PULONG Number)25 Ext2ExpandLast(
26     IN PEXT2_IRP_CONTEXT    IrpContext,
27     IN PEXT2_VCB            Vcb,
28     IN PEXT2_MCB            Mcb,
29     IN ULONG                Base,
30     IN ULONG                Layer,
31     IN PULONG *             Data,
32     IN PULONG               Hint,
33     IN PULONG               Block,
34     IN OUT PULONG           Number
35 )
36 {
37     PULONG      pData = NULL;
38     ULONG       i;
39     NTSTATUS    Status = STATUS_SUCCESS;
40 
41     if (Layer > 0 || IsMcbDirectory(Mcb)) {
42 
43         /* allocate buffer for new block */
44         pData = (ULONG *) Ext2AllocatePool(
45                     PagedPool,
46                     BLOCK_SIZE,
47                     EXT2_DATA_MAGIC
48                 );
49         if (!pData) {
50             DEBUG(DL_ERR, ( "Ex2ExpandBlock: failed to allocate memory for Data.\n"));
51             Status = STATUS_INSUFFICIENT_RESOURCES;
52             goto errorout;
53         }
54         RtlZeroMemory(pData, BLOCK_SIZE);
55         INC_MEM_COUNT(PS_BLOCK_DATA, pData, BLOCK_SIZE);
56     }
57 
58     /* allocate block from disk */
59     Status = Ext2NewBlock(
60                  IrpContext,
61                  Vcb,
62                  (Mcb->Inode.i_ino - 1) / BLOCKS_PER_GROUP,
63                  *Hint,
64                  Block,
65                  Number
66              );
67 
68     if (!NT_SUCCESS(Status)) {
69         goto errorout;
70     }
71 
72     /* increase inode i_blocks */
73     Mcb->Inode.i_blocks += (*Number << (BLOCK_BITS - 9));
74 
75     if (Layer == 0) {
76 
77         if (IsMcbDirectory(Mcb)) {
78             /* for directory we need initialize it's entry structure */
79             PEXT2_DIR_ENTRY2 pEntry;
80             pEntry = (PEXT2_DIR_ENTRY2) pData;
81             pEntry->rec_len = (USHORT)(BLOCK_SIZE);
82             ASSERT(*Number == 1);
83             Ext2SaveBlock(IrpContext, Vcb, *Block, (PVOID)pData);
84         }
85 
86         /* add new Extent into Mcb */
87         if (!Ext2AddBlockExtent(Vcb, Mcb, Base, (*Block), *Number)) {
88             DbgBreak();
89             ClearFlag(Mcb->Flags, MCB_ZONE_INITED);
90             Ext2ClearAllExtents(&Mcb->Extents);
91         }
92 
93     } else {
94 
95         /* zero the content of all meta blocks */
96         for (i = 0; i < *Number; i++) {
97             Ext2SaveBlock(IrpContext, Vcb, *Block + i, (PVOID)pData);
98             /* add block to meta extents */
99             if (!Ext2AddMcbMetaExts(Vcb, Mcb, *Block + i, 1)) {
100                 DbgBreak();
101                 Ext2Sleep(500);
102                 Ext2AddMcbMetaExts(Vcb, Mcb, *Block + i, 1);
103             }
104         }
105     }
106 
107 errorout:
108 
109     if (NT_SUCCESS(Status)) {
110         *Hint = *Block + *Number;
111         if (Data) {
112             *Data = pData;
113             ASSERT(*Number == 1);
114         } else {
115             if (pData) {
116                 Ext2FreePool(pData, EXT2_DATA_MAGIC);
117                 DEC_MEM_COUNT(PS_BLOCK_DATA, pData, BLOCK_SIZE);
118             }
119         }
120     } else {
121         if (pData) {
122             Ext2FreePool(pData, EXT2_DATA_MAGIC);
123             DEC_MEM_COUNT(PS_BLOCK_DATA, pData, BLOCK_SIZE);
124         }
125         if (*Block) {
126             Ext2FreeBlock(IrpContext, Vcb, *Block, *Number);
127             Mcb->Inode.i_blocks -= (*Number << (BLOCK_BITS - 9));
128             *Block = 0;
129         }
130     }
131 
132     return Status;
133 }
134 
135 NTSTATUS
Ext2GetBlock(IN PEXT2_IRP_CONTEXT IrpContext,IN PEXT2_VCB Vcb,IN PEXT2_MCB Mcb,IN ULONG Base,IN ULONG Layer,IN ULONG Start,IN ULONG SizeArray,IN PULONG BlockArray,IN BOOLEAN bAlloc,IN OUT PULONG Hint,OUT PULONG Block,OUT PULONG Number)136 Ext2GetBlock(
137     IN PEXT2_IRP_CONTEXT    IrpContext,
138     IN PEXT2_VCB            Vcb,
139     IN PEXT2_MCB            Mcb,
140     IN ULONG                Base,
141     IN ULONG                Layer,
142     IN ULONG                Start,
143     IN ULONG                SizeArray,
144     IN PULONG               BlockArray,
145     IN BOOLEAN              bAlloc,
146     IN OUT PULONG           Hint,
147     OUT PULONG              Block,
148     OUT PULONG              Number
149 )
150 {
151     NTSTATUS    Status = STATUS_SUCCESS;
152     PBCB        Bcb = NULL;
153     PULONG      pData = NULL;
154     ULONG       Slot = 0, i = 0;
155     ULONG       Unit = 1;
156 
157     LARGE_INTEGER Offset;
158 
159     if (Layer == 0) {
160 
161         *Number = 1;
162         if (BlockArray[0] == 0 && bAlloc) {
163 
164             /* now allocate new block */
165             Status = Ext2ExpandLast(
166                          IrpContext,
167                          Vcb,
168                          Mcb,
169                          Base,
170                          Layer,
171                          NULL,
172                          Hint,
173                          &BlockArray[0],
174                          Number
175                      );
176 
177             if (!NT_SUCCESS(Status)) {
178                 goto errorout;
179             }
180         } else {
181             /* check the block is valid or not */
182             if (BlockArray[0] >= TOTAL_BLOCKS) {
183                 DbgBreak();
184                 Status = STATUS_DISK_CORRUPT_ERROR;
185                 goto errorout;
186             }
187         }
188 
189         *Block = BlockArray[0];
190         for (i=1; i < SizeArray; i++) {
191             if (BlockArray[i] == BlockArray[i-1] + 1) {
192                 *Number = *Number + 1;
193             } else {
194                 break;
195             }
196         }
197         *Hint = BlockArray[*Number - 1];
198 
199     } else if (Layer <= 3) {
200 
201         /* check the block is valid or not */
202         if (BlockArray[0] == 0 || BlockArray[0] >= TOTAL_BLOCKS) {
203             DbgBreak();
204             Status = STATUS_DISK_CORRUPT_ERROR;
205             goto errorout;
206         }
207 
208         /* add block to meta extents */
209         if (!Ext2AddMcbMetaExts(Vcb, Mcb, BlockArray[0], 1)) {
210             DbgBreak();
211             Ext2Sleep(500);
212             Ext2AddMcbMetaExts(Vcb, Mcb, BlockArray[0], 1);
213         }
214 
215         /* map memory in cache for the index block */
216         Offset.QuadPart = ((LONGLONG)BlockArray[0]) << BLOCK_BITS;
217         if ( !CcPinRead( Vcb->Volume,
218                          (PLARGE_INTEGER) (&Offset),
219                          BLOCK_SIZE,
220                          PIN_WAIT,
221                          &Bcb,
222 #ifdef __REACTOS__
223                          (void **)&pData )) {
224 #else
225                          &pData )) {
226 #endif
227 
228             DEBUG(DL_ERR, ( "Ext2GetBlock: Failed to PinLock block: %xh ...\n",
229                             BlockArray[0] ));
230             Status = STATUS_CANT_WAIT;
231             goto errorout;
232         }
233 
234         if (Layer > 1) {
235             Unit = Vcb->max_blocks_per_layer[Layer - 1];
236         } else {
237             Unit = 1;
238         }
239 
240         Slot  = Start / Unit;
241         Start = Start % Unit;
242 
243         if (pData[Slot] == 0) {
244 
245             if (bAlloc) {
246 
247                 /* we need allocate new block and zero all data in case
248                    it's an in-direct block. Index stores the new block no. */
249                 ULONG   Count = 1;
250                 Status = Ext2ExpandLast(
251                              IrpContext,
252                              Vcb,
253                              Mcb,
254                              Base,
255                              Layer,
256                              NULL,
257                              Hint,
258                              &pData[Slot],
259                              &Count
260                          );
261 
262                 if (!NT_SUCCESS(Status)) {
263                     goto errorout;
264                 }
265 
266                 /* refresh hint block */
267                 *Hint = pData[Slot];
268 
269                 /* set dirty bit to notify system to flush */
270                 CcSetDirtyPinnedData(Bcb, NULL );
271                 SetFlag(Vcb->Volume->Flags, FO_FILE_MODIFIED);
272                 if (!Ext2AddVcbExtent(Vcb, Offset.QuadPart,
273                                       (LONGLONG)BLOCK_SIZE)) {
274                     DbgBreak();
275                     Ext2Sleep(100);
276                     if (!Ext2AddVcbExtent(Vcb, Offset.QuadPart,
277                                           (LONGLONG)BLOCK_SIZE)) {
278                         Status = STATUS_INSUFFICIENT_RESOURCES;
279                         goto errorout;
280                     }
281                 }
282 
283                 /* save inode information here */
284                 Ext2SaveInode(IrpContext, Vcb, &Mcb->Inode);
285 
286             } else {
287 
288                 *Number = 1;
289 
290                 if (Layer == 1) {
291                     for (i = Slot + 1; i < BLOCK_SIZE/4; i++) {
292                         if (pData[i] == 0) {
293                             *Number = *Number + 1;
294                         } else {
295                             break;
296                         }
297                     }
298                 } else if (Layer == 2) {
299                     *Number = BLOCK_SIZE/4 - Start;
300                 } else {
301                     *Number = BLOCK_SIZE/4;
302                 }
303 
304                 goto errorout;
305             }
306         }
307 
308         /* transfer to next recursion call */
309         Status = Ext2GetBlock(
310                      IrpContext,
311                      Vcb,
312                      Mcb,
313                      Base,
314                      Layer - 1,
315                      Start,
316                      BLOCK_SIZE/4 - Slot,
317                      &pData[Slot],
318                      bAlloc,
319                      Hint,
320                      Block,
321                      Number
322                  );
323 
324         if (!NT_SUCCESS(Status)) {
325             goto errorout;
326         }
327     }
328 
329 errorout:
330 
331     /* free the memory of pData */
332     if (Bcb) {
333         CcUnpinData(Bcb);
334     }
335 
336     if (!NT_SUCCESS(Status)) {
337         *Block = 0;
338     }
339 
340     return Status;
341 }
342 
343 
344 NTSTATUS
345 Ext2ExpandBlock(
346     IN PEXT2_IRP_CONTEXT IrpContext,
347     IN PEXT2_VCB         Vcb,
348     IN PEXT2_MCB         Mcb,
349     IN ULONG             Base,
350     IN ULONG             Layer,
351     IN ULONG             Start,
352     IN ULONG             SizeArray,
353     IN PULONG            BlockArray,
354     IN PULONG            Hint,
355     IN PULONG            Extra
356 )
357 {
358     ULONG       i = 0;
359     ULONG       j;
360     ULONG       Slot;
361     ULONG       Block = 0;
362     LARGE_INTEGER Offset;
363 
364     PBCB        Bcb = NULL;
365     PULONG      pData = NULL;
366     ULONG       Skip = 0;
367 
368     ULONG       Number;
369     ULONG       Wanted;
370 
371     NTSTATUS    Status = STATUS_SUCCESS;
372 
373     if (Layer == 1) {
374 
375         /*
376          * try to make all leaf block continuous to avoid fragments
377          */
378 
379         Number = min(SizeArray, ((*Extra + (Start & (BLOCK_SIZE/4 - 1))) * 4 / BLOCK_SIZE));
380         Wanted = 0;
381         DEBUG(DL_BLK, ("Ext2ExpandBlock: SizeArray=%xh Extra=%xh Start=%xh %xh\n",
382                        SizeArray, *Extra, Start, Number ));
383 
384         for (i=0; i < Number; i++) {
385             if (BlockArray[i] == 0) {
386                 Wanted += 1;
387             }
388         }
389 
390         i = 0;
391         while (Wanted > 0) {
392 
393             Number = Wanted;
394             Status = Ext2ExpandLast(
395                          IrpContext,
396                          Vcb,
397                          Mcb,
398                          Base,
399                          Layer,
400                          NULL,
401                          Hint,
402                          &Block,
403                          &Number
404                      );
405             if (!NT_SUCCESS(Status)) {
406                 goto errorout;
407             }
408 
409             ASSERT(Number > 0);
410             Wanted -= Number;
411             while (Number) {
412                 if (BlockArray[i] == 0) {
413                     BlockArray[i] = Block++;
414                     Number--;
415                 }
416                 i++;
417             }
418         }
419 
420     } else if (Layer == 0) {
421 
422         /*
423          * bulk allocation for inode data blocks
424          */
425 
426         i = 0;
427 
428         while (*Extra && i < SizeArray) {
429 
430             Wanted = 0;
431             Number = 1;
432 
433             for (j = i; j < SizeArray && j < i + *Extra; j++) {
434 
435                 if (BlockArray[j] >= TOTAL_BLOCKS) {
436                     DbgBreak();
437                     BlockArray[j] = 0;
438                 }
439 
440                 if (BlockArray[j] == 0) {
441                     Wanted += 1;
442                 } else {
443                     break;
444                 }
445             }
446 
447             if (Wanted == 0) {
448 
449                 /* add block extent into Mcb */
450                 ASSERT(BlockArray[i] != 0);
451                 if (!Ext2AddBlockExtent(Vcb, Mcb, Base + i, BlockArray[i], 1)) {
452                     DbgBreak();
453                     ClearFlag(Mcb->Flags, MCB_ZONE_INITED);
454                     Ext2ClearAllExtents(&Mcb->Extents);
455                 }
456 
457             } else {
458 
459                 Number = Wanted;
460                 Status = Ext2ExpandLast(
461                              IrpContext,
462                              Vcb,
463                              Mcb,
464                              Base + i,
465                              0,
466                              NULL,
467                              Hint,
468                              &Block,
469                              &Number
470                          );
471                 if (!NT_SUCCESS(Status)) {
472                     goto errorout;
473                 }
474 
475                 ASSERT(Number > 0);
476                 for (j = 0; j < Number; j++) {
477                     BlockArray[i + j] = Block++;
478                 }
479             }
480 
481             *Extra -= Number;
482             i += Number;
483         }
484 
485         goto errorout;
486     }
487 
488 
489     /*
490      * only for meta blocks allocation
491      */
492 
493     for (i = 0; *Extra && i < SizeArray; i++) {
494 
495         if (Layer <= 3) {
496 
497             if (BlockArray[i] >= TOTAL_BLOCKS) {
498                 DbgBreak();
499                 BlockArray[i] = 0;
500             }
501 
502             if (BlockArray[i] == 0) {
503                 Number = 1;
504                 Status = Ext2ExpandLast(
505                              IrpContext,
506                              Vcb,
507                              Mcb,
508                              Base,
509                              Layer,
510                              &pData,
511                              Hint,
512                              &BlockArray[i],
513                              &Number
514                          );
515                 if (!NT_SUCCESS(Status)) {
516                     goto errorout;
517                 }
518 
519             } else {
520 
521                 Offset.QuadPart = (((LONGLONG)BlockArray[i]) << BLOCK_BITS);
522                 if (!CcPinRead(
523                             Vcb->Volume,
524                             &Offset,
525                             BLOCK_SIZE,
526                             PIN_WAIT,
527                             &Bcb,
528 #ifdef __REACTOS__
529                             (void **)&pData )) {
530 #else
531                             &pData )) {
532 #endif
533 
534                     DEBUG(DL_ERR, ( "Ext2ExpandInode: failed to PinLock offset :%I64xh...\n",
535                                     Offset.QuadPart));
536                     Status = STATUS_CANT_WAIT;
537                     DbgBreak();
538                     goto errorout;
539                 }
540 
541                 /* add block to meta extents */
542                 if (!Ext2AddMcbMetaExts(Vcb, Mcb,  BlockArray[i], 1)) {
543                     DbgBreak();
544                     Ext2Sleep(500);
545                     Ext2AddMcbMetaExts(Vcb, Mcb,  BlockArray[i], 1);
546                 }
547             }
548 
549             Skip = Vcb->max_blocks_per_layer[Layer] * i;
550 
551             if (i == 0) {
552                 if (Layer > 1) {
553                     Slot  = Start / Vcb->max_blocks_per_layer[Layer - 1];
554                     Start = Start % Vcb->max_blocks_per_layer[Layer - 1];
555                     Skip += Slot * Vcb->max_blocks_per_layer[Layer - 1];
556                 } else {
557                     Slot  = Start;
558                     Start = 0;
559                     Skip += Slot;
560                 }
561             } else {
562                 Start = 0;
563                 Slot  = 0;
564             }
565 
566             Status = Ext2ExpandBlock(
567                          IrpContext,
568                          Vcb,
569                          Mcb,
570                          Base + Skip,
571                          Layer - 1,
572                          Start,
573                          BLOCK_SIZE/4 - Slot,
574                          &pData[Slot],
575                          Hint,
576                          Extra
577                      );
578 
579             if (Bcb) {
580                 CcSetDirtyPinnedData(Bcb, NULL);
581                 if (!Ext2AddBlockExtent(Vcb, NULL,
582                                         BlockArray[i],
583                                         BlockArray[i], 1)) {
584                     DbgBreak();
585                     Ext2Sleep(500);
586                     if (!Ext2AddBlockExtent(Vcb, NULL,
587                                             BlockArray[i],
588                                             BlockArray[i], 1)) {
589                     }
590                 }
591             } else {
592                 Ext2SaveBlock(IrpContext, Vcb, BlockArray[i], (PVOID)pData);
593             }
594 
595             if (pData) {
596                 if (Bcb) {
597                     CcUnpinData(Bcb);
598                     Bcb = NULL;
599                 } else {
600                     Ext2FreePool(pData, EXT2_DATA_MAGIC);
601                     DEC_MEM_COUNT(PS_BLOCK_DATA, pData, BLOCK_SIZE);
602                 }
603                 pData = NULL;
604             }
605 
606             if (!NT_SUCCESS(Status)) {
607                 DbgBreak();
608                 break;
609             }
610         }
611     }
612 
613 errorout:
614 
615     return Status;
616 }
617 
618 BOOLEAN
619 Ext2IsBlockEmpty(PULONG BlockArray, ULONG SizeArray)
620 {
621     ULONG i = 0;
622     for (i=0; i < SizeArray; i++) {
623         if (BlockArray[i]) {
624             break;
625         }
626     }
627     return (i == SizeArray);
628 }
629 
630 
631 NTSTATUS
632 Ext2TruncateBlock(
633     IN PEXT2_IRP_CONTEXT IrpContext,
634     IN PEXT2_VCB         Vcb,
635     IN PEXT2_MCB         Mcb,
636     IN ULONG             Base,
637     IN ULONG             Start,
638     IN ULONG             Layer,
639     IN ULONG             SizeArray,
640     IN PULONG            BlockArray,
641     IN PULONG            Extra
642 )
643 {
644     NTSTATUS    Status = STATUS_SUCCESS;
645     ULONG       i = 0;
646     ULONG       Slot = 0;
647     ULONG       Skip = 0;
648 
649     LONGLONG    Offset;
650     PBCB        Bcb = NULL;
651     PULONG      pData = NULL;
652 
653     ASSERT(Mcb != NULL);
654 
655     for (i = 0; i < SizeArray; i++) {
656 
657         if (Layer == 0) {
658 
659             ULONG   Number = 1;
660 
661             while (Extra &&  SizeArray > i + 1 && Number < *Extra) {
662 
663                 if (BlockArray[SizeArray - i - 1] ==
664                     BlockArray[SizeArray - i - 2] + 1) {
665 
666                     BlockArray[SizeArray - i - 1] = 0;
667                     Number++;
668                     SizeArray--;
669 
670                 } else {
671                     break;
672                 }
673             }
674 
675             if (BlockArray[SizeArray - i - 1]) {
676 
677                 Status = Ext2FreeBlock(IrpContext, Vcb, BlockArray[SizeArray - i - 1], Number);
678                 if (NT_SUCCESS(Status)) {
679                     ASSERT(Mcb->Inode.i_blocks >= (Number << (BLOCK_BITS - 9)));
680                     if (Mcb->Inode.i_blocks < (Number << (BLOCK_BITS - 9))) {
681                         Mcb->Inode.i_blocks = 0;
682                         DbgBreak();
683                     } else {
684                         Mcb->Inode.i_blocks -= (Number << (BLOCK_BITS - 9));
685                     }
686                     BlockArray[SizeArray - i - 1] = 0;
687                 }
688             }
689 
690             if (Extra) {
691 
692                 /* dec blocks count */
693                 ASSERT(*Extra >= Number);
694                 *Extra = *Extra - Number;
695 
696                 /* remove block mapping frm Mcb Extents */
697                 if (!Ext2RemoveBlockExtent(Vcb, Mcb, Base + SizeArray - 1 - i, Number)) {
698                     DbgBreak();
699                     ClearFlag(Mcb->Flags, MCB_ZONE_INITED);
700                     Ext2ClearAllExtents(&Mcb->Extents);
701                 }
702             }
703 
704         } else {
705 
706             ASSERT(Layer <= 3);
707 
708             if (BlockArray[SizeArray - i - 1] >= TOTAL_BLOCKS) {
709                 DbgBreak();
710                 BlockArray[SizeArray - i - 1] = 0;
711             }
712 
713             if (i == 0) {
714                 if (Layer > 1) {
715                     Slot  = Start / Vcb->max_blocks_per_layer[Layer - 1];
716                     Start = Start % Vcb->max_blocks_per_layer[Layer - 1];
717                 } else {
718                     Slot  = Start;
719                     Start = (BLOCK_SIZE / 4) - 1;
720                 }
721             } else {
722                 Slot = Start = (BLOCK_SIZE / 4) - 1;
723             }
724 
725             Skip = (SizeArray - i - 1) * Vcb->max_blocks_per_layer[Layer];
726 
727             if (BlockArray[SizeArray - i - 1]) {
728 
729                 Offset = (LONGLONG) (BlockArray[SizeArray - i - 1]);
730                 Offset = Offset << BLOCK_BITS;
731 
732                 if (!CcPinRead( Vcb->Volume,
733                                 (PLARGE_INTEGER) (&Offset),
734                                 BLOCK_SIZE,
735                                 PIN_WAIT,
736                                 &Bcb,
737 #ifdef __REACTOS__
738                                 (void **)&pData )) {
739 #else
740                                 &pData )) {
741 #endif
742 
743                     DEBUG(DL_ERR, ( "Ext2TruncateBlock: PinLock failed on block %xh ...\n",
744                                     BlockArray[SizeArray - i - 1]));
745                     Status = STATUS_CANT_WAIT;
746                     DbgBreak();
747                     goto errorout;
748                 }
749 
750                 Status = Ext2TruncateBlock(
751                              IrpContext,
752                              Vcb,
753                              Mcb,
754                              Base + Skip,
755                              Start,
756                              Layer - 1,
757                              Slot + 1,
758                              &pData[0],
759                              Extra
760                          );
761 
762                 if (!NT_SUCCESS(Status)) {
763                     break;
764                 }
765 
766                 CcSetDirtyPinnedData(Bcb, NULL);
767                 Ext2AddVcbExtent(Vcb, Offset, (LONGLONG)BLOCK_SIZE);
768 
769                 if (*Extra || Ext2IsBlockEmpty(pData, BLOCK_SIZE/4)) {
770 
771                     Ext2TruncateBlock(
772                                  IrpContext,
773                                  Vcb,
774                                  Mcb,
775                                  Base + Skip,    /* base */
776                                  0,              /* start */
777                                  0,              /* layer */
778                                  1,
779                                  &BlockArray[SizeArray - i - 1],
780                                  NULL
781                              );
782 
783                     if (!Ext2RemoveMcbMetaExts(Vcb, Mcb, BlockArray[SizeArray - i - 1], 1)) {
784                         DbgBreak();
785                         Ext2Sleep(500);
786                         Ext2RemoveMcbMetaExts(Vcb, Mcb, BlockArray[SizeArray - i - 1], 1);
787                     }
788                 }
789 
790                 if (pData) {
791                     CcUnpinData(Bcb);
792                     Bcb = NULL;
793                     pData = NULL;
794                 }
795 
796             } else {
797 
798                 if (Layer > 1) {
799                     if (*Extra > Slot * Vcb->max_blocks_per_layer[Layer - 1] + Start + 1) {
800                         *Extra -= (Slot * Vcb->max_blocks_per_layer[Layer - 1] + Start + 1);
801                     } else {
802                         *Extra  = 0;
803                     }
804                 } else {
805                     if (*Extra > Slot + 1) {
806                         *Extra -= (Slot + 1);
807                     } else {
808                         *Extra  = 0;
809                     }
810                 }
811 
812                 if (!Ext2RemoveBlockExtent(Vcb, Mcb, Base + Skip, (Start + 1))) {
813                     DbgBreak();
814                     ClearFlag(Mcb->Flags, MCB_ZONE_INITED);
815                     Ext2ClearAllExtents(&Mcb->Extents);
816                 }
817             }
818         }
819 
820         if (Extra && *Extra == 0) {
821             break;
822         }
823     }
824 
825 errorout:
826 
827     if (pData) {
828         CcUnpinData(Bcb);
829     }
830 
831     return Status;
832 }
833 
834 NTSTATUS
835 Ext2MapIndirect(
836     IN PEXT2_IRP_CONTEXT    IrpContext,
837     IN PEXT2_VCB            Vcb,
838     IN PEXT2_MCB            Mcb,
839     IN ULONG                Index,
840     IN BOOLEAN              bAlloc,
841     OUT PULONG              pBlock,
842     OUT PULONG              Number
843 )
844 {
845     ULONG   Layer;
846     ULONG   Slot;
847 
848     ULONG   Base = Index;
849 
850     NTSTATUS Status = STATUS_SUCCESS;
851 
852     *pBlock = 0;
853     *Number = 0;
854 
855     for (Layer = 0; Layer < EXT2_BLOCK_TYPES; Layer++) {
856 
857         if (Index < Vcb->max_blocks_per_layer[Layer]) {
858 
859             ULONG   dwRet = 0, dwBlk = 0, dwHint = 0, dwArray = 0;
860 
861             Slot = (Layer==0) ? (Index):(Layer + EXT2_NDIR_BLOCKS - 1);
862             dwBlk = Mcb->Inode.i_block[Slot];
863 
864             if (dwBlk == 0) {
865 
866                 if (!bAlloc) {
867 
868                     *Number = 1;
869                     goto errorout;
870 
871                 } else {
872 
873                     if (Slot) {
874                         dwHint = Mcb->Inode.i_block[Slot - 1];
875                     }
876 
877                     /* allocate and zero block if necessary */
878                     *Number = 1;
879                     Status = Ext2ExpandLast(
880                                  IrpContext,
881                                  Vcb,
882                                  Mcb,
883                                  Base,
884                                  Layer,
885                                  NULL,
886                                  &dwHint,
887                                  &dwBlk,
888                                  Number
889                              );
890 
891                     if (!NT_SUCCESS(Status)) {
892                         goto errorout;
893                     }
894 
895                     /* save the it into inode*/
896                     Mcb->Inode.i_block[Slot] = dwBlk;
897 
898                     /* save the inode */
899                     if (!Ext2SaveInode(IrpContext, Vcb, &Mcb->Inode)) {
900                         DbgBreak();
901                         Status = STATUS_UNSUCCESSFUL;
902                         goto errorout;
903                     }
904                 }
905             }
906 
907             if (Layer == 0)
908                 dwArray = Vcb->max_blocks_per_layer[Layer] - Index;
909             else
910                 dwArray = 1;
911 
912             /* querying block number of the index-th file block */
913             Status = Ext2GetBlock(
914                          IrpContext,
915                          Vcb,
916                          Mcb,
917                          Base,
918                          Layer,
919                          Index,
920                          dwArray,
921 #ifdef __REACTOS__
922                         (PULONG)&Mcb->Inode.i_block[Slot],
923 #else
924                         &Mcb->Inode.i_block[Slot],
925 #endif
926                          bAlloc,
927                          &dwHint,
928                          &dwRet,
929                          Number
930                      );
931 
932             if (NT_SUCCESS(Status)) {
933                 *pBlock = dwRet;
934             }
935 
936             break;
937         }
938 
939         Index -= Vcb->max_blocks_per_layer[Layer];
940     }
941 
942 errorout:
943 
944     return Status;
945 }
946 
947 NTSTATUS
948 Ext2ExpandIndirect(
949     PEXT2_IRP_CONTEXT IrpContext,
950     PEXT2_VCB         Vcb,
951     PEXT2_MCB         Mcb,
952     ULONG             Start,
953     ULONG             End,
954     PLARGE_INTEGER    Size
955 )
956 {
957     NTSTATUS Status = STATUS_SUCCESS;
958 
959     ULONG    Layer = 0;
960     ULONG    Extra = 0;
961     ULONG    Hint = 0;
962     ULONG    Slot = 0;
963     ULONG    Base = 0;
964 
965     Extra = End - Start;
966 
967 	/* exceeds the biggest file size (indirect) */
968     if (End > Vcb->max_data_blocks) {
969         return STATUS_INVALID_PARAMETER;
970     }
971 
972     for (Layer = 0; Layer < EXT2_BLOCK_TYPES && Extra; Layer++) {
973 
974         if (Start >= Vcb->max_blocks_per_layer[Layer]) {
975 
976             Base  += Vcb->max_blocks_per_layer[Layer];
977             Start -= Vcb->max_blocks_per_layer[Layer];
978 
979         } else {
980 
981             /* get the slot in i_block array */
982             if (Layer == 0) {
983                 Base = Slot = Start;
984             } else {
985                 Slot = Layer + EXT2_NDIR_BLOCKS - 1;
986             }
987 
988             /* set block hint to avoid fragments */
989             if (Hint == 0) {
990                 if (Mcb->Inode.i_block[Slot] != 0) {
991                     Hint = Mcb->Inode.i_block[Slot];
992                 } else if (Slot > 1) {
993                     Hint = Mcb->Inode.i_block[Slot-1];
994                 }
995             }
996 
997             /* now expand this slot */
998             Status = Ext2ExpandBlock(
999                          IrpContext,
1000                          Vcb,
1001                          Mcb,
1002                          Base,
1003                          Layer,
1004                          Start,
1005                          (Layer == 0) ? (Vcb->max_blocks_per_layer[Layer] - Start) : 1,
1006 #ifdef __REACTOS__
1007                          (PULONG)&Mcb->Inode.i_block[Slot],
1008 #else
1009                          &Mcb->Inode.i_block[Slot],
1010 #endif
1011                          &Hint,
1012                          &Extra
1013                      );
1014             if (!NT_SUCCESS(Status)) {
1015                 break;
1016             }
1017 
1018             Start = 0;
1019             if (Layer == 0) {
1020                 Base = 0;
1021             }
1022             Base += Vcb->max_blocks_per_layer[Layer];
1023         }
1024     }
1025 
1026     Size->QuadPart = ((LONGLONG)(End - Extra)) << BLOCK_BITS;
1027 
1028     /* save inode whatever it succeeds to expand or not */
1029     Ext2SaveInode(IrpContext, Vcb, &Mcb->Inode);
1030 
1031     return Status;
1032 }
1033 
1034 NTSTATUS
1035 Ext2TruncateIndirectFast(
1036     PEXT2_IRP_CONTEXT IrpContext,
1037     PEXT2_VCB         Vcb,
1038     PEXT2_MCB         Mcb
1039     )
1040 {
1041     LONGLONG            Vba;
1042     LONGLONG            Lba;
1043     LONGLONG            Length;
1044     NTSTATUS            Status = STATUS_SUCCESS;
1045     int                 i;
1046 
1047     /* try to load all indirect blocks if mcb zone is not initialized */
1048     if (!IsZoneInited(Mcb)) {
1049         Status = Ext2InitializeZone(IrpContext, Vcb, Mcb);
1050         if (!NT_SUCCESS(Status)) {
1051             DbgBreak();
1052             ClearLongFlag(Mcb->Flags, MCB_ZONE_INITED);
1053             goto errorout;
1054         }
1055     }
1056 
1057     ASSERT (IsZoneInited(Mcb));
1058 
1059     /* delete all data blocks here */
1060     if (FsRtlNumberOfRunsInLargeMcb(&Mcb->Extents) != 0) {
1061         for (i = 0; FsRtlGetNextLargeMcbEntry(&Mcb->Extents, i, &Vba, &Lba, &Length); i++) {
1062             /* ignore the non-existing runs */
1063             if (-1 == Lba || Vba == 0 || Length <= 0)
1064                 continue;
1065             /* now do data block free */
1066             Ext2FreeBlock(IrpContext, Vcb, (ULONG)(Lba - 1), (ULONG)Length);
1067         }
1068     }
1069 
1070     /* delete all meta blocks here */
1071     if (FsRtlNumberOfRunsInLargeMcb(&Mcb->MetaExts) != 0) {
1072         for (i = 0; FsRtlGetNextLargeMcbEntry(&Mcb->MetaExts, i, &Vba, &Lba, &Length); i++) {
1073             /* ignore the non-existing runs */
1074             if (-1 == Lba || Vba == 0 || Length <= 0)
1075                 continue;
1076             /* now do meta block free */
1077             Ext2FreeBlock(IrpContext, Vcb, (ULONG)(Lba - 1), (ULONG)Length);
1078         }
1079     }
1080 
1081     /* clear data and meta extents */
1082     Ext2ClearAllExtents(&Mcb->Extents);
1083     Ext2ClearAllExtents(&Mcb->MetaExts);
1084     ClearFlag(Mcb->Flags, MCB_ZONE_INITED);
1085 
1086     /* clear inode blocks & sizes */
1087     Mcb->Inode.i_blocks = 0;
1088     Mcb->Inode.i_size = 0;
1089     memset(&Mcb->Inode.i_block[0], 0, sizeof(__u32) * 15);
1090 
1091     /* the caller will do inode save */
1092 
1093 errorout:
1094 
1095     return Status;
1096 }
1097 
1098 NTSTATUS
1099 Ext2TruncateIndirect(
1100     PEXT2_IRP_CONTEXT IrpContext,
1101     PEXT2_VCB         Vcb,
1102     PEXT2_MCB         Mcb,
1103     PLARGE_INTEGER    Size
1104 )
1105 {
1106     NTSTATUS Status = STATUS_SUCCESS;
1107 
1108     ULONG    Layer = 0;
1109 
1110     ULONG    Extra = 0;
1111     ULONG    Wanted = 0;
1112     ULONG    End;
1113     ULONG    Base;
1114 
1115     ULONG    SizeArray = 0;
1116     PULONG   BlockArray = NULL;
1117 
1118     /* translate file size to block */
1119     End = Base = Vcb->max_data_blocks;
1120     Wanted = (ULONG)((Size->QuadPart + BLOCK_SIZE - 1) >> BLOCK_BITS);
1121 
1122     /* do fast deletion here */
1123     if (Wanted == 0) {
1124         Status = Ext2TruncateIndirectFast(IrpContext, Vcb, Mcb);
1125         if (NT_SUCCESS(Status))
1126             goto errorout;
1127     }
1128 
1129     /* calculate blocks to be freed */
1130     Extra = End - Wanted;
1131 
1132     for (Layer = EXT2_BLOCK_TYPES; Layer > 0 && Extra; Layer--) {
1133 
1134         if (Vcb->max_blocks_per_layer[Layer - 1] == 0) {
1135             continue;
1136         }
1137 
1138         Base -= Vcb->max_blocks_per_layer[Layer - 1];
1139 
1140         if (Layer - 1 == 0) {
1141 #ifdef __REACTOS__
1142             BlockArray = (PULONG)&Mcb->Inode.i_block[0];
1143 #else
1144             BlockArray = &Mcb->Inode.i_block[0];
1145 #endif
1146             SizeArray = End;
1147             ASSERT(End == EXT2_NDIR_BLOCKS && Base == 0);
1148         } else {
1149 #ifdef __REACTOS__
1150             BlockArray = (PULONG)&Mcb->Inode.i_block[EXT2_NDIR_BLOCKS - 1 + Layer - 1];
1151 #else
1152             BlockArray = &Mcb->Inode.i_block[EXT2_NDIR_BLOCKS - 1 + Layer - 1];
1153 #endif
1154             SizeArray = 1;
1155         }
1156 
1157         Status = Ext2TruncateBlock(
1158                      IrpContext,
1159                      Vcb,
1160                      Mcb,
1161                      Base,
1162                      End - Base - 1,
1163                      Layer - 1,
1164                      SizeArray,
1165                      BlockArray,
1166                      &Extra
1167                  );
1168         if (!NT_SUCCESS(Status)) {
1169             break;
1170         }
1171 
1172         End = Base;
1173     }
1174 
1175 errorout:
1176 
1177     if (!NT_SUCCESS(Status)) {
1178         Size->QuadPart += ((ULONGLONG)Extra << BLOCK_BITS);
1179     }
1180 
1181     /* save inode */
1182     if (Mcb->Inode.i_size > (loff_t)(Size->QuadPart))
1183         Mcb->Inode.i_size = (loff_t)(Size->QuadPart);
1184     Ext2SaveInode(IrpContext, Vcb, &Mcb->Inode);
1185 
1186     return Status;
1187 }
1188