1 ////////////////////////////////////////////////////////////////////
2 // Copyright (C) Alexander Telyatnikov, Ivan Keliukh, Yegor Anchishkin, SKIF Software, 1999-2013. Kiev, Ukraine
3 // All rights reserved
4 // This file was released under the GPLv2 on June 2015.
5 ////////////////////////////////////////////////////////////////////
6 /*
7         Module name:
8 
9    udf_info.cpp
10 
11         Abstract:
12 
13    This file contains filesystem-specific routines
14    for Directory tree & related structures support
15 
16 */
17 
18 #include "udf.h"
19 
20 #ifdef UDF_CHECK_UTIL
21   #include "..\namesup.h"
22 #else
23   #ifdef UDF_BUG_CHECK_ID
24     #undef UDF_BUG_CHECK_ID
25   #endif //UDF_BUG_CHECK_ID
26 #endif //UDF_CHECK_UTIL
27 
28 #define         UDF_BUG_CHECK_ID                UDF_FILE_UDF_INFO_DIR
29 
30 #define         MEM_USDIRHASH_TAG               "USDirHash"
31 
32 #define UDF_DUMP_DIRTREE
33 #ifdef UDF_DUMP_DIRTREE
34 #define DirPrint(x)  UDFPrint(x)
35 #else
36 #define DirPrint(x)  {;}
37 #endif
38 
39 /*
40     This routine initializes DirIndex array
41  */
42 PDIR_INDEX_HDR
43 UDFDirIndexAlloc(
44     IN uint_di i
45     )
46 {
47     uint_di j,k;
48     PDIR_INDEX_HDR hDirNdx;
49     PDIR_INDEX_ITEM* FrameList;
50 
51     if(!i)
52         return NULL;
53 #ifdef UDF_LIMIT_DIR_SIZE
54     if(i>UDF_DIR_INDEX_FRAME)
55         return NULL;
56 #endif //UDF_LIMIT_DIR_SIZE
57 
58     j = i >> UDF_DIR_INDEX_FRAME_SH;
59     i &= (UDF_DIR_INDEX_FRAME-1);
60 
61     hDirNdx = (PDIR_INDEX_HDR)MyAllocatePoolTag__(UDF_DIR_INDEX_MT, sizeof(DIR_INDEX_HDR)+(j+(i!=0))*sizeof(PDIR_INDEX_ITEM), MEM_DIR_HDR_TAG);
62     if(!hDirNdx) return NULL;
63     RtlZeroMemory(hDirNdx, sizeof(DIR_INDEX_HDR));
64 
65     FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
66     for(k=0; k<j; k++, FrameList++) {
67         (*FrameList) = (PDIR_INDEX_ITEM)MyAllocatePoolTag__(UDF_DIR_INDEX_MT, UDF_DIR_INDEX_FRAME*sizeof(DIR_INDEX_ITEM), MEM_DIR_NDX_TAG);
68         if(!(*FrameList)) {
69 free_hdi:
70             // item pointet by FrameList is NULL, it could not be allocated
71             while(k) {
72                 k--;
73                 FrameList--;
74                 MyFreePool__(*FrameList);
75             }
76             MyFreePool__(hDirNdx);
77             return NULL;
78         }
79         RtlZeroMemory((*FrameList), UDF_DIR_INDEX_FRAME*sizeof(DIR_INDEX_ITEM));
80     }
81     if(i) {
82         (*FrameList) = (PDIR_INDEX_ITEM)MyAllocatePoolTag__(UDF_DIR_INDEX_MT, AlignDirIndex(i)*sizeof(DIR_INDEX_ITEM), MEM_DIR_NDX_TAG);
83         if(!(*FrameList))
84             goto free_hdi;
85         RtlZeroMemory((*FrameList), i*sizeof(DIR_INDEX_ITEM));
86     }
87 
88     hDirNdx->FrameCount = j+(i!=0);
89     hDirNdx->LastFrameCount = i ? i : UDF_DIR_INDEX_FRAME;
90 
91     return hDirNdx;
92 } // UDFDirIndexAlloc()
93 
94 /*
95     This routine releases DirIndex array
96  */
97 void
98 UDFDirIndexFree(
99     PDIR_INDEX_HDR hDirNdx
100     )
101 {
102     uint32 k;
103     PDIR_INDEX_ITEM* FrameList;
104 
105     FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
106     if(!hDirNdx) return;
107     for(k=0; k<hDirNdx->FrameCount; k++, FrameList++) {
108         if(*FrameList) MyFreePool__(*FrameList);
109     }
110     MyFreePool__(hDirNdx);
111 } // UDFDirIndexFree();
112 
113 /*
114     This routine grows DirIndex array
115  */
116 OSSTATUS
117 UDFDirIndexGrow(
118     IN PDIR_INDEX_HDR* _hDirNdx,
119     IN uint_di d // increment
120     )
121 {
122     uint_di j,k;
123     PDIR_INDEX_HDR hDirNdx = *_hDirNdx;
124     PDIR_INDEX_ITEM* FrameList;
125 
126     if(d > UDF_DIR_INDEX_FRAME)
127         return STATUS_INVALID_PARAMETER;
128 
129     j = hDirNdx->LastFrameCount+d;
130 
131     if(j > UDF_DIR_INDEX_FRAME) {
132 #ifndef UDF_LIMIT_DIR_SIZE // release
133         // Grow header
134         k = hDirNdx->FrameCount;
135         if(!MyReallocPool__((int8*)hDirNdx, sizeof(DIR_INDEX_HDR) + k*sizeof(PDIR_INDEX_ITEM),
136                        (int8**)(&hDirNdx), sizeof(DIR_INDEX_HDR) + (k+1)*sizeof(PDIR_INDEX_ITEM) ) )
137             return STATUS_INSUFFICIENT_RESOURCES;
138         FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
139         // Grow last frame
140         if(!MyReallocPool__((int8*)(FrameList[k-1]), AlignDirIndex(hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM),
141                        (int8**)(&(FrameList[k-1])), UDF_DIR_INDEX_FRAME*sizeof(DIR_INDEX_ITEM) ) )
142             return STATUS_INSUFFICIENT_RESOURCES;
143         RtlZeroMemory(&(FrameList[k-1][hDirNdx->LastFrameCount]),
144                        (UDF_DIR_INDEX_FRAME-hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM));
145         hDirNdx->LastFrameCount = UDF_DIR_INDEX_FRAME;
146         // Allocate new frame
147         FrameList[k] = (PDIR_INDEX_ITEM)MyAllocatePoolTag__(UDF_DIR_INDEX_MT, AlignDirIndex(j-UDF_DIR_INDEX_FRAME)*sizeof(DIR_INDEX_ITEM), MEM_DIR_NDX_TAG );
148         if(!FrameList[k])
149             return STATUS_INSUFFICIENT_RESOURCES;
150         hDirNdx->FrameCount++;
151         RtlZeroMemory(FrameList[k], (j-UDF_DIR_INDEX_FRAME)*sizeof(DIR_INDEX_ITEM));
152         hDirNdx->LastFrameCount = j-UDF_DIR_INDEX_FRAME;
153         (*_hDirNdx) = hDirNdx;
154 #else   // UDF_LIMIT_DIR_SIZE
155         return STATUS_INSUFFICIENT_RESOURCES;
156 #endif  // UDF_LIMIT_DIR_SIZE
157     } else {
158         k = hDirNdx->FrameCount;
159         FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
160         if(!MyReallocPool__((int8*)(FrameList[k-1]), AlignDirIndex(hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM),
161                        (int8**)(&(FrameList[k-1])), AlignDirIndex(j)*sizeof(DIR_INDEX_ITEM) ) )
162             return STATUS_INSUFFICIENT_RESOURCES;
163         RtlZeroMemory(&(FrameList[k-1][hDirNdx->LastFrameCount]),
164                        (j-hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM));
165         hDirNdx->LastFrameCount = j;
166     }
167     return STATUS_SUCCESS;
168 } // end UDFDirIndexGrow()
169 
170 /*
171     Thisd routine truncates DirIndex array
172  */
173 OSSTATUS
174 UDFDirIndexTrunc(
175     IN PDIR_INDEX_HDR* _hDirNdx,
176     IN uint_di d // decrement
177     )
178 {
179     uint_di j,k;
180 
181     if(d > UDF_DIR_INDEX_FRAME) {
182         OSSTATUS status;
183         while(d) {
184             k = (d > UDF_DIR_INDEX_FRAME) ? UDF_DIR_INDEX_FRAME : d;
185             if(!OS_SUCCESS(status = UDFDirIndexTrunc(_hDirNdx, k))) {
186                 return status;
187             }
188             d -= k;
189         }
190         return STATUS_SUCCESS;
191     }
192 
193     PDIR_INDEX_HDR hDirNdx = *_hDirNdx;
194     PDIR_INDEX_ITEM* FrameList;
195 
196     j = UDF_DIR_INDEX_FRAME+hDirNdx->LastFrameCount-d;
197     FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
198     k = hDirNdx->FrameCount-1;
199 
200     if(j <= UDF_DIR_INDEX_FRAME) {
201         // free last frame
202         if(!k && (j < 2)) {
203             // someone tries to trunc. residual entries...
204             return STATUS_INVALID_PARAMETER;
205         }
206         MyFreePool__(FrameList[k]);
207         FrameList[k] = NULL;
208         hDirNdx->LastFrameCount = UDF_DIR_INDEX_FRAME;
209         hDirNdx->FrameCount--;
210         // Truncate new last frame
211         if(!MyReallocPool__((int8*)(FrameList[k-1]), UDF_DIR_INDEX_FRAME*sizeof(DIR_INDEX_ITEM),
212                        (int8**)(&(FrameList[k-1])), AlignDirIndex(j)*sizeof(DIR_INDEX_ITEM) ) )
213             return STATUS_INSUFFICIENT_RESOURCES;
214         hDirNdx->LastFrameCount = j;
215         // Truncate header
216         if(!MyReallocPool__((int8*)hDirNdx, sizeof(DIR_INDEX_HDR) + (k+1)*sizeof(PDIR_INDEX_ITEM),
217                        (int8**)(&hDirNdx), sizeof(DIR_INDEX_HDR) + k*sizeof(PDIR_INDEX_ITEM) ) )
218             return STATUS_INSUFFICIENT_RESOURCES;
219 
220         (*_hDirNdx) = hDirNdx;
221 
222     } else {
223 
224         j -= UDF_DIR_INDEX_FRAME;
225         if(!k && (j < 2)) {
226             // someone tries to trunc. residual entries...
227             return STATUS_INVALID_PARAMETER;
228         }
229 
230         if(!MyReallocPool__((int8*)(FrameList[k]), AlignDirIndex(hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM),
231                        (int8**)(&(FrameList[k])), AlignDirIndex(j)*sizeof(DIR_INDEX_ITEM) ) )
232             return STATUS_INSUFFICIENT_RESOURCES;
233         hDirNdx->LastFrameCount = j;
234     }
235     return STATUS_SUCCESS;
236 } // end UDFDirIndexTrunc()
237 
238 #if defined _X86_ && !defined UDF_LIMIT_DIR_SIZE
239 #ifdef _MSC_VER
240 #pragma warning(disable:4035)               // re-enable below
241 #endif
242 /*
243     This routine returns pointer to DirIndex item with index i.
244  */
245 #if defined(_MSC_VER) && !defined(__clang__)
246 __declspec (naked)
247 #endif
248 PDIR_INDEX_ITEM
249 __fastcall
250 UDFDirIndex(
251     IN PDIR_INDEX_HDR hDirNdx, // ECX
252     IN uint32 i                // EDX
253     )
254 {
255 #if defined(_MSC_VER) && !defined(__clang__)
256     __asm {
257         push ebx
258         push ecx
259         push edx
260 
261 //        mov  ebx,hDirNdx
262         mov  ebx,ecx
263         mov  ecx,edx
264         or   ebx,ebx
265         jz   EO_udi_err
266 
267         mov  eax,ecx
268         shr  ecx,UDF_DIR_INDEX_FRAME_SH           ; ecx = j
269         mov  edx,[ebx]hDirNdx.FrameCount          ; edx = k
270         cmp  ecx,edx
271         jae  EO_udi_err
272 
273         and  eax,(1 shl UDF_DIR_INDEX_FRAME_SH)-1 ; eax = i
274         dec  edx
275         cmp  ecx,edx
276         jb   No_check
277 
278         cmp  eax,[ebx].LastFrameCount
279         jae  EO_udi_err
280 No_check:
281         add  ebx,size DIR_INDEX_HDR      ; ((PDIR_INDEX_ITEM*)(hDirNdx+1))...
282         mov  ebx,[ebx+ecx*4]             ; ...[j]...
283         mov  edx,size DIR_INDEX_ITEM
284         mul  edx                         ; ...[i]...
285         add  eax,ebx                     ; &(...)
286         jmp  udi_OK
287 EO_udi_err:
288         xor  eax,eax
289 udi_OK:
290         pop  edx
291         pop  ecx
292         pop  ebx
293 
294         ret
295     }
296 #else
297     /* FIXME ReactOS */
298     uint_di j, k;
299     if( hDirNdx &&
300         ((j = (i >> UDF_DIR_INDEX_FRAME_SH)) < (k = hDirNdx->FrameCount) ) &&
301         ((i = (i & (UDF_DIR_INDEX_FRAME-1))) < ((j < (k-1)) ? UDF_DIR_INDEX_FRAME : hDirNdx->LastFrameCount)) )
302         return &( (((PDIR_INDEX_ITEM*)(hDirNdx+1))[j])[i] );
303     return NULL;
304 #endif
305 }
306 #ifdef _MSC_VER
307 #pragma warning(default:4035)
308 #endif
309 #endif // _X86_
310 
311 /*
312     This routine returns pointer to DirIndex'es frame & index inside it
313     according to start Index parameter. It also initializes scan parameters
314  */
315 PDIR_INDEX_ITEM
316 UDFDirIndexGetFrame(
317     IN PDIR_INDEX_HDR hDirNdx,
318     IN uint32 Frame,
319    OUT uint32* FrameLen,
320    OUT uint_di* Index,
321     IN uint_di Rel
322     )
323 {
324     if(Frame >= hDirNdx->FrameCount)
325         return NULL;
326     if(Index) {
327 #ifdef UDF_LIMIT_DIR_SIZE
328         (*Index) = Rel;
329 //    if(FrameLen)
330         (*FrameLen) = hDirNdx->LastFrameCount;
331 #else //UDF_LIMIT_DIR_SIZE
332         (*Index) = Frame*UDF_DIR_INDEX_FRAME+Rel;
333 //    if(FrameLen)
334         (*FrameLen) = (Frame < (hDirNdx->FrameCount-1)) ? UDF_DIR_INDEX_FRAME :
335                                                           hDirNdx->LastFrameCount;
336 #endif //UDF_LIMIT_DIR_SIZE
337     }
338     return ((PDIR_INDEX_ITEM*)(hDirNdx+1))[Frame]+Rel;
339 } // end UDFDirIndexGetFrame()
340 
341 /*
342     This routine initializes indexes for optimized DirIndex scan
343     according to start Index parameter
344  */
345 
346 BOOLEAN
347 UDFDirIndexInitScan(
348     IN PUDF_FILE_INFO DirInfo,   //
349    OUT PUDF_DIR_SCAN_CONTEXT Context,
350     IN uint_di Index
351     )
352 {
353     Context->DirInfo = DirInfo;
354     Context->hDirNdx = DirInfo->Dloc->DirIndex;
355     if( (Context->frame = (Index >> UDF_DIR_INDEX_FRAME_SH)) >=
356                                                  Context->hDirNdx->FrameCount) {
357         return FALSE;
358     }
359     if( (Context->j = Index & (UDF_DIR_INDEX_FRAME-1)) >=
360                ((Context->frame < (Context->hDirNdx->FrameCount-1))
361                                     ?
362                                  UDF_DIR_INDEX_FRAME : Context->hDirNdx->LastFrameCount) ) {
363         return FALSE;
364     }
365     Context->DirNdx = UDFDirIndexGetFrame(Context->hDirNdx,
366                                           Context->frame,
367                                           &(Context->d),
368                                           &(Context->i),
369                                           Context->j);
370     Context->i--;
371     Context->j--;
372     Context->DirNdx--;
373 
374     return TRUE;
375 } // end UDFDirIndexInitScan()
376 
377 PDIR_INDEX_ITEM
378 UDFDirIndexScan(
379     PUDF_DIR_SCAN_CONTEXT Context,
380     PUDF_FILE_INFO* _FileInfo
381     )
382 {
383     PUDF_FILE_INFO FileInfo;
384     PUDF_FILE_INFO ParFileInfo;
385 
386     Context->i++;
387     Context->j++;
388     Context->DirNdx++;
389 
390     if(Context->j >= Context->d) {
391         Context->j=0;
392         Context->frame++;
393         Context->DirNdx = UDFDirIndexGetFrame(Context->hDirNdx,
394                                               Context->frame,
395                                               &(Context->d),
396                                               &(Context->i),
397                                               Context->j);
398     }
399     if(!Context->DirNdx) {
400         if(_FileInfo)
401             (*_FileInfo) = NULL;
402         return NULL;
403     }
404 
405     if(_FileInfo) {
406         if((FileInfo = Context->DirNdx->FileInfo)) {
407             if(FileInfo->ParentFile != Context->DirInfo) {
408                 ParFileInfo = UDFLocateParallelFI(Context->DirInfo,
409                                                   Context->i,
410                                                   FileInfo);
411 #ifdef UDF_DBG
412                 if(ParFileInfo->ParentFile != Context->DirInfo) {
413                     BrutePoint();
414                 }
415 #endif // UDF_DBG
416                 FileInfo = ParFileInfo;
417             }
418         }
419         (*_FileInfo) = FileInfo;
420     }
421 
422     return (Context->DirNdx);
423 } // end UDFDirIndexScan()
424 
425 /*
426     This routine calculates hashes for directory search
427  */
428 uint8
429 UDFBuildHashEntry(
430     IN PVCB Vcb,
431     IN PUNICODE_STRING Name,
432    OUT PHASH_ENTRY hashes,
433     IN uint8 Mask
434     )
435 {
436     UNICODE_STRING UName;
437     WCHAR ShortNameBuffer[13];
438     uint8 RetFlags = 0;
439 
440     if(!Name->Buffer) return 0;
441 
442     if(Mask & HASH_POSIX)
443         hashes->hPosix = crc32((uint8*)(Name->Buffer), Name->Length);
444 
445     if(Mask & HASH_ULFN) {
446 /*        if(OS_SUCCESS(MyInitUnicodeString(&UName, L"")) &&
447            OS_SUCCESS(MyAppendUnicodeStringToStringTag(&UName, Name, MEM_USDIRHASH_TAG))) {*/
448         if(OS_SUCCESS(MyCloneUnicodeString(&UName, Name))) {
449             RtlUpcaseUnicodeString(&UName, &UName, FALSE);
450     /*        if(!RtlCompareUnicodeString(Name, &UName, FALSE)) {
451                 RetFlags |= UDF_FI_FLAG_LFN;
452             }*/
453             hashes->hLfn = crc32((uint8*)(UName.Buffer), UName.Length);
454         } else {
455             BrutePoint();
456         }
457         MyFreePool__(UName.Buffer);
458     }
459 
460     if(Mask & HASH_DOS) {
461         UName.Buffer = (PWCHAR)(&ShortNameBuffer);
462         UName.MaximumLength = 13*sizeof(WCHAR);
463         UDFDOSName(Vcb, &UName, Name, (Mask & HASH_KEEP_NAME) ? TRUE : FALSE);
464         if(!RtlCompareUnicodeString(Name, &UName, TRUE)) {
465             RetFlags |= UDF_FI_FLAG_DOS;
466         }
467         hashes->hDos = crc32((uint8*)(UName.Buffer), UName.Length);
468     }
469     return RetFlags;
470 } // UDFBuildHashEntry()
471 
472 #ifdef UDF_CHECK_UTIL
473 uint32
474 UDFFindNextFI(
475     IN int8* buff,
476     IN uint32 prevOffset,
477     IN uint32 Length
478     )
479 {
480     PFILE_IDENT_DESC FileId;
481     while(prevOffset+sizeof(FILE_IDENT_DESC) < Length) {
482         prevOffset++;
483         FileId = (PFILE_IDENT_DESC)(buff+prevOffset);
484         if(FileId->descTag.tagIdent != TID_FILE_IDENT_DESC)
485             continue;
486         if(FileId->descTag.descVersion != 2 && FileId->descTag.descVersion != 3)
487             continue;
488         if(FileId->fileVersionNum != 1)
489             continue;
490         if(FileId->fileCharacteristics & (~0x1f))
491             continue;
492         if(prevOffset + ((FileId->lengthFileIdent + FileId->lengthOfImpUse + sizeof(FILE_IDENT_DESC) + 3) & (~((uint32)3))) <= Length) {
493             UDFPrint(("UDFFindNextFI OK: %x\n", prevOffset));
494             return prevOffset;
495         }
496     }
497     return 0;
498 } // end UDFFindNextFI()
499 #else //UDF_CHECK_UTIL
500 #define UDFFindNextFI(a,b,c)  0
501 #endif //UDF_CHECK_UTIL
502 
503 /*
504     This routine scans directory extent & builds index table for FileIdents
505  */
506 OSSTATUS
507 UDFIndexDirectory(
508     IN PVCB Vcb,
509  IN OUT PUDF_FILE_INFO FileInfo
510     )
511 {
512     PDIR_INDEX_HDR hDirNdx;
513     PDIR_INDEX_ITEM DirNdx;
514     PFILE_IDENT_DESC FileId;
515     uint32 Offset = 0;
516 //    uint32 prevOffset = 0;
517     uint_di Count = 0;
518     OSSTATUS status;
519     int8* buff;
520     PEXTENT_INFO ExtInfo;  // Extent array for directory
521     uint16 PartNum;
522     uint32 ReadBytes;
523     uint16 valueCRC;
524 
525     if(!FileInfo) return STATUS_INVALID_PARAMETER;
526     ValidateFileInfo(FileInfo);
527 
528     ExtInfo = &(FileInfo->Dloc->DataLoc);
529     FileInfo->Dloc->DirIndex = NULL;
530     UDFPrint(("UDF: scaning directory\n"));
531     // allocate buffer for the whole directory
532     ASSERT((uint32)(ExtInfo->Length));
533     if(!ExtInfo->Length)
534         return STATUS_FILE_CORRUPT_ERROR;
535     buff = (int8*)DbgAllocatePool(PagedPool, (uint32)(ExtInfo->Length));
536     if(!buff)
537         return STATUS_INSUFFICIENT_RESOURCES;
538 
539     ExtInfo->Flags |= EXTENT_FLAG_ALLOC_SEQUENTIAL;
540 
541     // read FileIdents
542     status = UDFReadExtent(Vcb, ExtInfo, 0, (uint32)(ExtInfo->Length), FALSE, buff, &ReadBytes);
543     if(!OS_SUCCESS(status)) {
544         DbgFreePool(buff);
545         return status;
546     }
547     // scan Dir to get entry counter
548     FileId = (PFILE_IDENT_DESC)buff;
549     DirPrint(("  ExtInfo->Length %x\n", ExtInfo->Length));
550 //    prevOffset = 0;
551     while(Offset<ExtInfo->Length) {
552         DirPrint(("  Offset %x\n", Offset));
553         if(!FileId->descTag.tagIdent) {
554             DirPrint(("  term item\n"));
555             break;
556         }
557         if(FileId->descTag.tagIdent != TID_FILE_IDENT_DESC) {
558             DirPrint(("  Inv. tag %x\n", FileId->descTag.tagIdent));
559             Offset = UDFFindNextFI(buff, prevOffset, (ULONG)(ExtInfo->Length));
560             if(!Offset) {
561                 DirPrint(("  can't find next\n"));
562                 break;
563             } else {
564                 DirPrint(("  found next offs %x\n", Offset));
565                 FileId = (PFILE_IDENT_DESC)((buff)+Offset);
566             }
567         }
568         if(((ULONG)Offset & (Vcb->LBlockSize-1)) > (Vcb->LBlockSize-sizeof(FILE_IDENT_DESC))) {
569             DirPrint(("  badly aligned\n", Offset));
570             if(Vcb->Modified) {
571                 DirPrint(("  queue repack request\n"));
572                 FileInfo->Dloc->DirIndex->DelCount = Vcb->PackDirThreshold+1;
573             }
574         }
575 //        prevOffset = Offset;
576         Offset += (FileId->lengthFileIdent + FileId->lengthOfImpUse + sizeof(FILE_IDENT_DESC) + 3) & (~((uint32)3));
577         FileId = (PFILE_IDENT_DESC)((buff)+Offset);
578         Count++;
579         if(Offset+sizeof(FILE_IDENT_DESC) > ExtInfo->Length) {
580             if(Offset != ExtInfo->Length) {
581                 UDFPrint(("  Trash at the end of Dir\n"));
582             }
583 //            BrutePoint();
584             break;
585         }
586     }
587     DirPrint(("  final Offset %x\n", Offset));
588     if(Offset > ExtInfo->Length) {
589         BrutePoint();
590         UDFPrint(("  Unexpected end of Dir\n"));
591         DbgFreePool(buff);
592         return STATUS_FILE_CORRUPT_ERROR;
593     }
594     // allocate buffer for directory index & zero it
595     DirPrint(("  Count %x\n", Count));
596     hDirNdx = UDFDirIndexAlloc(Count+1);
597     if(!hDirNdx) {
598         DbgFreePool(buff);
599         return STATUS_INSUFFICIENT_RESOURCES;
600     }
601 
602     Offset = Count = 0;
603     hDirNdx->DIFlags |= (ExtInfo->Offset ? UDF_DI_FLAG_INIT_IN_ICB : 0);
604     // add entry pointing to the directory itself
605     DirNdx = UDFDirIndex(hDirNdx,0);
606     ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation);
607     DirNdx->FileEntryLoc.partitionReferenceNum = PartNum =
608         (uint16)UDFGetPartNumByPhysLba(Vcb, FileInfo->Dloc->FELoc.Mapping[0].extLocation);
609     ASSERT(PartNum != -1);
610     DirNdx->FileEntryLoc.logicalBlockNum =
611         UDFPhysLbaToPart(Vcb, PartNum, FileInfo->Dloc->FELoc.Mapping[0].extLocation);
612     if(DirNdx->FileEntryLoc.logicalBlockNum == (ULONG)-1) {
613         DirPrint(("  err: FileEntryLoc=-1\n"));
614         DbgFreePool(buff);
615         UDFDirIndexFree(hDirNdx);
616         return STATUS_FILE_CORRUPT_ERROR;
617     }
618     DirNdx->FileCharacteristics = (FileInfo->FileIdent) ?
619                          FileInfo->FileIdent->fileCharacteristics :
620                          FILE_DIRECTORY;
621 //    DirNdx->Offset = 0;
622 //    DirNdx->Length = 0;
623     RtlInitUnicodeString(&DirNdx->FName, L".");
624     DirNdx->FileInfo = FileInfo;
625     DirNdx->FI_Flags |= UDF_FI_FLAG_KEEP_NAME;
626     DirNdx->FI_Flags |= UDFBuildHashEntry(Vcb, &(DirNdx->FName), &(DirNdx->hashes),
627         HASH_ALL | HASH_KEEP_NAME);
628     Count++;
629     FileId = (PFILE_IDENT_DESC)buff;
630     status = STATUS_SUCCESS;
631 //    prevOffset = 0;
632     while((Offset<ExtInfo->Length) && FileId->descTag.tagIdent) {
633         // add new entry to index list
634         if(FileId->descTag.tagIdent != TID_FILE_IDENT_DESC) {
635             UDFPrint(("  Invalid tagIdent %x (expected %x) offst %x\n", FileId->descTag.tagIdent, TID_FILE_IDENT_DESC, Offset));
636             DirPrint(("    FileId: filen %x, iulen %x, charact %x\n",
637                 FileId->lengthFileIdent, FileId->lengthOfImpUse, FileId->fileCharacteristics));
638             DirPrint(("    loc: @%x\n", UDFExtentOffsetToLba(Vcb, ExtInfo->Mapping, Offset, NULL, NULL, NULL, NULL)));
639             KdDump(FileId, sizeof(FileId->descTag));
640             Offset = UDFFindNextFI(buff, prevOffset, (ULONG)(ExtInfo->Length));
641             if(!Offset) {
642                 DbgFreePool(buff);
643                 UDFDirIndexFree(hDirNdx);
644                 return STATUS_FILE_CORRUPT_ERROR;
645             } else {
646                 DirPrint(("  found next offs %x\n", Offset));
647                 FileId = (PFILE_IDENT_DESC)((buff)+Offset);
648             }
649         }
650         DirNdx = UDFDirIndex(hDirNdx,Count);
651         // allocate buffer & fill it with decompressed unicode filename
652         if(FileId->fileCharacteristics & FILE_DELETED) {
653             DirPrint(("  FILE_DELETED\n"));
654             hDirNdx->DelCount++;
655         }
656         DirPrint(("  FileId: offs %x, filen %x, iulen %x\n", Offset, FileId->lengthFileIdent, FileId->lengthOfImpUse));
657         DirNdx->Length = (FileId->lengthFileIdent + FileId->lengthOfImpUse + sizeof(FILE_IDENT_DESC) + 3) & (~((uint32)3));
658         DirPrint(("  DirNdx: Length %x, Charact %x\n", DirNdx->Length, FileId->fileCharacteristics));
659         if(FileId->fileCharacteristics & FILE_PARENT) {
660             DirPrint(("  parent\n"));
661             // init 'parent' entry
662             // '..' points to Parent Object (if any),
663             // otherwise it points to the Dir itself
664             RtlInitUnicodeString(&DirNdx->FName, L"..");
665             DirNdx->FileInfo = (FileInfo->ParentFile) ?
666                                       FileInfo->ParentFile : FileInfo;
667             DirNdx->FI_Flags |= UDF_FI_FLAG_KEEP_NAME;
668             DirNdx->FI_Flags |= UDFBuildHashEntry(Vcb, &(DirNdx->FName), &(DirNdx->hashes), HASH_ALL | HASH_KEEP_NAME);
669         } else {
670             // init plain file/dir entry
671             ASSERT( (Offset+sizeof(FILE_IDENT_DESC)+FileId->lengthOfImpUse+FileId->lengthFileIdent) <=
672                     ExtInfo->Length );
673             UDFDecompressUnicode(&(DirNdx->FName),
674                              ((uint8*)(FileId+1)) + (FileId->lengthOfImpUse),
675                              FileId->lengthFileIdent,
676                              &valueCRC);
677             UDFNormalizeFileName(&(DirNdx->FName), valueCRC);
678             DirNdx->FI_Flags |= UDFBuildHashEntry(Vcb, &(DirNdx->FName), &(DirNdx->hashes), HASH_ALL);
679         }
680         if((FileId->fileCharacteristics & FILE_METADATA)
681                        ||
682               !DirNdx->FName.Buffer
683                        ||
684            ((DirNdx->FName.Length >= sizeof(UDF_RESERVED_NAME_HDR)-sizeof(WCHAR)) &&
685             (RtlCompareMemory(DirNdx->FName.Buffer, UDF_RESERVED_NAME_HDR, sizeof(UDF_RESERVED_NAME_HDR)-sizeof(WCHAR)) == sizeof(UDF_RESERVED_NAME_HDR)-sizeof(WCHAR)) )) {
686             DirPrint(("  metadata\n"));
687             DirNdx->FI_Flags |= UDF_FI_FLAG_FI_INTERNAL;
688         }
689 #if 0
690         UDFPrint(("%ws\n", DirNdx->FName.Buffer));
691 #endif
692         DirPrint(("%ws\n", DirNdx->FName.Buffer));
693         // remember FileEntry location...
694         DirNdx->FileEntryLoc = FileId->icb.extLocation;
695         // ... and some file characteristics
696         DirNdx->FileCharacteristics = FileId->fileCharacteristics;
697         DirNdx->Offset = Offset;
698 #ifdef UDF_CHECK_DISK_ALLOCATION
699         if(!(FileId->fileCharacteristics & FILE_DELETED) &&
700             (UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) != LBA_OUT_OF_EXTENT) &&
701              UDFGetFreeBit(((uint32*)(Vcb->FSBM_Bitmap)), UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) )) {
702 
703             AdPrint(("Ref to Discarded block %x\n",UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) ));
704             BrutePoint();
705             FileId->fileCharacteristics |= FILE_DELETED;
706         } else
707         if(UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) == LBA_OUT_OF_EXTENT) {
708             AdPrint(("Ref to Invalid block %x\n", UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) ));
709             BrutePoint();
710             FileId->fileCharacteristics |= FILE_DELETED;
711         }
712 #endif // UDF_CHECK_DISK_ALLOCATION
713 //        prevOffset = Offset;
714         Offset += DirNdx->Length;
715         FileId = (PFILE_IDENT_DESC)(((int8*)FileId)+DirNdx->Length);
716         Count++;
717         if(Offset+sizeof(FILE_IDENT_DESC) > ExtInfo->Length) {
718             if(Offset != ExtInfo->Length) {
719                 UDFPrint(("  Trash at the end of Dir (2)\n"));
720             }
721 //            BrutePoint();
722             break;
723         }
724     } // while()
725     // we needn't writing terminator 'cause the buffer is already zero-filled
726     DbgFreePool(buff);
727     if(Count < 2) {
728         UDFDirIndexFree(hDirNdx);
729         UDFPrint(("  Directory too short\n"));
730         return STATUS_FILE_CORRUPT_ERROR;
731     }
732     // store index
733     FileInfo->Dloc->DirIndex = hDirNdx;
734     return status;
735 } // end UDFIndexDirectory()
736 
737 #ifndef UDF_READ_ONLY_BUILD
738 /*
739     This routine removes all DELETED entries from Dir & resizes it.
740     It must be called before closing, no files sould be opened.
741  */
742 OSSTATUS
743 UDFPackDirectory__(
744     IN PVCB Vcb,
745  IN OUT PUDF_FILE_INFO FileInfo   // source (opened)
746     )
747 {
748 #ifdef UDF_PACK_DIRS
749     uint32 d, LBS;
750     uint_di i, j;
751     uint32 IUl, FIl, l;
752     uint32 DataLocOffset;
753     uint32 Offset, curOffset;
754     int8* Buf;
755     OSSTATUS status;
756     uint32 ReadBytes;
757     int8* storedFI;
758     PUDF_FILE_INFO curFileInfo;
759     PDIR_INDEX_ITEM DirNdx, DirNdx2;
760     UDF_DIR_SCAN_CONTEXT ScanContext;
761     uint_di dc=0;
762     uint16 PartNum;
763 #endif //UDF_PACK_DIRS
764 
765     ValidateFileInfo(FileInfo);
766     PDIR_INDEX_HDR hDirNdx = FileInfo->Dloc->DirIndex;
767     if(!hDirNdx) return STATUS_NOT_A_DIRECTORY;
768 #ifndef UDF_PACK_DIRS
769     return STATUS_SUCCESS;
770 #else // UDF_PACK_DIRS
771 
772     // do not pack dirs on unchanged disks
773     if(!Vcb->Modified)
774         return STATUS_SUCCESS;
775     // start packing
776     LBS = Vcb->LBlockSize;
777     Buf = (int8*)DbgAllocatePool(PagedPool, LBS*2);
778     if(!Buf) return STATUS_INSUFFICIENT_RESOURCES;
779     // we shall never touch 1st entry 'cause it can't be deleted
780     Offset = UDFDirIndex(hDirNdx,2)->Offset;
781     DataLocOffset = FileInfo->Dloc->DataLoc.Offset;
782 
783     i=j=2;
784 
785     if(!UDFDirIndexInitScan(FileInfo, &ScanContext, i)) {
786         DbgFreePool(Buf);
787         return STATUS_SUCCESS;
788     }
789 
790     ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation);
791     PartNum = (uint16)UDFGetPartNumByPhysLba(Vcb, FileInfo->Dloc->FELoc.Mapping[0].extLocation);
792     ASSERT(PartNum != -1);
793 
794     while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
795 
796         if(UDFIsDeleted(DirNdx))
797             dc++;
798 
799         if(!UDFIsDeleted(DirNdx) ||
800              DirNdx->FileInfo) {
801             // move down valid entry
802             status = UDFReadFile__(Vcb, FileInfo, curOffset = DirNdx->Offset,
803                                                           l = DirNdx->Length, FALSE, Buf, &ReadBytes);
804             if(!OS_SUCCESS(status)) {
805                 DbgFreePool(Buf);
806                 return status;
807             }
808             // remove ImpUse field
809             IUl = ((PFILE_IDENT_DESC)Buf)->lengthOfImpUse;
810             curFileInfo = DirNdx->FileInfo;
811             // align next entry
812             if((d = LBS - ((curOffset + (l - IUl) + DataLocOffset) & (LBS-1)) ) < sizeof(FILE_IDENT_DESC)) {
813 
814                 // insufficient space at the end of last sector for
815                 // next FileIdent's tag. fill it with ImpUse data
816 
817                 // generally, all data should be DWORD-aligned, but if it is not so
818                 // this opearation will help us to avoid glitches
819                 d = (d+3) & ~(3);
820                 if(d != IUl) {
821                     l = l + d - IUl;
822                     FIl = ((PFILE_IDENT_DESC)Buf)->lengthFileIdent;
823                     // copy filename to upper addr
824                     RtlMoveMemory(Buf+sizeof(FILE_IDENT_DESC)+d,
825                                   Buf+sizeof(FILE_IDENT_DESC)+IUl, FIl);
826                     RtlZeroMemory(Buf+sizeof(FILE_IDENT_DESC), d);
827                     ((PFILE_IDENT_DESC)Buf)->lengthOfImpUse = (uint16)d;
828 
829                     if(curFileInfo && curFileInfo->FileIdent) {
830                         // update stored FI if any
831                         if(!MyReallocPool__((int8*)(curFileInfo->FileIdent), l,
832                                      (int8**)&(curFileInfo->FileIdent), (l+IUl-d) )) {
833                             DbgFreePool(Buf);
834                             return STATUS_INSUFFICIENT_RESOURCES;
835                         }
836                         storedFI = (int8*)(curFileInfo->FileIdent);
837                         RtlMoveMemory(storedFI+sizeof(FILE_IDENT_DESC)+d,
838                                       storedFI+sizeof(FILE_IDENT_DESC)+IUl, FIl);
839                         RtlZeroMemory(storedFI+sizeof(FILE_IDENT_DESC), d);
840                         ((PFILE_IDENT_DESC)storedFI)->lengthOfImpUse = (uint16)d;
841                         FileInfo->Dloc->FELoc.Modified = TRUE;
842                         FileInfo->Dloc->FE_Flags |= UDF_FE_FLAG_FE_MODIFIED;
843                     }
844                 }
845             } else {
846                 d = 0;
847             }
848             // write modified to new addr
849             if((d != IUl) ||
850                (curOffset != Offset)) {
851 
852                 UDFSetUpTag(Vcb, (tag*)Buf, (uint16)l,
853                           UDFPhysLbaToPart(Vcb, PartNum,
854                                      UDFExtentOffsetToLba(Vcb, FileInfo->Dloc->DataLoc.Mapping,
855                                                 Offset, NULL, NULL, NULL, NULL)));
856 
857                 status = UDFWriteFile__(Vcb, FileInfo, Offset, l, FALSE, Buf, &ReadBytes);
858                 if(!OS_SUCCESS(status)) {
859                     DbgFreePool(Buf);
860                     return status;
861                 }
862             }
863             DirNdx2 = UDFDirIndex(hDirNdx, j);
864             *DirNdx2 = *DirNdx;
865             DirNdx2->Offset = Offset;
866             DirNdx2->Length = l;
867             if(curFileInfo) {
868                 curFileInfo->Index = j;
869                 DirNdx2->FI_Flags |= UDF_FI_FLAG_FI_MODIFIED;
870             }
871             Offset += l;
872             j++;
873         }
874     }
875     // resize DirIndex
876     DbgFreePool(Buf);
877     if(dc) {
878         if(!OS_SUCCESS(status = UDFDirIndexTrunc(&(FileInfo->Dloc->DirIndex), dc))) {
879             return status;
880         }
881     }
882     // terminator is set by UDFDirIndexTrunc()
883     FileInfo->Dloc->DirIndex->DelCount = 0;
884     ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation);
885 
886     // now Offset points to EOF. Let's truncate directory
887     return UDFResizeFile__(Vcb, FileInfo, Offset);
888 #endif // UDF_PACK_DIRS
889 } // end UDFPackDirectory__()
890 
891 /*
892     This routine rebuilds tags for all entries from Dir.
893  */
894 OSSTATUS
895 UDFReTagDirectory(
896     IN PVCB Vcb,
897  IN OUT PUDF_FILE_INFO FileInfo   // source (opened)
898     )
899 {
900     uint32 l;
901     uint32 Offset;
902     int8* Buf;
903     OSSTATUS status;
904     uint32 ReadBytes;
905     PUDF_FILE_INFO curFileInfo;
906     PDIR_INDEX_ITEM DirNdx;
907     UDF_DIR_SCAN_CONTEXT ScanContext;
908     uint16 PartNum;
909 
910     ValidateFileInfo(FileInfo);
911     PDIR_INDEX_HDR hDirNdx = FileInfo->Dloc->DirIndex;
912     if(!hDirNdx) return STATUS_NOT_A_DIRECTORY;
913 
914     // do not pack dirs on unchanged disks
915     if(!Vcb->Modified)
916         return STATUS_SUCCESS;
917 
918     if( ((hDirNdx->DIFlags & UDF_DI_FLAG_INIT_IN_ICB) ? TRUE : FALSE) ==
919         ((FileInfo->Dloc->DataLoc.Offset) ? TRUE : FALSE) ) {
920         return STATUS_SUCCESS;
921     }
922 
923     // start packing
924     Buf = (int8*)DbgAllocatePool(PagedPool, Vcb->LBlockSize*2);
925     if(!Buf) return STATUS_INSUFFICIENT_RESOURCES;
926 
927     Offset = UDFDirIndex(hDirNdx,1)->Offset;
928 
929     if(!UDFDirIndexInitScan(FileInfo, &ScanContext, 1)) {
930         DbgFreePool(Buf);
931         return STATUS_SUCCESS;
932     }
933 
934     ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation);
935     PartNum = (uint16)UDFGetPartNumByPhysLba(Vcb, FileInfo->Dloc->FELoc.Mapping[0].extLocation);
936     ASSERT(PartNum != -1);
937 
938     while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
939 
940         status = UDFReadFile__(Vcb, FileInfo, Offset = DirNdx->Offset,
941                                                    l = DirNdx->Length, FALSE, Buf, &ReadBytes);
942         if(!OS_SUCCESS(status)) {
943             DbgFreePool(Buf);
944             return status;
945         }
946         curFileInfo = DirNdx->FileInfo;
947         // write modified
948         UDFSetUpTag(Vcb, (tag*)Buf, (uint16)l,
949                   UDFPhysLbaToPart(Vcb, PartNum,
950                              UDFExtentOffsetToLba(Vcb, FileInfo->Dloc->DataLoc.Mapping,
951                                         Offset, NULL, NULL, NULL, NULL)));
952 
953         if(curFileInfo && curFileInfo->FileIdent) {
954             FileInfo->Dloc->FELoc.Modified = TRUE;
955             FileInfo->Dloc->FE_Flags |= UDF_FE_FLAG_FE_MODIFIED;
956         }
957 
958         status = UDFWriteFile__(Vcb, FileInfo, Offset, l, FALSE, Buf, &ReadBytes);
959         if(!OS_SUCCESS(status)) {
960             DbgFreePool(Buf);
961             return status;
962         }
963         if(curFileInfo) {
964             DirNdx->FI_Flags |= UDF_FI_FLAG_FI_MODIFIED;
965         }
966     }
967     // resize DirIndex
968     DbgFreePool(Buf);
969 
970     hDirNdx->DIFlags &= ~UDF_DI_FLAG_INIT_IN_ICB;
971     hDirNdx->DIFlags |= (FileInfo->Dloc->DataLoc.Offset ? UDF_DI_FLAG_INIT_IN_ICB : 0);
972     return status;
973 
974 } // end UDFReTagDirectory()
975 #endif //UDF_READ_ONLY_BUILD
976 
977 /*
978     This routine performs search for specified file in specified directory &
979     returns corresponding offset in extent if found.
980  */
981 OSSTATUS
982 UDFFindFile(
983     IN PVCB Vcb,
984     IN BOOLEAN IgnoreCase,
985     IN BOOLEAN NotDeleted,
986     IN PUNICODE_STRING Name,
987     IN PUDF_FILE_INFO DirInfo,
988  IN OUT uint_di* Index      // IN:start index OUT:found file index
989     )
990 {
991 //    PDIR_INDEX_HDR hDirIndex = DirInfo->Dloc->DirIndex;
992     UNICODE_STRING ShortName;
993     WCHAR ShortNameBuffer[13];
994     PDIR_INDEX_ITEM DirNdx;
995     UDF_DIR_SCAN_CONTEXT ScanContext;
996     uint_di j=(-1), k=(-1);
997     HASH_ENTRY hashes;
998     BOOLEAN CanBe8d3;
999 
1000     UDFBuildHashEntry(Vcb, Name, &hashes, HASH_POSIX | HASH_ULFN);
1001 
1002     if((CanBe8d3 = UDFCanNameBeA8dot3(Name))) {
1003         ShortName.MaximumLength = 13 * sizeof(WCHAR);
1004         ShortName.Buffer = (PWCHAR)&ShortNameBuffer;
1005     }
1006 
1007     if(!UDFDirIndexInitScan(DirInfo, &ScanContext, (*Index)))
1008         return STATUS_OBJECT_NAME_NOT_FOUND;
1009 
1010     if(!IgnoreCase && !CanBe8d3) {
1011         // perform case sensetive sequential directory scan
1012 
1013         while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
1014             if( (DirNdx->hashes.hPosix == hashes.hPosix) &&
1015                  DirNdx->FName.Buffer &&
1016                 (!RtlCompareUnicodeString(&(DirNdx->FName), Name, FALSE)) &&
1017                ( (!UDFIsDeleted(DirNdx)) || (!NotDeleted) ) ) {
1018                 (*Index) = ScanContext.i;
1019                 return STATUS_SUCCESS;
1020             }
1021         }
1022         return STATUS_OBJECT_NAME_NOT_FOUND;
1023     }
1024 
1025     if(hashes.hPosix == hashes.hLfn) {
1026 
1027         while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
1028             if(!DirNdx->FName.Buffer ||
1029                (NotDeleted && UDFIsDeleted(DirNdx)) )
1030                 continue;
1031             if( (DirNdx->hashes.hLfn == hashes.hLfn) &&
1032                 (!RtlCompareUnicodeString(&(DirNdx->FName), Name, IgnoreCase)) ) {
1033                 (*Index) = ScanContext.i;
1034                 return STATUS_SUCCESS;
1035             } else
1036             if( CanBe8d3 &&
1037                 !(DirNdx->FI_Flags & UDF_FI_FLAG_DOS) &&
1038                 (DirNdx->hashes.hDos == hashes.hLfn) &&
1039                 (k == (uint_di)(-1))) {
1040                 UDFDOSName(Vcb, &ShortName, &(DirNdx->FName), ScanContext.i < 2) ;
1041                 if(!RtlCompareUnicodeString(&ShortName, Name, IgnoreCase))
1042                     k = ScanContext.i;
1043             }
1044         }
1045 
1046     } else {
1047 
1048         while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
1049             // perform sequential directory scan
1050             if(!DirNdx->FName.Buffer ||
1051                (NotDeleted && UDFIsDeleted(DirNdx)) )
1052                 continue;
1053             if( (DirNdx->hashes.hPosix == hashes.hPosix) &&
1054                 (!RtlCompareUnicodeString(&(DirNdx->FName), Name, FALSE)) ) {
1055                 (*Index) = ScanContext.i;
1056                 return STATUS_SUCCESS;
1057             } else
1058             if( (DirNdx->hashes.hLfn == hashes.hLfn) &&
1059                 (j == (uint_di)(-1)) &&
1060                 (!RtlCompareUnicodeString(&(DirNdx->FName), Name, IgnoreCase)) ) {
1061                 j = ScanContext.i;
1062             } else
1063             if( CanBe8d3 &&
1064                 !(DirNdx->FI_Flags & UDF_FI_FLAG_DOS) &&
1065                 (DirNdx->hashes.hDos == hashes.hLfn) &&
1066                 (k == (uint_di)(-1))) {
1067                 UDFDOSName(Vcb, &ShortName, &(DirNdx->FName), ScanContext.i < 2 );
1068                 if(!RtlCompareUnicodeString(&ShortName, Name, IgnoreCase)) {
1069                     k = ScanContext.i;
1070                 }
1071             }
1072         }
1073     }
1074 
1075     if(j != (uint_di)(-1)) {
1076         (*Index) = j;
1077         return STATUS_SUCCESS;
1078     } else
1079     if(k != (uint_di)(-1)) {
1080         (*Index) = k;
1081         return STATUS_SUCCESS;
1082     }
1083 
1084     return STATUS_OBJECT_NAME_NOT_FOUND;
1085 
1086 } // end UDFFindFile()
1087 
1088 /*
1089     This routine returns pointer to parent DirIndex
1090 */
1091 PDIR_INDEX_HDR
1092 UDFGetDirIndexByFileInfo(
1093     IN PUDF_FILE_INFO FileInfo
1094     )
1095 {
1096     ValidateFileInfo(FileInfo);
1097 
1098     if(!FileInfo) {
1099         BrutePoint();
1100         return NULL;
1101     }
1102     if (FileInfo->ParentFile) {
1103         ValidateFileInfo(FileInfo->ParentFile);
1104 
1105         if(UDFIsAStreamDir(FileInfo))
1106             return NULL;
1107         if(FileInfo->ParentFile->Dloc)
1108             return FileInfo->ParentFile->Dloc->DirIndex;
1109         return NULL;
1110     }
1111     if(FileInfo->Dloc)
1112         return FileInfo->Dloc->DirIndex;
1113     return NULL;
1114 }
1115 
1116 /*
1117     File Data Location support routines (UDFXxxDloc)
1118     This group is responsible for caching FE locations
1119     If requested FE referenced by another FI the file is assumed to be linked
1120     All linked files reference to common Data Location (& attr) structure
1121  */
1122 
1123 /*
1124     Check if given FE is already in use
1125  */
1126 LONG
1127 UDFFindDloc(
1128     IN PVCB Vcb,
1129     IN uint32 Lba
1130     )
1131 {
1132     PUDF_DATALOC_INDEX DlocList;
1133     uint32 l;
1134 
1135     if(!(DlocList = Vcb->DlocList) || !Lba) return (-1);
1136     // scan FE location cache
1137     l = Vcb->DlocCount;
1138     for(uint32 i=0; i<l; i++, DlocList++) {
1139         if(DlocList->Lba == Lba)
1140             return i;
1141     }
1142     return (-1);
1143 } // end UDFFindDloc()
1144 
1145 /*
1146     Check if given FE is already stored in memory
1147  */
1148 LONG
1149 UDFFindDlocInMem(
1150     IN PVCB Vcb,
1151     IN PUDF_DATALOC_INFO Dloc
1152     )
1153 {
1154     PUDF_DATALOC_INDEX DlocList;
1155     uint32 l;
1156 
1157     if(!(DlocList = Vcb->DlocList) || !Dloc) return (-1);
1158     // scan FE location cache
1159     l = Vcb->DlocCount;
1160     for(uint32 i=0; i<l; i++, DlocList++) {
1161         if(DlocList->Dloc == Dloc)
1162             return i;
1163     }
1164     return (-1);
1165 } // end UDFFindDlocInMem()
1166 
1167 /*
1168     Find free cache entry
1169  */
1170 LONG
1171 UDFFindFreeDloc(
1172     IN PVCB Vcb,
1173     IN uint32 Lba
1174     )
1175 {
1176     PUDF_DATALOC_INDEX DlocList;
1177     uint32 l;
1178 
1179     if(!Vcb->DlocList) {
1180         // init FE location cache
1181         if(!(Vcb->DlocList = (PUDF_DATALOC_INDEX)MyAllocatePoolTag__(NonPagedPool, sizeof(UDF_DATALOC_INDEX)*DLOC_LIST_GRANULARITY, MEM_DLOC_NDX_TAG)))
1182             return (-1);
1183         RtlZeroMemory(Vcb->DlocList, DLOC_LIST_GRANULARITY*sizeof(UDF_DATALOC_INDEX));
1184         Vcb->DlocCount = DLOC_LIST_GRANULARITY;
1185     }
1186     // scan for free entry
1187     DlocList = Vcb->DlocList;
1188     l = Vcb->DlocCount;
1189     for(uint32 i=0; i<l; i++, DlocList++) {
1190         if(!DlocList->Dloc)
1191             return i;
1192     }
1193     // alloc some free entries
1194     if(!MyReallocPool__((int8*)(Vcb->DlocList), Vcb->DlocCount*sizeof(UDF_DATALOC_INDEX),
1195                      (int8**)&(Vcb->DlocList), (Vcb->DlocCount+DLOC_LIST_GRANULARITY)*sizeof(UDF_DATALOC_INDEX))) {
1196         return (-1);
1197     }
1198     RtlZeroMemory(&(Vcb->DlocList[Vcb->DlocCount]), DLOC_LIST_GRANULARITY*sizeof(UDF_DATALOC_INDEX));
1199     Vcb->DlocCount += DLOC_LIST_GRANULARITY;
1200     return (Vcb->DlocCount - DLOC_LIST_GRANULARITY);
1201 } // end UDFFindFreeDloc()
1202 
1203 /*
1204  */
1205 OSSTATUS
1206 UDFAcquireDloc(
1207     IN PVCB Vcb,
1208     IN PUDF_DATALOC_INFO Dloc
1209     )
1210 {
1211     UDFAcquireResourceExclusive(&(Vcb->DlocResource2),TRUE);
1212     if(Dloc->FE_Flags & UDF_FE_FLAG_UNDER_INIT) {
1213         UDFReleaseResource(&(Vcb->DlocResource2));
1214         return STATUS_SHARING_PAUSED;
1215     }
1216     Dloc->FE_Flags |= UDF_FE_FLAG_UNDER_INIT;
1217     UDFReleaseResource(&(Vcb->DlocResource2));
1218     return STATUS_SUCCESS;
1219 } // end UDFAcquireDloc()
1220 
1221 /*
1222  */
1223 OSSTATUS
1224 UDFReleaseDloc(
1225     IN PVCB Vcb,
1226     IN PUDF_DATALOC_INFO Dloc
1227     )
1228 {
1229     UDFAcquireResourceExclusive(&(Vcb->DlocResource2),TRUE);
1230     Dloc->FE_Flags &= ~UDF_FE_FLAG_UNDER_INIT;
1231     UDFReleaseResource(&(Vcb->DlocResource2));
1232     return STATUS_SUCCESS;
1233 } // end UDFReleaseDloc()
1234 
1235 /*
1236     Try to store FE location in cache
1237     If it is already in use, caller will be informed about it
1238  */
1239 OSSTATUS
1240 UDFStoreDloc(
1241     IN PVCB Vcb,
1242     IN PUDF_FILE_INFO fi,
1243     IN uint32 Lba
1244     )
1245 {
1246     LONG i;
1247     PUDF_DATALOC_INFO Dloc;
1248 
1249     if(!Lba) return STATUS_INVALID_PARAMETER;
1250     if(Lba == (ULONG)-1) return STATUS_INVALID_PARAMETER;
1251 
1252     UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
1253 
1254     // check if FE specified is already in use
1255     if((i = UDFFindDloc(Vcb, Lba)) == (-1)) {
1256         // not used
1257         if((i = UDFFindFreeDloc(Vcb, Lba)) == (-1)) {
1258             UDFReleaseResource(&(Vcb->DlocResource));
1259             return STATUS_INSUFFICIENT_RESOURCES;
1260         }
1261     } else {
1262         if(!OS_SUCCESS(UDFAcquireDloc(Vcb, Dloc = Vcb->DlocList[i].Dloc))) {
1263             UDFReleaseResource(&(Vcb->DlocResource));
1264             return STATUS_SHARING_PAUSED;
1265         }
1266         // update caller's structures & exit
1267         fi->Dloc = Dloc;
1268         UDFReleaseDloc(Vcb, Dloc);
1269 #if defined UDF_DBG && !defined _CONSOLE
1270         if(fi->Dloc->CommonFcb) {
1271             ASSERT((uint32)(fi->Dloc->CommonFcb) != 0xDEADDA7A);
1272             ASSERT(fi->Dloc->CommonFcb->CommonFCBHeader.NodeTypeCode == UDF_NODE_TYPE_NT_REQ_FCB);
1273         }
1274 #endif // UDF_DBG
1275         UDFReleaseResource(&(Vcb->DlocResource));
1276         return STATUS_SUCCESS;
1277     }
1278     // allocate common DataLocation (Dloc) descriptor
1279     Dloc = fi->Dloc = (PUDF_DATALOC_INFO)MyAllocatePoolTag__(UDF_DATALOC_INFO_MT, sizeof(UDF_DATALOC_INFO), MEM_DLOC_INF_TAG);
1280     if(!Dloc) {
1281         UDFReleaseResource(&(Vcb->DlocResource));
1282         return STATUS_INSUFFICIENT_RESOURCES;
1283     }
1284     Vcb->DlocList[i].Lba = Lba;
1285     Vcb->DlocList[i].Dloc = Dloc;
1286     RtlZeroMemory(Dloc, sizeof(UDF_DATALOC_INFO));
1287     Dloc->LinkedFileInfo = fi;
1288     UDFAcquireDloc(Vcb, Dloc);
1289     UDFReleaseResource(&(Vcb->DlocResource));
1290     return STATUS_SUCCESS;
1291 } // end UDFStoreDloc()
1292 
1293 /*
1294     Remove unreferenced FE location from cache & free allocated memory
1295     This routine must be invoked when there are no more opened files
1296     associated with given FE
1297  */
1298 OSSTATUS
1299 UDFRemoveDloc(
1300     IN PVCB Vcb,
1301     IN PUDF_DATALOC_INFO Dloc
1302     )
1303 {
1304     LONG i;
1305 
1306     UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
1307 
1308     if((i = UDFFindDlocInMem(Vcb, Dloc)) == (-1)) {
1309         // FE specified is not in cache. exit
1310         UDFReleaseResource(&(Vcb->DlocResource));
1311         return STATUS_INVALID_PARAMETER;
1312     }
1313     // remove from cache
1314     ASSERT(Vcb->DlocList);
1315     RtlZeroMemory(&(Vcb->DlocList[i]), sizeof(UDF_DATALOC_INDEX));
1316     UDFReleaseResource(&(Vcb->DlocResource));
1317     MyFreePool__(Dloc);
1318     return STATUS_SUCCESS;
1319 } // end UDFRemoveDloc()
1320 
1321 /*
1322     Remove unlinked FE location from cache & keep allocated memory
1323     This routine must be invoked when there are no more opened files
1324     associated with given FE
1325  */
1326 OSSTATUS
1327 UDFUnlinkDloc(
1328     IN PVCB Vcb,
1329     IN PUDF_DATALOC_INFO Dloc
1330     )
1331 {
1332     LONG i;
1333 
1334     UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
1335 
1336     if((i = UDFFindDlocInMem(Vcb, Dloc)) == (-1)) {
1337         // FE specified is not in cache. exit
1338         UDFReleaseResource(&(Vcb->DlocResource));
1339         return STATUS_INVALID_PARAMETER;
1340     }
1341     // remove from cache
1342     ASSERT(Vcb->DlocList);
1343     RtlZeroMemory(&(Vcb->DlocList[i]), sizeof(UDF_DATALOC_INDEX));
1344     UDFReleaseResource(&(Vcb->DlocResource));
1345     return STATUS_SUCCESS;
1346 } // end UDFUnlinkDloc()
1347 
1348 /*
1349     This routine releases memory allocated for Dloc & removes it from
1350     cache (if it is still there)
1351  */
1352 void
1353 UDFFreeDloc(
1354     IN PVCB Vcb,
1355     IN PUDF_DATALOC_INFO Dloc
1356     )
1357 {
1358     LONG i;
1359 
1360     UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
1361 
1362     if((i = UDFFindDlocInMem(Vcb, Dloc)) != (-1)) {
1363         ASSERT(Vcb->DlocList);
1364         RtlZeroMemory(&(Vcb->DlocList[i]), sizeof(UDF_DATALOC_INDEX));
1365     }
1366     UDFReleaseResource(&(Vcb->DlocResource));
1367     MyFreePool__(Dloc);
1368 } // end UDFFreeDloc()
1369 
1370 /*
1371     This routine updates Dloc LBA after relocation
1372  */
1373 void
1374 UDFRelocateDloc(
1375     IN PVCB Vcb,
1376     IN PUDF_DATALOC_INFO Dloc,
1377     IN uint32 NewLba
1378     )
1379 {
1380     LONG i;
1381 
1382     UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
1383 
1384     if((i = UDFFindDlocInMem(Vcb, Dloc)) != (-1)) {
1385         ASSERT(Vcb->DlocList);
1386         Vcb->DlocList[i].Lba = NewLba;
1387     }
1388     UDFReleaseResource(&(Vcb->DlocResource));
1389 
1390 } // end UDFRelocateDloc()
1391 
1392 /*
1393     Release FE cache
1394  */
1395 void
1396 UDFReleaseDlocList(
1397     IN PVCB Vcb
1398     )
1399 {
1400     if(!Vcb->DlocList) return;
1401     UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
1402     for(uint32 i=0; i<Vcb->DlocCount; i++) {
1403         if(Vcb->DlocList[i].Dloc)
1404             MyFreePool__(Vcb->DlocList[i].Dloc);
1405     }
1406     MyFreePool__(Vcb->DlocList);
1407     Vcb->DlocList = NULL;
1408     Vcb->DlocCount = 0;
1409     UDFReleaseResource(&(Vcb->DlocResource));
1410 } // end UDFReleaseDlocList()
1411 
1412 /*
1413     This routine walks through Linked/Parallel FI chain and looks for
1414     FE with same Index & Parent File
1415  */
1416 PUDF_FILE_INFO
1417 UDFLocateParallelFI(
1418     PUDF_FILE_INFO di,  // parent FileInfo
1419     uint_di i,            // Index
1420     PUDF_FILE_INFO fi   // FileInfo to start search from
1421     )
1422 {
1423     PUDF_FILE_INFO    ParFileInfo = fi->NextLinkedFile;
1424 //    PUDF_DATALOC_INFO Dloc = di->Dloc;
1425     while((ParFileInfo != fi) &&
1426           ((ParFileInfo->ParentFile != di) ||
1427            (ParFileInfo->Index != i)) ) {
1428         ParFileInfo = ParFileInfo->NextLinkedFile;
1429     }
1430     return ParFileInfo;
1431 //   BrutePoint();
1432 } // end UDFLocateParallelFI()
1433 
1434 /*
1435     This routine walks through Linked/Parallel FI chain and looks for
1436     FE with same Index & Parent Dloc
1437  */
1438 PUDF_FILE_INFO
1439 UDFLocateAnyParallelFI(
1440     PUDF_FILE_INFO fi   // FileInfo to start search from
1441     )
1442 {
1443     if(!fi->ParentFile) {
1444         if(fi->NextLinkedFile == fi)
1445             return NULL;
1446         return fi->NextLinkedFile;
1447     }
1448     PUDF_FILE_INFO    ParFileInfo = fi->NextLinkedFile;
1449     PUDF_DATALOC_INFO Dloc = fi->ParentFile->Dloc;
1450     uint_di i = fi->Index;
1451     BOOLEAN NotFound = TRUE;
1452     while((ParFileInfo != fi) &&
1453           (NotFound =
1454            ((ParFileInfo->Index != i) ||
1455             (ParFileInfo->ParentFile->Dloc != Dloc))) ) {
1456         ParFileInfo = ParFileInfo->NextLinkedFile;
1457     }
1458 /*    if(NotFound) {
1459         if((ParFileInfo->Index == i) &&
1460            (ParFileInfo->ParentFile->Dloc == Dloc))
1461             return ParFileInfo;
1462         return NULL;
1463     }
1464     return ParFileInfo;*/
1465     return NotFound ? NULL : ParFileInfo;
1466 //   BrutePoint();
1467 } // end UDFLocateAnyParallelFI()
1468 
1469 void
1470 UDFInsertLinkedFile(
1471     PUDF_FILE_INFO fi,   // FileInfo to be added to chain
1472     PUDF_FILE_INFO fi2   // any FileInfo fro the chain
1473     )
1474 {
1475     fi->NextLinkedFile = fi2->NextLinkedFile;
1476     fi->PrevLinkedFile = fi2;
1477     fi->NextLinkedFile->PrevLinkedFile =
1478     fi->PrevLinkedFile->NextLinkedFile = fi;
1479     return;
1480 } // end UDFInsertLinkedFile()
1481 
1482