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 extent.cpp 10 11 Abstract: 12 13 This file contains filesystem-specific routines 14 responsible for extent & mapping management 15 16 */ 17 18 #include "udf.h" 19 20 #define UDF_BUG_CHECK_ID UDF_FILE_UDF_INFO_EXTENT 21 22 /* 23 This routine converts offset in extent to Lba & returns offset in the 1st 24 sector & bytes before end of block. 25 Here we assume no references to AllocDescs 26 */ 27 uint32 28 UDFExtentOffsetToLba( 29 IN PVCB Vcb, 30 IN PEXTENT_MAP Extent, // Extent array 31 IN int64 Offset, // offset in extent 32 OUT uint32* SectorOffset, 33 OUT PSIZE_T AvailLength, // available data in this block 34 OUT uint32* Flags, 35 OUT uint32* Index 36 ) 37 { 38 uint32 j=0, l, d, BSh = Vcb->BlockSizeBits; 39 uint32 Offs; 40 uint32 i=0, BOffset; // block nums 41 42 BOffset = (uint32)(Offset >> BSh); 43 // scan extent table for suitable range (frag) 44 ExtPrint(("ExtLen %x\n", Extent->extLength)); 45 while(i+(d = (l = (Extent->extLength & UDF_EXTENT_LENGTH_MASK)) >> BSh) <= BOffset) { 46 47 if(!l) { 48 if(Index) (*Index) = j-1; 49 if(Flags) { 50 Extent--; 51 (*Flags) = (Extent->extLength >> 30); 52 } 53 return LBA_OUT_OF_EXTENT; 54 } 55 if(!d) 56 break; 57 i += d; //frag offset 58 j++; // frag index 59 Extent++; 60 } 61 BOffset -= i; 62 Offs = (*((uint32*)&Offset)) - (i << BSh); // offset in frag 63 64 if(SectorOffset) 65 (*SectorOffset) = Offs & (Vcb->BlockSize-1);// offset in 1st Lba 66 if(AvailLength) 67 (*AvailLength) = l - Offs;// bytes to EO frag 68 if(Flags) 69 (*Flags) = (Extent->extLength >> 30); 70 if(Index) 71 (*Index) = j; 72 73 ASSERT(((Extent->extLength >> 30) == EXTENT_NOT_RECORDED_NOT_ALLOCATED) || Extent->extLocation); 74 75 return Extent->extLocation + BOffset;// 1st Lba 76 } // end UDFExtentOffsetToLba() 77 78 uint32 79 UDFNextExtentToLba( 80 IN PVCB Vcb, 81 IN PEXTENT_MAP Extent, // Extent array 82 OUT PSIZE_T AvailLength, // available data in this block 83 OUT uint32* Flags, 84 OUT uint32* Index 85 ) 86 { 87 // uint32 Lba; 88 89 uint32 l; 90 // uint32 d; 91 92 // scan extent table for suitable range (frag) 93 // d = (l = (Extent->extLength & UDF_EXTENT_LENGTH_MASK)); 94 l = (Extent->extLength & UDF_EXTENT_LENGTH_MASK); 95 96 if(!l) { 97 (*Index) = -1; 98 Extent--; 99 (*Flags) = (Extent->extLength >> 30); 100 return LBA_OUT_OF_EXTENT; 101 } 102 103 (*Index) = 0; 104 (*AvailLength) = l;// bytes to EO frag 105 (*Flags) = (Extent->extLength >> 30); 106 107 ASSERT(((*Flags) == EXTENT_NOT_RECORDED_NOT_ALLOCATED) || Extent->extLocation); 108 109 return Extent->extLocation;// 1st Lba 110 } // end UDFNextExtentToLba() 111 112 /* 113 This routine locates frag containing specified Lba in extent 114 */ 115 ULONG 116 UDFLocateLbaInExtent( 117 IN PVCB Vcb, 118 IN PEXTENT_MAP Extent, // Extent array 119 IN lba_t lba 120 ) 121 { 122 uint32 l, BSh = Vcb->BlockSizeBits; 123 uint32 i=0; 124 125 while((l = ((Extent->extLength & UDF_EXTENT_LENGTH_MASK) >> BSh))) { 126 127 if(Extent->extLocation >= lba && 128 Extent->extLocation+l < lba) { 129 return i; 130 } 131 i++; //frag offset 132 Extent++; 133 } 134 return LBA_OUT_OF_EXTENT;// index of item in extent, containing out Lba 135 } // end UDFLocateLbaInExtent() 136 137 /* 138 This routine calculates total length of specified extent. 139 Here we assume no references to AllocDescs 140 */ 141 int64 142 UDFGetExtentLength( 143 IN PEXTENT_MAP Extent // Extent array 144 ) 145 { 146 if(!Extent) return 0; 147 int64 i=0; 148 149 #if defined(_X86_) && defined(_MSC_VER) && !defined(__clang__) 150 151 __asm push ebx 152 __asm push ecx 153 __asm push esi 154 155 __asm lea ebx,i 156 __asm mov esi,Extent 157 __asm xor ecx,ecx 158 While_1: 159 __asm mov eax,[esi+ecx*8] // Extent[j].extLength 160 __asm and eax,UDF_EXTENT_LENGTH_MASK 161 __asm jz EO_While 162 __asm add [ebx],eax 163 __asm adc [ebx+4],0 164 __asm inc ecx 165 __asm jmp While_1 166 EO_While:; 167 __asm pop esi 168 __asm pop ecx 169 __asm pop ebx 170 171 #else // NO X86 optimization , use generic C/C++ 172 173 while(Extent->extLength) { 174 i += (Extent->extLength & UDF_EXTENT_LENGTH_MASK); 175 Extent++; 176 } 177 178 #endif // _X86_ 179 180 return i; 181 } // UDFGetExtentLength() 182 183 /* 184 This routine appends Zero-terminator to single Extent-entry. 185 Such operation makes it compatible with other internal routines 186 */ 187 PEXTENT_MAP 188 __fastcall 189 UDFExtentToMapping_( 190 IN PEXTENT_AD Extent 191 #ifdef UDF_TRACK_EXTENT_TO_MAPPING 192 ,IN ULONG src, 193 IN ULONG line 194 #endif //UDF_TRACK_EXTENT_TO_MAPPING 195 ) 196 { 197 PEXTENT_MAP Map; 198 199 #ifdef UDF_TRACK_EXTENT_TO_MAPPING 200 #define UDF_EXT_MAP_MULT 4 201 #else //UDF_TRACK_EXTENT_TO_MAPPING 202 #define UDF_EXT_MAP_MULT 2 203 #endif //UDF_TRACK_EXTENT_TO_MAPPING 204 205 Map = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , UDF_EXT_MAP_MULT * 206 sizeof(EXTENT_MAP), MEM_EXTMAP_TAG); 207 if(!Map) return NULL; 208 RtlZeroMemory((int8*)(Map+1), sizeof(EXTENT_MAP)); 209 Map[0].extLength = Extent->extLength; 210 Map[0].extLocation = Extent->extLocation; 211 #ifdef UDF_TRACK_EXTENT_TO_MAPPING 212 Map[2].extLength = src; 213 Map[2].extLocation = line; 214 #endif //UDF_TRACK_EXTENT_TO_MAPPING 215 return Map; 216 } // end UDFExtentToMapping() 217 218 /* 219 This routine calculates file mapping length (in bytes) including 220 ZERO-terminator 221 */ 222 uint32 223 UDFGetMappingLength( 224 IN PEXTENT_MAP Extent 225 ) 226 { 227 if(!Extent) return 0; 228 uint32 i=0; 229 230 #if defined(_X86_) && defined(_MSC_VER) && !defined(__clang__) 231 __asm push ebx 232 233 __asm mov ebx,Extent 234 __asm xor eax,eax 235 While_1: 236 __asm mov ecx,[ebx+eax*8] 237 __asm jecxz EO_While 238 __asm inc eax 239 __asm jmp While_1 240 EO_While: 241 __asm inc eax 242 __asm shl eax,3 243 __asm mov i,eax 244 245 __asm pop ebx 246 247 #else // NO X86 optimization , use generic C/C++ 248 249 while(Extent->extLength) { 250 i++; 251 Extent++; 252 } 253 i++; 254 i*=sizeof(EXTENT_MAP); 255 256 #endif // _X86_ 257 258 return i; // i*sizeof(EXTENT_MAP) 259 } // end UDFGetMappingLength() 260 261 /* 262 This routine merges 2 sequencial file mappings 263 */ 264 PEXTENT_MAP 265 __fastcall 266 UDFMergeMappings( 267 IN PEXTENT_MAP Extent, 268 IN PEXTENT_MAP Extent2 269 ) 270 { 271 PEXTENT_MAP NewExt; 272 uint32 len, len2; 273 274 len = UDFGetMappingLength(Extent); 275 len2 = UDFGetMappingLength(Extent2); 276 ASSERT(len2 && len); 277 if(!len2) { 278 return Extent; 279 } 280 if(MyReallocPool__((int8*)Extent, len, (int8**)(&NewExt), len+len2-sizeof(EXTENT_MAP))) { 281 RtlCopyMemory(((int8*)NewExt)+len-sizeof(EXTENT_MAP), (int8*)Extent2, len2); 282 } else { 283 ExtPrint(("UDFMergeMappings failed\n")); 284 BrutePoint(); 285 } 286 return NewExt; 287 } // end UDFMergeMappings() 288 289 /* 290 This routine builds file mapping according to ShortAllocDesc (SHORT_AD) 291 array 292 */ 293 PEXTENT_MAP 294 UDFShortAllocDescToMapping( 295 IN PVCB Vcb, 296 IN uint32 PartNum, 297 IN PSHORT_AD AllocDesc, 298 IN uint32 AllocDescLength, 299 IN uint32 SubCallCount, 300 OUT PEXTENT_INFO AllocLoc 301 ) 302 { 303 uint32 i, lim, l, len, type; 304 // uint32 BSh; 305 PEXTENT_MAP Extent, Extent2, AllocMap; 306 EXTENT_AD AllocExt; 307 PALLOC_EXT_DESC NextAllocDesc; 308 lb_addr locAddr; 309 SIZE_T ReadBytes; 310 EXTENT_INFO NextAllocLoc; 311 BOOLEAN w2k_compat = FALSE; 312 313 ExtPrint(("UDFShortAllocDescToMapping: len=%x\n", AllocDescLength)); 314 315 if(SubCallCount > ALLOC_DESC_MAX_RECURSE) return NULL; 316 317 locAddr.partitionReferenceNum = (uint16)PartNum; 318 // BSh = Vcb->BlockSizeBits; 319 l = ((lim = (AllocDescLength/sizeof(SHORT_AD))) + 1 ) * sizeof(EXTENT_AD); 320 Extent = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool, l, MEM_EXTMAP_TAG); 321 if(!Extent) return NULL; 322 323 NextAllocLoc.Offset = 0; 324 325 for(i=0;i<lim;i++) { 326 type = AllocDesc[i].extLength >> 30; 327 len = AllocDesc[i].extLength & UDF_EXTENT_LENGTH_MASK; 328 ExtPrint(("ShExt: type %x, loc %x, len %x\n", type, AllocDesc[i].extPosition, len)); 329 if(type == EXTENT_NEXT_EXTENT_ALLOCDESC) { 330 // read next frag of allocation descriptors if encountered 331 if(len < sizeof(ALLOC_EXT_DESC)) { 332 MyFreePool__(Extent); 333 return NULL; 334 } 335 NextAllocDesc = (PALLOC_EXT_DESC)MyAllocatePoolTag__(NonPagedPool, len, MEM_ALLOCDESC_TAG); 336 if(!NextAllocDesc) { 337 MyFreePool__(Extent); 338 return NULL; 339 } 340 // record information about this frag 341 locAddr.logicalBlockNum = AllocDesc[i].extPosition; 342 AllocExt.extLength = len; 343 AllocExt.extLocation = UDFPartLbaToPhys(Vcb, &locAddr); 344 if(AllocExt.extLocation == LBA_OUT_OF_EXTENT) { 345 UDFPrint(("bad address\n")); 346 MyFreePool__(NextAllocDesc); 347 MyFreePool__(Extent); 348 return NULL; 349 } 350 NextAllocLoc.Mapping = 351 AllocMap = UDFExtentToMapping(&AllocExt); 352 NextAllocLoc.Length = len; 353 if(!AllocMap) { 354 MyFreePool__(NextAllocDesc); 355 MyFreePool__(Extent); 356 return NULL; 357 } 358 AllocLoc->Mapping = UDFMergeMappings(AllocLoc->Mapping, AllocMap); 359 if(!AllocLoc->Mapping || 360 // read this frag 361 !OS_SUCCESS(UDFReadExtent(Vcb, &NextAllocLoc, 362 0, len, FALSE, (int8*)NextAllocDesc, &ReadBytes))) 363 { 364 MyFreePool__(AllocMap); 365 MyFreePool__(NextAllocDesc); 366 MyFreePool__(Extent); 367 return NULL; 368 } 369 MyFreePool__(AllocMap); 370 // check integrity 371 if((NextAllocDesc->descTag.tagIdent != TID_ALLOC_EXTENT_DESC) || 372 (NextAllocDesc->lengthAllocDescs > (len - sizeof(ALLOC_EXT_DESC))) ) { 373 UDFPrint(("Integrity check failed\n")); 374 UDFPrint(("NextAllocDesc->descTag.tagIdent = %x\n", NextAllocDesc->descTag.tagIdent)); 375 UDFPrint(("NextAllocDesc->lengthAllocDescs = %x\n", NextAllocDesc->lengthAllocDescs)); 376 UDFPrint(("len = %x\n", len)); 377 MyFreePool__(NextAllocDesc); 378 MyFreePool__(Extent); 379 return NULL; 380 } 381 // perform recursive call to obtain mapping 382 NextAllocLoc.Flags = 0; 383 Extent2 = UDFShortAllocDescToMapping(Vcb, PartNum, (PSHORT_AD)(NextAllocDesc+1), 384 NextAllocDesc->lengthAllocDescs, SubCallCount+1, AllocLoc); 385 if(!Extent2) { 386 MyFreePool__(NextAllocDesc); 387 MyFreePool__(Extent); 388 return NULL; 389 } 390 UDFCheckSpaceAllocation(Vcb, 0, Extent2, AS_USED); // check if used 391 // and merge this 2 mappings into 1 392 Extent[i].extLength = 0; 393 Extent[i].extLocation = 0; 394 Extent = UDFMergeMappings(Extent, Extent2); 395 396 if(NextAllocLoc.Flags & EXTENT_FLAG_2K_COMPAT) { 397 ExtPrint(("w2k-compat\n")); 398 AllocLoc->Flags |= EXTENT_FLAG_2K_COMPAT; 399 } 400 401 MyFreePool__(Extent2); 402 return Extent; 403 } 404 // 405 #ifdef UDF_CHECK_EXTENT_SIZE_ALIGNMENT 406 ASSERT(!(len & (Vcb->LBlockSize-1) )); 407 #endif //UDF_CHECK_EXTENT_SIZE_ALIGNMENT 408 if(len & (Vcb->LBlockSize-1)) { 409 w2k_compat = TRUE; 410 } 411 Extent[i].extLength = (len+Vcb->LBlockSize-1) & ~(Vcb->LBlockSize-1); 412 locAddr.logicalBlockNum = AllocDesc[i].extPosition; 413 // Note: for compatibility Adaptec DirectCD we check 'len' here 414 // That strange implementation records bogus extLocation in terminal entries 415 if(type != EXTENT_NOT_RECORDED_NOT_ALLOCATED && len) { 416 Extent[i].extLocation = UDFPartLbaToPhys(Vcb, &locAddr); 417 if(Extent[i].extLocation == LBA_OUT_OF_EXTENT) { 418 UDFPrint(("bad address (2)\n")); 419 MyFreePool__(Extent); 420 return NULL; 421 } 422 } else { 423 Extent[i].extLocation = 0; 424 } 425 if(!len) { 426 // some UDF implementations set strange AllocDesc sequence length, 427 // but terminates it with zeros in proper place, so handle 428 // this case 429 ASSERT(i>=(lim-1)); 430 ASSERT(!Extent[i].extLength); 431 Extent[i].extLocation = 0; 432 if(/*!SubCallCount &&*/ w2k_compat) { 433 ExtPrint(("w2k-compat\n")); 434 AllocLoc->Flags |= EXTENT_FLAG_2K_COMPAT; 435 } 436 return Extent; 437 } 438 Extent[i].extLength |= (type << 30); 439 } 440 // set terminator 441 Extent[i].extLength = 0; 442 Extent[i].extLocation = 0; 443 444 if(/*!SubCallCount &&*/ w2k_compat) { 445 ExtPrint(("w2k-compat\n")); 446 AllocLoc->Flags |= EXTENT_FLAG_2K_COMPAT; 447 } 448 449 return Extent; 450 } // end UDFShortAllocDescToMapping() 451 452 /* 453 This routine builds file mapping according to LongAllocDesc (LONG_AD) 454 array 455 */ 456 PEXTENT_MAP 457 UDFLongAllocDescToMapping( 458 IN PVCB Vcb, 459 IN PLONG_AD AllocDesc, 460 IN uint32 AllocDescLength, 461 IN uint32 SubCallCount, 462 OUT PEXTENT_INFO AllocLoc // .Mapping must be intialized (non-Zero) 463 ) 464 { 465 uint32 i, lim, l, len, type; 466 // uint32 BSh; 467 PEXTENT_MAP Extent, Extent2, AllocMap; 468 EXTENT_AD AllocExt; 469 PALLOC_EXT_DESC NextAllocDesc; 470 SIZE_T ReadBytes; 471 EXTENT_INFO NextAllocLoc; 472 473 ExtPrint(("UDFLongAllocDescToMapping: len=%x\n", AllocDescLength)); 474 475 if(SubCallCount > ALLOC_DESC_MAX_RECURSE) return NULL; 476 477 // BSh = Vcb->BlockSizeBits; 478 l = ((lim = (AllocDescLength/sizeof(LONG_AD))) + 1 ) * sizeof(EXTENT_AD); 479 Extent = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool, l, MEM_EXTMAP_TAG); 480 if(!Extent) return NULL; 481 482 NextAllocLoc.Offset = 0; 483 484 for(i=0;i<lim;i++) { 485 type = AllocDesc[i].extLength >> 30; 486 len = AllocDesc[i].extLength & UDF_EXTENT_LENGTH_MASK; 487 ExtPrint(("LnExt: type %x, loc %x (%x:%x), len %x\n", type, UDFPartLbaToPhys(Vcb,&(AllocDesc[i].extLocation)), 488 AllocDesc[i].extLocation.partitionReferenceNum, AllocDesc[i].extLocation.logicalBlockNum, 489 len)); 490 if(type == EXTENT_NEXT_EXTENT_ALLOCDESC) { 491 // read next frag of allocation descriptors if encountered 492 if(len < sizeof(ALLOC_EXT_DESC)) { 493 MyFreePool__(Extent); 494 return NULL; 495 } 496 NextAllocDesc = (PALLOC_EXT_DESC)MyAllocatePoolTag__(NonPagedPool, len, MEM_ALLOCDESC_TAG); 497 if(!NextAllocDesc) { 498 MyFreePool__(Extent); 499 return NULL; 500 } 501 // record information about this frag 502 AllocExt.extLength = len; 503 AllocExt.extLocation = UDFPartLbaToPhys(Vcb,&(AllocDesc[i].extLocation)); 504 if(AllocExt.extLocation == LBA_OUT_OF_EXTENT) { 505 UDFPrint(("bad address\n")); 506 MyFreePool__(NextAllocDesc); 507 MyFreePool__(Extent); 508 return NULL; 509 } 510 NextAllocLoc.Mapping = 511 AllocMap = UDFExtentToMapping(&AllocExt); 512 NextAllocLoc.Length = len; 513 if(!AllocMap) { 514 MyFreePool__(NextAllocDesc); 515 MyFreePool__(Extent); 516 return NULL; 517 } 518 AllocLoc->Mapping = UDFMergeMappings(AllocLoc->Mapping, AllocMap); 519 if(!AllocLoc->Mapping || 520 // read this frag 521 !OS_SUCCESS(UDFReadExtent(Vcb, &NextAllocLoc, 522 0, len, FALSE, (int8*)NextAllocDesc, &ReadBytes))) 523 { 524 MyFreePool__(AllocMap); 525 MyFreePool__(NextAllocDesc); 526 MyFreePool__(Extent); 527 return NULL; 528 } 529 MyFreePool__(AllocMap); 530 // check integrity 531 if((NextAllocDesc->descTag.tagIdent != TID_ALLOC_EXTENT_DESC) || 532 (NextAllocDesc->lengthAllocDescs > (len - sizeof(ALLOC_EXT_DESC))) ) { 533 UDFPrint(("Integrity check failed\n")); 534 UDFPrint(("NextAllocDesc->descTag.tagIdent = %x\n", NextAllocDesc->descTag.tagIdent)); 535 UDFPrint(("NextAllocDesc->lengthAllocDescs = %x\n", NextAllocDesc->lengthAllocDescs)); 536 UDFPrint(("len = %x\n", len)); 537 MyFreePool__(NextAllocDesc); 538 MyFreePool__(Extent); 539 return NULL; 540 } 541 // perform recursive call to obtain mapping 542 Extent2 = UDFLongAllocDescToMapping(Vcb, (PLONG_AD)(NextAllocDesc+1), 543 NextAllocDesc->lengthAllocDescs, SubCallCount+1, AllocLoc); 544 if(!Extent2) { 545 MyFreePool__(NextAllocDesc); 546 MyFreePool__(Extent); 547 return NULL; 548 } 549 // and merge this 2 mappings into 1 550 Extent[i].extLength = 0; 551 Extent[i].extLocation = 0; 552 Extent = UDFMergeMappings(Extent, Extent2); 553 MyFreePool__(Extent2); 554 return Extent; 555 } 556 // 557 Extent[i].extLength = len; 558 #ifdef UDF_CHECK_EXTENT_SIZE_ALIGNMENT 559 ASSERT(!(len & (Vcb->LBlockSize-1) )); 560 #endif //UDF_CHECK_EXTENT_SIZE_ALIGNMENT 561 Extent[i].extLength = (len+Vcb->LBlockSize-1) & ~(Vcb->LBlockSize-1); 562 // Note: for compatibility Adaptec DirectCD we check 'len' here 563 // That strange implementation records bogus extLocation in terminal entries 564 if(type != EXTENT_NOT_RECORDED_NOT_ALLOCATED && len) { 565 Extent[i].extLocation = UDFPartLbaToPhys(Vcb,&(AllocDesc[i].extLocation)); 566 if(Extent[i].extLocation == LBA_OUT_OF_EXTENT) { 567 UDFPrint(("bad address (2)\n")); 568 MyFreePool__(Extent); 569 return NULL; 570 } 571 } else { 572 Extent[i].extLocation = 0; 573 } 574 if(!len) { 575 // some UDF implementations set strange AllocDesc sequence length, 576 // but terminates it with zeros in proper place, so handle 577 // this case 578 Extent[i].extLocation = 0; 579 return Extent; 580 } 581 Extent[i].extLength |= (type << 30); 582 } 583 // set terminator 584 Extent[i].extLength = 0; 585 Extent[i].extLocation = 0; 586 587 return Extent; 588 } // end UDFLongAllocDescToMapping() 589 590 /* 591 This routine builds file mapping according to ExtendedAllocDesc (EXT_AD) 592 array 593 */ 594 PEXTENT_MAP 595 UDFExtAllocDescToMapping( 596 IN PVCB Vcb, 597 IN PEXT_AD AllocDesc, 598 IN uint32 AllocDescLength, 599 IN uint32 SubCallCount, 600 OUT PEXTENT_INFO AllocLoc // .Mapping must be intialized (non-Zero) 601 ) 602 { 603 uint32 i, lim, l, len, type; 604 // uint32 BSh; 605 PEXTENT_MAP Extent, Extent2, AllocMap; 606 EXTENT_AD AllocExt; 607 PALLOC_EXT_DESC NextAllocDesc; 608 SIZE_T ReadBytes; 609 EXTENT_INFO NextAllocLoc; 610 611 ExtPrint(("UDFExtAllocDescToMapping: len=%x\n", AllocDescLength)); 612 613 if(SubCallCount > ALLOC_DESC_MAX_RECURSE) return NULL; 614 615 // BSh = Vcb->BlockSizeBits; 616 l = ((lim = (AllocDescLength/sizeof(EXT_AD))) + 1 ) * sizeof(EXTENT_AD); 617 Extent = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool, l, MEM_EXTMAP_TAG); 618 if(!Extent) return NULL; 619 620 NextAllocLoc.Offset = 0; 621 622 for(i=0;i<lim;i++) { 623 type = AllocDesc[i].extLength >> 30; 624 len = AllocDesc[i].extLength & UDF_EXTENT_LENGTH_MASK; 625 ExtPrint(("ExExt: type %x, loc %x, len %x\n", type, UDFPartLbaToPhys(Vcb,&(AllocDesc[i].extLocation)), len)); 626 if(type == EXTENT_NEXT_EXTENT_ALLOCDESC) { 627 // read next frag of allocation descriptors if encountered 628 if(len < sizeof(ALLOC_EXT_DESC)) { 629 MyFreePool__(Extent); 630 return NULL; 631 } 632 NextAllocDesc = (PALLOC_EXT_DESC)MyAllocatePoolTag__(NonPagedPool, len, MEM_ALLOCDESC_TAG); 633 if(!NextAllocDesc) { 634 MyFreePool__(Extent); 635 return NULL; 636 } 637 // record information about this frag 638 AllocExt.extLength = len; 639 AllocExt.extLocation = UDFPartLbaToPhys(Vcb,&(AllocDesc[i].extLocation)); 640 if(AllocExt.extLocation == LBA_OUT_OF_EXTENT) { 641 UDFPrint(("bad address\n")); 642 MyFreePool__(NextAllocDesc); 643 MyFreePool__(Extent); 644 return NULL; 645 } 646 NextAllocLoc.Mapping = 647 AllocMap = UDFExtentToMapping(&AllocExt); 648 NextAllocLoc.Length = len; 649 if(!AllocMap) { 650 MyFreePool__(NextAllocDesc); 651 MyFreePool__(Extent); 652 return NULL; 653 } 654 AllocLoc->Mapping = UDFMergeMappings(AllocLoc->Mapping, AllocMap); 655 if(!AllocLoc->Mapping || 656 // read this frag 657 !OS_SUCCESS(UDFReadExtent(Vcb, &NextAllocLoc, 658 0, len, FALSE, (int8*)NextAllocDesc, &ReadBytes))) 659 { 660 MyFreePool__(AllocMap); 661 MyFreePool__(NextAllocDesc); 662 MyFreePool__(Extent); 663 return NULL; 664 } 665 MyFreePool__(AllocMap); 666 // check integrity 667 if((NextAllocDesc->descTag.tagIdent != TID_ALLOC_EXTENT_DESC) || 668 (NextAllocDesc->lengthAllocDescs > (len - sizeof(ALLOC_EXT_DESC))) ) { 669 UDFPrint(("Integrity check failed\n")); 670 MyFreePool__(NextAllocDesc); 671 MyFreePool__(Extent); 672 return NULL; 673 } 674 // perform recursive call to obtain mapping 675 Extent2 = UDFExtAllocDescToMapping(Vcb, (PEXT_AD)(NextAllocDesc+1), 676 NextAllocDesc->lengthAllocDescs, SubCallCount+1, AllocLoc); 677 if(!Extent2) { 678 MyFreePool__(NextAllocDesc); 679 MyFreePool__(Extent); 680 return NULL; 681 } 682 // and merge this 2 mappings into 1 683 Extent[i].extLength = 0; 684 Extent[i].extLocation = 0; 685 Extent = UDFMergeMappings(Extent, Extent2); 686 MyFreePool__(Extent2); 687 return Extent; 688 } 689 /* if((AllocDesc[i].extLength & UDF_EXTENT_LENGTH_MASK) > // Uncomment!!! 690 (AllocDesc[i].recordedLength & UDF_EXTENT_LENGTH_MASK)) { 691 Extent[i].extLength = AllocDesc[i].recordedLength; 692 Extent[i].extLocation = UDFPartLbaToPhys(Vcb,&(AllocDesc[i].extLocation)); 693 }*/ 694 Extent[i].extLength = len; 695 #ifdef UDF_CHECK_EXTENT_SIZE_ALIGNMENT 696 ASSERT(!(len & (Vcb->LBlockSize-1) )); 697 #endif //UDF_CHECK_EXTENT_SIZE_ALIGNMENT 698 // Note: for compatibility Adaptec DirectCD we check 'len' here 699 // That strange implementation records bogus extLocation in terminal entries 700 if(type != EXTENT_NOT_RECORDED_NOT_ALLOCATED && len) { 701 Extent[i].extLocation = UDFPartLbaToPhys(Vcb,&(AllocDesc[i].extLocation)); 702 if(Extent[i].extLocation == LBA_OUT_OF_EXTENT) { 703 UDFPrint(("bad address (2)\n")); 704 MyFreePool__(Extent); 705 return NULL; 706 } 707 } else { 708 Extent[i].extLocation = 0; 709 } 710 if(!len) { 711 // some UDF implementations set strange AllocDesc sequence length, 712 // but terminates it with zeros in proper place, so handle 713 // this case 714 Extent[i].extLocation = 0; 715 return Extent; 716 } 717 Extent[i].extLength |= (type << 30); 718 } 719 // set terminator 720 Extent[i].extLength = 0; 721 Extent[i].extLocation = 0; 722 723 return Extent; 724 } // end UDFExtAllocDescToMapping() 725 726 727 /* 728 This routine builds FileMapping according to given FileEntry 729 Return: pointer to EXTENT_MAP array 730 or offset inside FileEntry (negative) 731 when ICB_FLAG_AD_IN_ICB encountered 732 of NULL if an error occured 733 */ 734 PEXTENT_MAP 735 UDFReadMappingFromXEntry( 736 IN PVCB Vcb, 737 IN uint32 PartNum, 738 IN tag* XEntry, 739 IN OUT uint32* Offset, 740 OUT PEXTENT_INFO AllocLoc // .Mapping must be intialized (non-Zero) 741 ) 742 { 743 PEXTENT_AD Extent; 744 uint16 AllocMode; 745 int8* AllocDescs; 746 uint32 len; 747 // EntityID* eID; // for compatibility with Adaptec DirectCD 748 749 Extent = NULL; 750 (*Offset) = 0; 751 752 753 if(XEntry->tagIdent == TID_FILE_ENTRY) { 754 // UDFPrint(("Standard FileEntry\n")); 755 PFILE_ENTRY FileEntry = (PFILE_ENTRY)XEntry; 756 ExtPrint(("Standard FileEntry\n")); 757 758 AllocDescs = (int8*)(((int8*)(FileEntry+1))+(FileEntry->lengthExtendedAttr)); 759 len = FileEntry->lengthAllocDescs; 760 AllocLoc->Offset = sizeof(FILE_ENTRY) + FileEntry->lengthExtendedAttr; 761 // eID = &(FileEntry->impIdent); 762 763 AllocMode = FileEntry->icbTag.flags & ICB_FLAG_ALLOC_MASK; 764 765 } else if(XEntry->tagIdent == TID_EXTENDED_FILE_ENTRY) { 766 // UDFPrint(("Extended FileEntry\n")); 767 ExtPrint(("Extended FileEntry\n")); 768 PEXTENDED_FILE_ENTRY ExFileEntry = (PEXTENDED_FILE_ENTRY)XEntry; 769 770 AllocDescs = (((int8*)(ExFileEntry+1))+(ExFileEntry->lengthExtendedAttr)); 771 len = ExFileEntry->lengthAllocDescs; 772 AllocLoc->Offset = sizeof(EXTENDED_FILE_ENTRY) + ExFileEntry->lengthExtendedAttr; 773 // eID = &(FileEntry->impIdent); 774 775 AllocMode = ExFileEntry->icbTag.flags & ICB_FLAG_ALLOC_MASK; 776 777 } else { 778 return NULL; 779 } 780 781 // for compatibility with Adaptec DirectCD 782 // if(!(Vcb->UDF_VCB_IC_ADAPTEC_NONALLOC_COMPAT)) 783 784 AllocLoc->Length=len; 785 AllocLoc->Flags |= EXTENT_FLAG_VERIFY; // for metadata 786 787 switch (AllocMode) { 788 case ICB_FLAG_AD_SHORT: { 789 Extent = UDFShortAllocDescToMapping(Vcb, PartNum, (PSHORT_AD)AllocDescs, len, 0, AllocLoc); 790 break; 791 } 792 case ICB_FLAG_AD_LONG: { 793 Extent = UDFLongAllocDescToMapping(Vcb, (PLONG_AD)AllocDescs, len, 0, AllocLoc); 794 break; 795 } 796 case ICB_FLAG_AD_EXTENDED: { 797 Extent = UDFExtAllocDescToMapping(Vcb, (PEXT_AD)AllocDescs, len, 0, AllocLoc); 798 break; 799 } 800 default : { // case ICB_FLAG_AD_IN_ICB 801 Extent = NULL; 802 *Offset = (uintptr_t)AllocDescs - (uintptr_t)XEntry; 803 AllocLoc->Offset=0; 804 AllocLoc->Length=0; 805 if(AllocLoc->Mapping) MyFreePool__(AllocLoc->Mapping); 806 AllocLoc->Mapping=NULL; 807 break; 808 } 809 } 810 811 ExtPrint(("UDFReadMappingFromXEntry: mode %x, loc %x, len %x\n", AllocMode, 812 AllocLoc->Mapping ? AllocLoc->Mapping[0].extLocation : -1, len)); 813 814 UDFCheckSpaceAllocation(Vcb, 0, Extent, AS_USED); // check if used 815 816 return Extent; 817 }// end UDFReadMappingFromXEntry() 818 819 #ifndef UDF_READ_ONLY_BUILD 820 /* 821 This routine builds data for AllocDesc sequence for specified 822 extent 823 */ 824 OSSTATUS 825 UDFBuildShortAllocDescs( 826 IN PVCB Vcb, 827 IN uint32 PartNum, 828 OUT int8** Buff, // data for AllocLoc 829 IN uint32 InitSz, 830 IN OUT PUDF_FILE_INFO FileInfo 831 ) 832 { 833 uint32 i, j; 834 uint32 len=0; 835 PEXTENT_MAP Extent = FileInfo->Dloc->DataLoc.Mapping; 836 PEXTENT_INFO AllocExtent = &(FileInfo->Dloc->AllocLoc); 837 PSHORT_AD Alloc; 838 uint32 NewLen; 839 OSSTATUS status; 840 uint32 ph_len=0; // in general, this should be uint64, 841 // but we need its lower part only 842 #ifdef UDF_ALLOW_FRAG_AD 843 uint32 ts, ac, len2; 844 uint32 LBS = Vcb->LBlockSize; 845 uint32 LBSh = Vcb->BlockSizeBits; 846 uint32 TagLen = 0; 847 tag* Tag = NULL; 848 PSHORT_AD saved_Alloc; 849 uint32 TagLoc, prevTagLoc; 850 uint32 BufOffs; 851 uint32 ExtOffs; 852 uint32 saved_NewLen; 853 #endif //UDF_ALLOW_FRAG_AD 854 855 ValidateFileInfo(FileInfo); 856 ExtPrint(("UDFBuildShortAllocDescs: FE %x\n", FileInfo->Dloc->FELoc.Mapping[0].extLocation)); 857 // calculate length 858 for(len=0; (i=(Extent[len].extLength & UDF_EXTENT_LENGTH_MASK)); len++, ph_len+=i) { 859 ExtPrint(("bShExt: type %x, loc %x, len %x\n", 860 Extent[len].extLength >> 30, Extent[len].extLocation, Extent[len].extLength & UDF_EXTENT_LENGTH_MASK)); 861 } 862 Alloc = (PSHORT_AD)MyAllocatePoolTag__(NonPagedPool, (len+1)*sizeof(SHORT_AD), MEM_SHAD_TAG); 863 if(!Alloc) { 864 BrutePoint(); 865 return STATUS_INSUFFICIENT_RESOURCES; 866 } 867 // fill contiguous AllocDesc buffer (decribing UserData) 868 for(i=0;i<len;i++) { 869 Alloc[i].extLength = Extent[i].extLength; 870 Alloc[i].extPosition = UDFPhysLbaToPart(Vcb, PartNum, Extent[i].extLocation); 871 } 872 if((Vcb->CompatFlags & UDF_VCB_IC_W2K_COMPAT_ALLOC_DESCS) && i) { 873 Alloc[i-1].extLength -= (ph_len - (ULONG)(FileInfo->Dloc->DataLoc.Length)) & 874 (Vcb->LBlockSize-1); 875 ExtPrint(("bShExt: cut tail -> %x\n", 876 Alloc[i-1].extLength & UDF_EXTENT_LENGTH_MASK)); 877 } 878 Alloc[i].extLength = 879 Alloc[i].extPosition = 0; 880 j = len*sizeof(SHORT_AD); // required space 881 len = (InitSz & ~(sizeof(SHORT_AD)-1)); // space available in 1st block 882 ASSERT(len == InitSz); 883 884 // Ok. Let's init AllocLoc 885 if(!(FileInfo->Dloc->AllocLoc.Mapping)) { 886 FileInfo->Dloc->AllocLoc.Mapping = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool, 2 * sizeof(EXTENT_MAP), MEM_EXTMAP_TAG); 887 if(!(FileInfo->Dloc->AllocLoc.Mapping)) { 888 BrutePoint(); 889 MyFreePool__(Alloc); 890 return STATUS_INSUFFICIENT_RESOURCES; 891 } 892 // allocation descriptors are located in the same sector as FileEntry 893 // (at least their 1st part), just after it 894 FileInfo->Dloc->AllocLoc.Mapping[0] = FileInfo->Dloc->FELoc.Mapping[0]; 895 FileInfo->Dloc->AllocLoc.Offset = FileInfo->Dloc->FileEntryLen; 896 FileInfo->Dloc->AllocLoc.Length = 0; 897 // set terminator 898 FileInfo->Dloc->AllocLoc.Mapping[1].extLength = 899 FileInfo->Dloc->AllocLoc.Mapping[1].extLocation = 0; 900 } 901 902 if(j <= len) { 903 // we needn't allocating additional blocks to store AllocDescs 904 AdPrint(("in-ICB AllocDescs, j=%x\n",j)); 905 RtlCopyMemory(*Buff, (int8*)Alloc, j); 906 NewLen = j; 907 MyFreePool__(Alloc); 908 } else { 909 #ifndef UDF_ALLOW_FRAG_AD 910 AdPrint((" DISK_FULL\n")); 911 return STATUS_DISK_FULL; 912 #else //UDF_ALLOW_FRAG_AD 913 AdPrint(("multi-block AllocDescs, j=%x\n",j)); 914 BufOffs = 0; 915 TagLoc = prevTagLoc = 0; 916 // calculate the space available for SHORT_ADs in each block 917 ac = (LBS - (sizeof(ALLOC_EXT_DESC) + sizeof(SHORT_AD))) & ~(sizeof(SHORT_AD)-1); 918 len2 = len; 919 // tail size 920 ts = InitSz - len2; 921 len -= sizeof(SHORT_AD); 922 // calculate actual AllocSequence length (in bytes) 923 NewLen = ( ((j - len + ac - 1) / ac) << LBSh) + InitSz + sizeof(SHORT_AD); 924 MyFreePool__(*Buff); 925 (*Buff) = (int8*)MyAllocatePoolTag__(NonPagedPool, NewLen, MEM_SHAD_TAG); 926 if(!(*Buff)) { 927 status = STATUS_INSUFFICIENT_RESOURCES; 928 UDFPrint(("UDFResizeExtent() failed (%x)\n",status)); 929 BrutePoint(); 930 goto sh_alloc_err; 931 } 932 if(UDFGetExtentLength(AllocExtent->Mapping) < NewLen) { 933 status = UDFResizeExtent(Vcb, PartNum, NewLen, TRUE, AllocExtent); 934 if(!OS_SUCCESS(status)) { 935 UDFPrint(("UDFResizeExtent(2) failed (%x)\n",status)); 936 BrutePoint(); 937 sh_alloc_err: 938 MyFreePool__(Alloc); 939 return status; 940 } 941 } 942 ExtOffs = AllocExtent->Offset; 943 RtlZeroMemory(*Buff, NewLen); 944 saved_NewLen = NewLen; 945 NewLen = 0; // recorded length 946 saved_Alloc = Alloc; 947 // fill buffer sector by sector (adding links at the end of each one) 948 while(TRUE) { 949 950 // j - remained AllocDescs length (in bytes) 951 // len - bytes available for AllocDescs in current block 952 // ac - bytes available for AllocDescs in each block 953 954 // leave space for terminator or pointer to next part of sequence 955 if(j == len2) { 956 // if we have only 1 SHORT_AD that we can fit in last sector 957 // we shall do it instead of recording link & allocating new block 958 len = 959 TagLen = len2; 960 } 961 ASSERT(saved_NewLen >= (BufOffs + len)); 962 RtlCopyMemory( (*Buff)+BufOffs, (int8*)Alloc, len); 963 Alloc = (PSHORT_AD)((int8*)Alloc + len); 964 j -= len; 965 BufOffs += len; 966 if(Tag) { 967 // Set up Tag for AllocDesc 968 Tag->tagIdent = TID_ALLOC_EXTENT_DESC; 969 UDFSetUpTag(Vcb, Tag, (uint16)TagLen, TagLoc); 970 prevTagLoc = TagLoc; 971 } 972 if(!j) { 973 // terminate loop 974 NewLen = BufOffs; 975 break; 976 } 977 len = ac; 978 if(j <= (len + sizeof(SHORT_AD))) 979 len = j - sizeof(SHORT_AD); 980 len2 = len + sizeof(SHORT_AD); 981 // we have more than 1 SHORT_AD that we can't fit in current block 982 // so we shall set up pointer to the next block 983 ((PSHORT_AD)((*Buff)+BufOffs))->extLength = /*LBS*/ len2 | 984 (((uint32)EXTENT_NEXT_EXTENT_ALLOCDESC) << 30) ; 985 ((PSHORT_AD)((*Buff)+BufOffs))->extPosition = TagLoc = 986 UDFPhysLbaToPart(Vcb, PartNum, 987 UDFExtentOffsetToLba(Vcb, AllocExtent->Mapping, 988 ExtOffs+BufOffs+sizeof(SHORT_AD)+ts, 989 NULL, NULL, NULL, NULL) ); 990 // reflect additional (link) block & LBlock tail (if any) 991 BufOffs += ts+sizeof(SHORT_AD); 992 // init AllocDesc 993 ( (PALLOC_EXT_DESC) ((*Buff)+BufOffs))->lengthAllocDescs = len2; 994 ( (PALLOC_EXT_DESC) ((*Buff)+BufOffs))->previousAllocExtLocation = prevTagLoc; 995 Tag = (tag*)((*Buff)+BufOffs); 996 TagLen = len2; 997 ts = LBS-len2-sizeof(ALLOC_EXT_DESC); 998 BufOffs += sizeof(ALLOC_EXT_DESC); 999 } 1000 MyFreePool__(saved_Alloc); 1001 #endif //UDF_ALLOW_FRAG_AD 1002 } 1003 status = UDFResizeExtent(Vcb, PartNum, NewLen, TRUE, AllocExtent); 1004 return status; 1005 } // end UDFBuildShortAllocDescs() 1006 1007 /* 1008 This routine builds data for AllocDesc sequence for specified 1009 extent 1010 */ 1011 OSSTATUS 1012 UDFBuildLongAllocDescs( 1013 IN PVCB Vcb, 1014 IN uint32 PartNum, 1015 OUT int8** Buff, // data for AllocLoc 1016 IN uint32 InitSz, 1017 IN OUT PUDF_FILE_INFO FileInfo 1018 ) 1019 { 1020 uint32 i, j; 1021 uint32 len=0; 1022 PEXTENT_MAP Extent = FileInfo->Dloc->DataLoc.Mapping; 1023 PEXTENT_INFO AllocExtent = &(FileInfo->Dloc->AllocLoc); 1024 PLONG_AD Alloc; 1025 uint32 NewLen; 1026 OSSTATUS status; 1027 uint32 ph_len=0; // in general, this should be uint64, 1028 // but we need its lower part only 1029 #ifdef UDF_ALLOW_FRAG_AD 1030 uint32 ac, len2, ts; 1031 uint32 TagLoc, prevTagLoc; 1032 uint32 LBS = Vcb->LBlockSize; 1033 uint32 LBSh = Vcb->BlockSizeBits; 1034 uint32 BufOffs; 1035 uint32 ExtOffs = AllocExtent->Offset; 1036 PLONG_AD saved_Alloc; 1037 uint32 TagLen = 0; 1038 tag* Tag = NULL; 1039 #endif //UDF_ALLOW_FRAG_AD 1040 1041 ValidateFileInfo(FileInfo); 1042 ExtPrint(("UDFBuildLongAllocDescs: FE %x\n", FileInfo->Dloc->FELoc.Mapping[0].extLocation)); 1043 // calculate length 1044 //for(len=0; i=(Extent[len].extLength & UDF_EXTENT_LENGTH_MASK); len++, ph_len+=i); 1045 for(len=0; (i=(Extent[len].extLength & UDF_EXTENT_LENGTH_MASK)); len++, ph_len+=i) { 1046 ExtPrint(("bLnExt: type %x, loc %x, len %x\n", 1047 Extent[len].extLength >> 30, Extent[len].extLocation, Extent[len].extLength & UDF_EXTENT_LENGTH_MASK)); 1048 } 1049 Alloc = (PLONG_AD)MyAllocatePoolTag__(NonPagedPool, (len+1)*sizeof(LONG_AD), MEM_LNGAD_TAG); 1050 if(!Alloc) return STATUS_INSUFFICIENT_RESOURCES; 1051 // fill contiguous AllocDesc buffer (decribing UserData) 1052 for(i=0;i<len;i++) { 1053 Alloc[i].extLength = Extent[i].extLength; 1054 Alloc[i].extLocation.logicalBlockNum = UDFPhysLbaToPart(Vcb, PartNum, Extent[i].extLocation); 1055 Alloc[i].extLocation.partitionReferenceNum = (uint16)PartNum; 1056 RtlZeroMemory(&(Alloc[i].impUse), sizeof(Alloc[i].impUse)); 1057 } 1058 if((Vcb->CompatFlags & UDF_VCB_IC_W2K_COMPAT_ALLOC_DESCS) && i) { 1059 Alloc[i-1].extLength -= (ph_len - (ULONG)(FileInfo->Dloc->DataLoc.Length)) & 1060 (Vcb->LBlockSize-1); 1061 ExtPrint(("bLnExt: cut tail -> %x\n", 1062 Alloc[i-1].extLength & UDF_EXTENT_LENGTH_MASK)); 1063 } 1064 RtlZeroMemory(&(Alloc[i]), sizeof(LONG_AD)); 1065 j = len*sizeof(LONG_AD); // required space 1066 len = (InitSz & ~(sizeof(LONG_AD)-1)); // space available in 1st block 1067 ASSERT(len == InitSz); 1068 1069 // Ok. Let's init AllocLoc 1070 if(!(FileInfo->Dloc->AllocLoc.Mapping)) { 1071 FileInfo->Dloc->AllocLoc.Mapping = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool, 2 * sizeof(EXTENT_MAP), MEM_EXTMAP_TAG); 1072 if(!(FileInfo->Dloc->AllocLoc.Mapping)) { 1073 MyFreePool__(Alloc); 1074 return STATUS_INSUFFICIENT_RESOURCES; 1075 } 1076 // allocation descriptors are located in the same sector as FileEntry 1077 // (at least their 1st part), just after it 1078 FileInfo->Dloc->AllocLoc.Mapping[0] = FileInfo->Dloc->FELoc.Mapping[0]; 1079 FileInfo->Dloc->AllocLoc.Offset = FileInfo->Dloc->FileEntryLen; 1080 FileInfo->Dloc->AllocLoc.Length = 0; 1081 // set terminator 1082 FileInfo->Dloc->AllocLoc.Mapping[1].extLength = 1083 FileInfo->Dloc->AllocLoc.Mapping[1].extLocation = 0; 1084 } 1085 1086 if(j <= len) { 1087 // we needn't allocating additional blocks to store AllocDescs 1088 RtlCopyMemory(*Buff, (int8*)Alloc, j); 1089 NewLen = j; 1090 MyFreePool__(Alloc); 1091 } else { 1092 #ifndef UDF_ALLOW_FRAG_AD 1093 AdPrint((" DISK_FULL\n")); 1094 return STATUS_DISK_FULL; 1095 #else //UDF_ALLOW_FRAG_AD 1096 BufOffs = 0; 1097 TagLoc = prevTagLoc = 0; 1098 // calculate the space available for LONG_ADs in each block 1099 ac = (LBS - (sizeof(ALLOC_EXT_DESC) + sizeof(LONG_AD))) & ~(sizeof(LONG_AD)-1); 1100 len2 = len; 1101 // tail size 1102 ts = InitSz - len2; 1103 len -= sizeof(LONG_AD); 1104 // calculate actual AllocSequence length (in LBlocks) 1105 NewLen = ( ((j - len + ac - 1) / ac) << LBSh) + InitSz + sizeof(LONG_AD); 1106 MyFreePool__(*Buff); 1107 (*Buff) = (int8*)MyAllocatePoolTag__(NonPagedPool, NewLen, MEM_LNGAD_TAG); 1108 if(!(*Buff)) { 1109 status = STATUS_INSUFFICIENT_RESOURCES; 1110 goto lad_alloc_err; 1111 } 1112 if(UDFGetExtentLength(AllocExtent->Mapping) < NewLen) { 1113 status = UDFResizeExtent(Vcb, PartNum, NewLen, TRUE, AllocExtent); 1114 if(!OS_SUCCESS(status)) { 1115 lad_alloc_err: 1116 MyFreePool__(Alloc); 1117 return status; 1118 } 1119 } 1120 ExtOffs = AllocExtent->Offset; 1121 RtlZeroMemory(*Buff, NewLen); 1122 NewLen = 0; // recorded length 1123 saved_Alloc = Alloc; 1124 len2 = len+sizeof(LONG_AD); 1125 // fill buffer sector by sector (adding links at the end of each one) 1126 while(TRUE) { 1127 1128 // j - remained AllocDescs length (in bytes) 1129 // len - bytes available for in AllocDescs each block 1130 1131 // leave space for terminator or pointer to next part of sequence 1132 if(j == len2) { 1133 // if we have only 1 LONG_AD that we can fit in last sector 1134 // we shall do it instead of recording link & allocating new block 1135 len = 1136 TagLen = len2; 1137 } 1138 RtlCopyMemory( (*Buff)+BufOffs, (int8*)Alloc, len); 1139 Alloc = (PLONG_AD)((int8*)Alloc + len); 1140 j -= len; 1141 BufOffs += len; 1142 if(Tag) { 1143 // Set up Tag for AllocDesc 1144 Tag->tagIdent = TID_ALLOC_EXTENT_DESC; 1145 UDFSetUpTag(Vcb, Tag, (uint16)TagLen, TagLoc); 1146 prevTagLoc = TagLoc; 1147 } 1148 if(!j) { 1149 // terminate loop 1150 NewLen = BufOffs; 1151 break; 1152 } 1153 len = ac; 1154 if(j <= (len + sizeof(LONG_AD))) 1155 len = j - sizeof(LONG_AD); 1156 len2 = len+sizeof(LONG_AD); 1157 // we have more than 1 LONG_AD that we can't fit in current block 1158 // so we shall set up pointer to the next block 1159 ((PLONG_AD)((*Buff)+BufOffs))->extLength = /*LBS*/ len2 | 1160 (((uint32)EXTENT_NEXT_EXTENT_ALLOCDESC) << 30) ; 1161 ((PLONG_AD)((*Buff)+BufOffs))->extLocation.logicalBlockNum = TagLoc = 1162 UDFPhysLbaToPart(Vcb, PartNum, 1163 UDFExtentOffsetToLba(Vcb, AllocExtent->Mapping, 1164 ExtOffs+BufOffs+sizeof(LONG_AD)+ts, 1165 NULL, NULL, NULL, NULL) ); 1166 ((PLONG_AD)((*Buff)+BufOffs))->extLocation.partitionReferenceNum = (uint16)PartNum; 1167 // reflect additional (link) block & LBlock tail (if any) 1168 BufOffs += ts+sizeof(LONG_AD); 1169 // init AllocDesc 1170 ( (PALLOC_EXT_DESC) ((*Buff)+BufOffs))->lengthAllocDescs = len2; 1171 ( (PALLOC_EXT_DESC) ((*Buff)+BufOffs))->previousAllocExtLocation = prevTagLoc; 1172 Tag = (tag*)((*Buff)+BufOffs); 1173 TagLen = len2; 1174 ts = LBS-len2-sizeof(ALLOC_EXT_DESC); 1175 BufOffs += sizeof(ALLOC_EXT_DESC); 1176 } 1177 MyFreePool__(saved_Alloc); 1178 #endif //UDF_ALLOW_FRAG_AD 1179 } 1180 status = UDFResizeExtent(Vcb, PartNum, NewLen, TRUE, AllocExtent); 1181 return status; 1182 } // end UDFBuildLongAllocDescs() 1183 1184 /* 1185 This routine builds data for AllocDesc sequence for specified 1186 extent 1187 */ 1188 /*OSSTATUS 1189 UDFBuildExtAllocDescs( 1190 IN PVCB Vcb, 1191 IN uint32 PartNum, 1192 OUT int8** Buff, // data for AllocLoc 1193 IN uint32 InitSz, 1194 IN OUT PUDF_FILE_INFO FileInfo 1195 ) 1196 { 1197 uint32 i, j; 1198 uint32 len=0, ac, len2; 1199 uint32 TagLoc, prevTagLoc; 1200 uint32 LBS = Vcb->LBlockSize; 1201 uint32 LBSh = Vcb->BlockSizeBits; 1202 PEXTENT_MAP Extent = FileInfo->Dloc->DataLoc.Mapping; 1203 PEXTENT_INFO AllocExtent = &(FileInfo->Dloc->AllocLoc); 1204 PEXT_AD Alloc, saved_Alloc; 1205 uint32 BufOffs; 1206 uint32 ExtOffs = AllocExtent->Offset; 1207 uint32 NewLen; 1208 OSSTATUS status; 1209 uint32 TagLen = 0; 1210 tag* Tag = NULL; 1211 1212 ValidateFileInfo(FileInfo); 1213 // calculate length 1214 for(len=0; Extent[len].extLength; len++); 1215 Alloc = (PEXT_AD)MyAllocatePool__(NonPagedPool, (len+1)*sizeof(EXT_AD)); 1216 if(!Alloc) return STATUS_INSUFFICIENT_RESOURCES; 1217 // fill contiguous AllocDesc buffer (decribing UserData) 1218 for(i=0;i<len;i++) { 1219 Alloc[i].extLength = 1220 Alloc[i].recordedLength = 1221 Alloc[i].informationLength = Extent[i].extLength; 1222 Alloc[i].extLocation.logicalBlockNum = UDFPhysLbaToPart(Vcb, PartNum, Extent[i].extLocation); 1223 Alloc[i].extLocation.partitionReferenceNum = (uint16)PartNum; 1224 } 1225 RtlZeroMemory(&(Alloc[i]), sizeof(EXT_AD)); 1226 j = len*sizeof(EXT_AD); // required space 1227 len = InitSz; // space available in 1st block 1228 1229 // Ok. Let's init AllocLoc 1230 if(!(FileInfo->Dloc->AllocLoc.Mapping)) { 1231 FileInfo->Dloc->AllocLoc.Mapping = (PEXTENT_MAP)MyAllocatePool__(NonPagedPool, 2 * sizeof(EXTENT_MAP)); 1232 if(!(FileInfo->Dloc->AllocLoc.Mapping)) { 1233 MyFreePool__(Alloc); 1234 return STATUS_INSUFFICIENT_RESOURCES; 1235 } 1236 // allocation descriptors are located in the same sector as FileEntry 1237 // (at least their 1st part), just after it 1238 FileInfo->Dloc->AllocLoc.Mapping[0] = FileInfo->Dloc->FELoc.Mapping[0]; 1239 FileInfo->Dloc->AllocLoc.Offset = FileInfo->Dloc->FileEntryLen; 1240 FileInfo->Dloc->AllocLoc.Length = 0; 1241 // set terminator 1242 FileInfo->Dloc->AllocLoc.Mapping[1].extLength = 1243 FileInfo->Dloc->AllocLoc.Mapping[1].extLocation = 0; 1244 } 1245 1246 if(j <= len) { 1247 // we needn't allocating additional blocks to store AllocDescs 1248 RtlCopyMemory(*Buff, (int8*)Alloc, j); 1249 NewLen = j; 1250 MyFreePool__(Alloc); 1251 } else { 1252 BufOffs = 0; 1253 TagLoc = prevTagLoc = 0; 1254 // calculate the space available for EXT_ADs in each block 1255 ac = (LBS - (sizeof(ALLOC_EXT_DESC) + sizeof(EXT_AD))) & ~(sizeof(EXT_AD)-1); 1256 // calculate actual AllocSequence length (in LBlocks) 1257 len -= sizeof(EXT_AD); 1258 NewLen = ( ((j - len + ac - 1) / ac) << LBSh) + len + sizeof(EXT_AD); 1259 MyFreePool__(*Buff); 1260 (*Buff) = (int8*)MyAllocatePool__(NonPagedPool, NewLen); 1261 if(UDFGetExtentLength(AllocExtent->Mapping) < NewLen) { 1262 status = UDFResizeExtent(Vcb, PartNum, NewLen, TRUE, AllocExtent); 1263 if(!OS_SUCCESS(status)) { 1264 MyFreePool__(Alloc); 1265 return status; 1266 } 1267 } 1268 RtlZeroMemory(*Buff, NewLen); 1269 NewLen = 0; // recorded length 1270 saved_Alloc = Alloc; 1271 len2 = len + sizeof(EXT_AD); 1272 // fill buffer sector by sector (adding links at the end of each one) 1273 while(TRUE) { 1274 1275 // j - remained AllocDescs length (in bytes) 1276 // len - bytes available for in AllocDescs each block 1277 1278 // leave space for terminator or pointer to next part of sequence 1279 if(j == len2) { 1280 // if we have only 1 EXT_AD that we can fit in last sector 1281 // we shall do it instead of recording link & allocating new block 1282 len = 1283 TagLen = len2; 1284 } 1285 RtlCopyMemory( (*Buff)+BufOffs, (int8*)Alloc, len); 1286 Alloc = (PEXT_AD)((int8*)Alloc + len); 1287 j -= len; 1288 BufOffs += len; 1289 if(Tag) { 1290 // Set up Tag for AllocDesc 1291 Tag->tagIdent = TID_ALLOC_EXTENT_DESC; 1292 UDFSetUpTag(Vcb, Tag, (uint16)TagLen, TagLoc); 1293 prevTagLoc = TagLoc; 1294 } 1295 if(!j) { 1296 // terminate loop 1297 NewLen = BufOffs; 1298 break; 1299 } 1300 len = ac; 1301 if(j <= (len + sizeof(EXT_AD))) 1302 len = j - sizeof(EXT_AD); 1303 len2 = len + sizeof(EXT_AD); 1304 // we have more than 1 EXT_AD that we can't fit in current block 1305 // so we shall set up pointer to the next block 1306 ((PEXT_AD)((*Buff)+BufOffs))->extLength = 1307 ((PEXT_AD)((*Buff)+BufOffs))->recordedLength = LBS; 1308 ((PEXT_AD)((*Buff)+BufOffs))->informationLength = len2 | 1309 (((uint32)EXTENT_NEXT_EXTENT_ALLOCDESC) << 30) ; 1310 ((PEXT_AD)((*Buff)+BufOffs))->extLocation.logicalBlockNum = TagLoc = 1311 UDFPhysLbaToPart(Vcb, PartNum, 1312 UDFExtentOffsetToLba(Vcb, AllocExtent->Mapping, ExtOffs + BufOffs + 2*sizeof(EXT_AD)-1, NULL, NULL, NULL, NULL) ); 1313 ((PEXT_AD)((*Buff)+BufOffs))->extLocation.partitionReferenceNum = (uint16)PartNum; 1314 BufOffs = (BufOffs + 2*sizeof(EXT_AD) - 1) & ~(sizeof(EXT_AD)-1) ; 1315 // init AllocDesc 1316 ( (PALLOC_EXT_DESC) ((*Buff)+BufOffs))->lengthAllocDescs = len2; 1317 ( (PALLOC_EXT_DESC) ((*Buff)+BufOffs))->previousAllocExtLocation = prevTagLoc; 1318 Tag = (tag*)((*Buff)+BufOffs); 1319 TagLen = len2; 1320 BufOffs += sizeof(ALLOC_EXT_DESC); 1321 } 1322 MyFreePool__(saved_Alloc); 1323 } 1324 status = UDFResizeExtent(Vcb, PartNum, NewLen, TRUE, AllocExtent); 1325 return status; 1326 } // end UDFBuildExtAllocDescs()*/ 1327 1328 void 1329 UDFDiscardFESpace( 1330 IN PVCB Vcb, 1331 IN PEXTENT_MAP Mapping, 1332 IN uint32 lim 1333 ) 1334 { 1335 #ifdef UDF_FE_ALLOCATION_CHARGE // UDF_FE_ALLOCATION_CHARGE 1336 PEXTENT_MAP Mapping2; 1337 uint32 i; 1338 1339 UDFPrint((" DiscardFESpace\n")); 1340 Mapping2 = Mapping; 1341 for(i=0;i<lim;i++, Mapping++) { 1342 // we should not discard allocated FEs 1343 if( (Mapping->extLength >> 30) == EXTENT_RECORDED_ALLOCATED) { 1344 UDFPrint((" used @ %x\n", Mapping->extLocation)); 1345 Mapping->extLength = Vcb->LBlockSize | (EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30); 1346 Mapping->extLocation = 0; 1347 } else { 1348 UDFPrint((" free @ %x\n", Mapping->extLocation)); 1349 } 1350 } 1351 UDFMarkSpaceAsXXX(Vcb, 0, Mapping2, AS_DISCARDED); 1352 1353 MyFreePool__(Mapping2); 1354 #else // UDF_FE_ALLOCATION_CHARGE 1355 ASSERT(!Dloc->DirIndex->FECharge.Mapping); 1356 return; 1357 #endif // UDF_FE_ALLOCATION_CHARGE 1358 } // end UDFDiscardFESpace() 1359 1360 OSSTATUS 1361 UDFInitAllocationCache( 1362 IN PVCB Vcb, 1363 IN uint32 AllocClass, 1364 OUT PUDF_ALLOCATION_CACHE_ITEM* _AllocCache, 1365 OUT uint32* _lim, 1366 IN BOOLEAN Init 1367 ) 1368 { 1369 PUDF_ALLOCATION_CACHE_ITEM AllocCache; 1370 PUDF_ALLOCATION_CACHE_ITEM* pAllocCache; 1371 uint32 i, lim; 1372 uint32* plim; 1373 1374 switch(AllocClass) { 1375 case UDF_PREALLOC_CLASS_FE: 1376 UDFPrint(("AllocationCache FE:\n")); 1377 pAllocCache = &(Vcb->FEChargeCache); 1378 plim = &(Vcb->FEChargeCacheMaxSize); 1379 lim = 32; 1380 break; 1381 case UDF_PREALLOC_CLASS_DIR: 1382 UDFPrint(("AllocationCache DIR:\n")); 1383 pAllocCache = &(Vcb->PreallocCache); 1384 plim = &(Vcb->PreallocCacheMaxSize); 1385 lim = 32; 1386 break; 1387 default: 1388 return STATUS_INVALID_PARAMETER; 1389 } 1390 if(!(*plim)) { 1391 if(!Init) { 1392 return STATUS_UNSUCCESSFUL; 1393 } 1394 (*pAllocCache) = AllocCache = 1395 (PUDF_ALLOCATION_CACHE_ITEM) 1396 MyAllocatePoolTag__(NonPagedPool , sizeof(UDF_ALLOCATION_CACHE_ITEM)*lim, 1397 MEM_ALLOC_CACHE_TAG); 1398 if(!AllocCache) { 1399 return STATUS_INSUFFICIENT_RESOURCES; 1400 } 1401 RtlZeroMemory(AllocCache, sizeof(UDF_ALLOCATION_CACHE_ITEM)*lim); 1402 for(i=0; i<lim; i++) { 1403 AllocCache[i].ParentLocation = LBA_NOT_ALLOCATED; 1404 } 1405 (*plim) = lim; 1406 } else { 1407 lim = (*plim); 1408 AllocCache = (*pAllocCache); 1409 } 1410 (*_lim) = lim; 1411 (*_AllocCache) = AllocCache; 1412 1413 return STATUS_SUCCESS; 1414 } // end UDFInitAllocationCache() 1415 1416 OSSTATUS 1417 UDFGetCachedAllocation( 1418 IN PVCB Vcb, 1419 IN uint32 ParentLocation, 1420 OUT PEXTENT_INFO Ext, 1421 OUT uint32* Items, // optional 1422 IN uint32 AllocClass 1423 ) 1424 { 1425 PUDF_ALLOCATION_CACHE_ITEM AllocCache; 1426 uint32 i, lim; 1427 OSSTATUS status; 1428 1429 UDFAcquireResourceExclusive(&(Vcb->PreallocResource),TRUE); 1430 1431 status = UDFInitAllocationCache(Vcb, AllocClass, &AllocCache, &lim, FALSE); 1432 if(!OS_SUCCESS(status)) { 1433 UDFReleaseResource(&(Vcb->PreallocResource)); 1434 return status; 1435 } 1436 UDFPrint(("Get AllocationCache for %x\n", ParentLocation)); 1437 1438 for(i=0; i<lim; i++) { 1439 if(AllocCache[i].ParentLocation == ParentLocation) { 1440 (*Ext) = AllocCache[i].Ext; 1441 AdPrint((" map %x (%x)\n", Ext->Mapping, i)); 1442 if(Items) { 1443 (*Items) = AllocCache[i].Items; 1444 } 1445 RtlZeroMemory(&(AllocCache[i]), sizeof(AllocCache[i])); 1446 AllocCache[i].ParentLocation = LBA_NOT_ALLOCATED; 1447 UDFReleaseResource(&(Vcb->PreallocResource)); 1448 return STATUS_SUCCESS; 1449 } 1450 } 1451 AdPrint((" no map\n")); 1452 UDFReleaseResource(&(Vcb->PreallocResource)); 1453 return STATUS_UNSUCCESSFUL; 1454 } // end UDFGetCachedAllocation() 1455 1456 OSSTATUS 1457 UDFStoreCachedAllocation( 1458 IN PVCB Vcb, 1459 IN uint32 ParentLocation, 1460 IN PEXTENT_INFO Ext, 1461 IN uint32 Items, 1462 IN uint32 AllocClass 1463 ) 1464 { 1465 PUDF_ALLOCATION_CACHE_ITEM AllocCache; 1466 uint32 i, lim; 1467 OSSTATUS status; 1468 1469 UDFAcquireResourceExclusive(&(Vcb->PreallocResource),TRUE); 1470 1471 status = UDFInitAllocationCache(Vcb, AllocClass, &AllocCache, &lim, TRUE); 1472 if(!OS_SUCCESS(status)) { 1473 UDFReleaseResource(&(Vcb->PreallocResource)); 1474 return status; 1475 } 1476 UDFPrint(("Store AllocationCache for %x, map %x\n", ParentLocation, Ext->Mapping)); 1477 1478 for(i=0; i<lim; i++) { 1479 if(AllocCache[i].ParentLocation == LBA_NOT_ALLOCATED) { 1480 AdPrint((" stored in %x\n", i)); 1481 AllocCache[i].Ext = (*Ext); 1482 AllocCache[i].Items = Items; 1483 AllocCache[i].ParentLocation = ParentLocation; 1484 UDFReleaseResource(&(Vcb->PreallocResource)); 1485 return STATUS_SUCCESS; 1486 } 1487 } 1488 // 1489 AdPrint((" drop map %x (%x)\n", AllocCache[lim-1].Ext.Mapping, lim-1)); 1490 switch(AllocClass) { 1491 case UDF_PREALLOC_CLASS_FE: 1492 UDFDiscardFESpace(Vcb, AllocCache[lim-1].Ext.Mapping, AllocCache[lim-1].Items); 1493 break; 1494 case UDF_PREALLOC_CLASS_DIR: 1495 UDFMarkSpaceAsXXX(Vcb, 0, AllocCache[lim-1].Ext.Mapping, AS_DISCARDED); 1496 break; 1497 } 1498 RtlMoveMemory(&(AllocCache[1]), &(AllocCache[0]), sizeof(UDF_ALLOCATION_CACHE_ITEM)*(lim-1)); 1499 AllocCache[0].Ext = (*Ext); 1500 AllocCache[0].Items = Items; 1501 AllocCache[0].ParentLocation = ParentLocation; 1502 AdPrint((" stored in 0\n")); 1503 UDFReleaseResource(&(Vcb->PreallocResource)); 1504 return STATUS_SUCCESS; 1505 } // end UDFStoreCachedAllocation() 1506 1507 OSSTATUS 1508 UDFFlushAllCachedAllocations( 1509 IN PVCB Vcb, 1510 IN uint32 AllocClass 1511 ) 1512 { 1513 PUDF_ALLOCATION_CACHE_ITEM AllocCache; 1514 uint32 i, lim; 1515 OSSTATUS status; 1516 1517 UDFPrint(("Flush AllocationCache\n")); 1518 UDFAcquireResourceExclusive(&(Vcb->PreallocResource),TRUE); 1519 1520 status = UDFInitAllocationCache(Vcb, AllocClass, &AllocCache, &lim, FALSE); 1521 if(!OS_SUCCESS(status)) { 1522 UDFReleaseResource(&(Vcb->PreallocResource)); 1523 return status; 1524 } 1525 1526 for(i=0; i<lim; i++) { 1527 if(AllocCache[i].ParentLocation != LBA_NOT_ALLOCATED) { 1528 switch(AllocClass) { 1529 case UDF_PREALLOC_CLASS_FE: 1530 UDFDiscardFESpace(Vcb, AllocCache[i].Ext.Mapping, AllocCache[i].Items); 1531 break; 1532 case UDF_PREALLOC_CLASS_DIR: 1533 UDFMarkSpaceAsXXX(Vcb, 0, AllocCache[i].Ext.Mapping, AS_DISCARDED); 1534 break; 1535 } 1536 } 1537 } 1538 MyFreePool__(AllocCache); 1539 switch(AllocClass) { 1540 case UDF_PREALLOC_CLASS_FE: 1541 Vcb->FEChargeCache = NULL; 1542 Vcb->FEChargeCacheMaxSize = 0; 1543 break; 1544 case UDF_PREALLOC_CLASS_DIR: 1545 Vcb->PreallocCache = NULL; 1546 Vcb->PreallocCacheMaxSize = 0; 1547 break; 1548 } 1549 UDFReleaseResource(&(Vcb->PreallocResource)); 1550 // 1551 return STATUS_SUCCESS; 1552 } // end UDFFlushAllCachedAllocations() 1553 1554 /* 1555 This routine allocates space for FE of the file being created 1556 If FE-Charge is enabled it reserves an extent & allocates 1557 space in it. It works much faster then usual way both while 1558 allocating & accessing on disk 1559 If FE-Charge is disabled FE may be allocated at any suitable 1560 location 1561 */ 1562 OSSTATUS 1563 UDFAllocateFESpace( 1564 IN PVCB Vcb, 1565 IN PUDF_FILE_INFO DirInfo, 1566 IN uint32 PartNum, 1567 IN PEXTENT_INFO FEExtInfo, 1568 IN uint32 Len 1569 ) 1570 { 1571 #ifdef UDF_FE_ALLOCATION_CHARGE // UDF_FE_ALLOCATION_CHARGE 1572 OSSTATUS status; 1573 PEXTENT_INFO Ext; 1574 EXTENT_AD Extent; 1575 BOOLEAN retry = FALSE; 1576 uint32 i, lim; 1577 1578 /* 1579 1. #Dir1#->*File* -> Dir1's FECharge 1580 2. #Dir1#->*Dir* -> Dir1's FECharge 1581 3. #Dir1#->*SDir* -> Dir1's FECharge 1582 4. Dir1->#SDir#->*Stream* -> Dir1's FEChargeSDir 1583 5. Dir1->#File#->*SDir* -> Dir1's FEChargeSDir 1584 6. Dir1->#Dir#->*SDir* -> (see p.2) 1585 7. Dir1->File->#SDir#->*Stream* -> Dir1's FEChargeSDir 1586 8. Dir1->Dir->#SDir#->*Stream* -> (see p.4) 1587 1588 ## ~ DirInfo 1589 ** ~ Object to be created 1590 1591 */ 1592 1593 // ASSERT(!FEExtInfo->Mapping); 1594 // check if DirInfo we are called with is a Directory 1595 // (it can be a file with SDir) 1596 if(!DirInfo || !DirInfo->Dloc->DirIndex || 1597 ((lim = ((DirInfo->Dloc->FE_Flags & UDF_FE_FLAG_IS_SDIR) ? Vcb->FEChargeSDir : Vcb->FECharge)) <= 1)) 1598 #endif // UDF_FE_ALLOCATION_CHARGE 1599 return UDFAllocFreeExtent(Vcb, Len, 1600 UDFPartStart(Vcb, PartNum), UDFPartEnd(Vcb, PartNum), FEExtInfo, EXTENT_FLAG_VERIFY); 1601 #ifdef UDF_FE_ALLOCATION_CHARGE // UDF_FE_ALLOCATION_CHARGE 1602 1603 Ext = &(DirInfo->Dloc->DirIndex->FECharge); 1604 1605 while(TRUE) { 1606 1607 if(!Ext->Mapping) { 1608 ULONG p_start; 1609 ULONG p_end; 1610 ULONG fe_loc; 1611 ULONG l1, l2; 1612 1613 p_start = UDFPartStart(Vcb, PartNum); 1614 p_end = UDFPartEnd(Vcb, PartNum); 1615 fe_loc = DirInfo->Dloc->FELoc.Mapping[0].extLocation; 1616 1617 status = UDFGetCachedAllocation(Vcb, fe_loc, Ext, NULL, UDF_PREALLOC_CLASS_FE); 1618 if(OS_SUCCESS(status)) { 1619 // do nothing, even do not unpack 1620 } else 1621 if(Vcb->LowFreeSpace) { 1622 status = UDFAllocFreeExtent(Vcb, Len << Vcb->LBlockSizeBits,p_start, p_end, FEExtInfo, EXTENT_FLAG_VERIFY); 1623 if(OS_SUCCESS(status)) { 1624 UDFPrint(("FE @ %x (1)\n", FEExtInfo->Mapping[0].extLocation )); 1625 } 1626 return status; 1627 } else { 1628 if(fe_loc > p_start + 512*16) { 1629 l1 = fe_loc - 512*16; 1630 } else { 1631 l1 = p_start; 1632 } 1633 if(fe_loc + 512*16 < p_end) { 1634 l2 = fe_loc + 512*16; 1635 } else { 1636 l2 = p_end; 1637 } 1638 status = UDFAllocFreeExtent(Vcb, lim << Vcb->LBlockSizeBits, l1, l2, Ext, EXTENT_FLAG_VERIFY); 1639 if(!OS_SUCCESS(status)) { 1640 status = UDFAllocFreeExtent(Vcb, lim << Vcb->LBlockSizeBits, (p_start+fe_loc)/2, (fe_loc+p_end)/2, Ext, EXTENT_FLAG_VERIFY); 1641 } 1642 if(!OS_SUCCESS(status)) { 1643 status = UDFAllocFreeExtent(Vcb, lim << Vcb->LBlockSizeBits, p_start, p_end, Ext, EXTENT_FLAG_VERIFY); 1644 } 1645 if(!OS_SUCCESS(status)) { 1646 status = UDFAllocFreeExtent(Vcb, lim << Vcb->LBlockSizeBits, p_start+1024, p_end-1024, Ext, EXTENT_FLAG_VERIFY); 1647 } 1648 if(!OS_SUCCESS(status = UDFAllocFreeExtent(Vcb, lim << Vcb->LBlockSizeBits, p_start, p_end, Ext, EXTENT_FLAG_VERIFY) )) { 1649 // can't pre-allocate space for multiple FEs. Try single FE 1650 UDFPrint(("allocate single FE entry\n")); 1651 status = UDFAllocFreeExtent(Vcb, Len, 1652 p_start, p_end, FEExtInfo, EXTENT_FLAG_VERIFY); 1653 if(OS_SUCCESS(status)) { 1654 UDFPrint(("FE @ %x (2)\n", FEExtInfo->Mapping[0].extLocation )); 1655 } 1656 return status; 1657 } 1658 status = UDFUnPackMapping(Vcb, Ext); 1659 if(!OS_SUCCESS(status)) { 1660 MyFreePool__(Ext->Mapping); 1661 Ext->Mapping = NULL; 1662 return status; 1663 } 1664 } 1665 } 1666 1667 for(i=0;i<lim;i++) { 1668 if( (Ext->Mapping[i].extLength >> 30) == EXTENT_NOT_RECORDED_ALLOCATED ) { 1669 Ext->Mapping[i].extLength &= UDF_EXTENT_LENGTH_MASK; // EXTENT_RECORDED_ALLOCATED 1670 1671 Extent.extLength = Vcb->LBlockSize | (EXTENT_NOT_RECORDED_ALLOCATED << 30); 1672 Extent.extLocation = Ext->Mapping[i].extLocation; 1673 1674 if(Vcb->BSBM_Bitmap) { 1675 uint32 lba = Ext->Mapping[i].extLocation; 1676 if(UDFGetBadBit((uint32*)(Vcb->BSBM_Bitmap), lba)) { 1677 UDFPrint(("Remove BB @ %x from FE charge\n", lba)); 1678 Ext->Mapping[i].extLength |= (EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30); 1679 Ext->Mapping[i].extLocation = 0; 1680 continue; 1681 } 1682 } 1683 1684 FEExtInfo->Mapping = UDFExtentToMapping(&Extent); 1685 if(!FEExtInfo->Mapping) { 1686 ASSERT(!(Ext->Mapping[i].extLength >> 30)); 1687 Ext->Mapping[i].extLength |= (EXTENT_NOT_RECORDED_ALLOCATED << 30); 1688 return STATUS_INSUFFICIENT_RESOURCES; 1689 } 1690 UDFPrint(("FE @ %x (3)\n", FEExtInfo->Mapping[0].extLocation )); 1691 FEExtInfo->Length = Len; 1692 FEExtInfo->Offset = 0; 1693 FEExtInfo->Modified = TRUE; 1694 return STATUS_SUCCESS; 1695 } 1696 } 1697 1698 if(Vcb->LowFreeSpace) { 1699 status = UDFAllocFreeExtent(Vcb, Len, 1700 UDFPartStart(Vcb, PartNum), UDFPartEnd(Vcb, PartNum), FEExtInfo, EXTENT_FLAG_VERIFY); 1701 if(OS_SUCCESS(status)) { 1702 UDFPrint(("FE @ %x (4)\n", FEExtInfo->Mapping[0].extLocation )); 1703 } 1704 return status; 1705 } 1706 if(retry) 1707 return STATUS_INSUFFICIENT_RESOURCES; 1708 1709 // we can get here if there are no free slots in 1710 // preallocated FE charge. So, we should release 1711 // memory and try to allocate space for new FE charge. 1712 MyFreePool__(Ext->Mapping); 1713 Ext->Mapping = NULL; 1714 retry = TRUE; 1715 } 1716 return STATUS_INSUFFICIENT_RESOURCES; 1717 #endif // UDF_FE_ALLOCATION_CHARGE 1718 1719 } // end UDFAllocateFESpace() 1720 1721 /* 1722 This routine frees space allocated for FE. 1723 */ 1724 void 1725 UDFFreeFESpace( 1726 IN PVCB Vcb, 1727 IN PUDF_FILE_INFO DirInfo, 1728 IN PEXTENT_INFO FEExtInfo 1729 ) 1730 { 1731 #ifdef UDF_FE_ALLOCATION_CHARGE // UDF_FE_ALLOCATION_CHARGE 1732 PEXTENT_INFO Ext; 1733 uint32 i, lim, j=-1; 1734 uint32 Lba; 1735 1736 // check if the DirInfo we are called with is a Directory 1737 // (it can be a file with SDir) 1738 if(DirInfo && DirInfo->Dloc->DirIndex && 1739 (Ext = &(DirInfo->Dloc->DirIndex->FECharge))->Mapping) { 1740 if(!FEExtInfo->Mapping) 1741 return; 1742 Lba = FEExtInfo->Mapping[0].extLocation; 1743 1744 lim = (DirInfo->Dloc->FE_Flags & UDF_FE_FLAG_IS_SDIR) ? Vcb->FEChargeSDir : Vcb->FECharge; 1745 for(i=0;i<lim;i++) { 1746 if(Ext->Mapping[i].extLocation == Lba) { 1747 ASSERT(!(Ext->Mapping[i].extLength >> 30)); 1748 Ext->Mapping[i].extLength |= (EXTENT_NOT_RECORDED_ALLOCATED << 30); 1749 goto clean_caller; 1750 } 1751 if(!Ext->Mapping[i].extLocation) { 1752 j = i; 1753 } 1754 } 1755 if(j != (ULONG)-1) { 1756 i = j; 1757 Ext->Mapping[i].extLocation = Lba; 1758 Ext->Mapping[i].extLength = Vcb->LBlockSize | (EXTENT_NOT_RECORDED_ALLOCATED << 30); 1759 goto clean_caller; 1760 } 1761 } 1762 #endif // UDF_FE_ALLOCATION_CHARGE 1763 UDFMarkSpaceAsXXX(Vcb, 0, FEExtInfo->Mapping, AS_DISCARDED); // free 1764 clean_caller: 1765 FEExtInfo->Mapping[0].extLocation = 0; 1766 FEExtInfo->Mapping[0].extLength = (EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30); 1767 return; 1768 } // end UDFFreeFESpace() 1769 #endif //UDF_READ_ONLY_BUILD 1770 1771 /* 1772 This routine flushes FE-Charge buffer, marks unused blocks as free 1773 in bitmap & releases memory allocated for FE-Charge management 1774 */ 1775 void 1776 UDFFlushFESpace( 1777 IN PVCB Vcb, 1778 IN PUDF_DATALOC_INFO Dloc, 1779 IN BOOLEAN Discard 1780 ) 1781 { 1782 #ifdef UDF_FE_ALLOCATION_CHARGE // UDF_FE_ALLOCATION_CHARGE 1783 PEXTENT_MAP Mapping; 1784 uint32 lim; 1785 1786 if(!(Mapping = Dloc->DirIndex->FECharge.Mapping)) 1787 return; 1788 1789 lim = (Dloc->FE_Flags & UDF_FE_FLAG_IS_SDIR) ? Vcb->FEChargeSDir : Vcb->FECharge; 1790 1791 if(!Discard) { 1792 // cache it! 1793 if(OS_SUCCESS(UDFStoreCachedAllocation(Vcb, 1794 Dloc->FELoc.Mapping[0].extLocation, 1795 &Dloc->DirIndex->FECharge, lim, UDF_PREALLOC_CLASS_FE))) { 1796 Dloc->DirIndex->FECharge.Mapping = NULL; 1797 return; 1798 } 1799 } 1800 Dloc->DirIndex->FECharge.Mapping = NULL; 1801 UDFDiscardFESpace(Vcb, Mapping, lim); 1802 #else // UDF_FE_ALLOCATION_CHARGE 1803 ASSERT(!Dloc->DirIndex->FECharge.Mapping); 1804 return; 1805 #endif // UDF_FE_ALLOCATION_CHARGE 1806 } // end UDFFlushFESpace() 1807 1808 #ifndef UDF_READ_ONLY_BUILD 1809 /* 1810 This routine rebuilds mapping on write attempts to Alloc-Not-Rec area. 1811 Here we assume that required area lays in a single frag. 1812 */ 1813 OSSTATUS 1814 UDFMarkAllocatedAsRecorded( 1815 IN PVCB Vcb, 1816 IN int64 Offset, 1817 IN uint32 Length, 1818 IN PEXTENT_INFO ExtInfo // Extent array 1819 ) 1820 { 1821 uint32 i, len, lba, sLen; 1822 PEXTENT_MAP Extent = ExtInfo->Mapping; // Extent array 1823 PEXTENT_MAP NewExtent; 1824 uint32 BS = Vcb->BlockSize; 1825 uint32 LBS = Vcb->LBlockSize; 1826 uint32 BSh = Vcb->BlockSizeBits; 1827 BOOLEAN TryPack = TRUE; 1828 #ifdef UDF_DBG 1829 int64 check_size; 1830 #endif //UDF_DBG 1831 // I don't know what else comment can be added here. 1832 // Just belive that it works 1833 lba = UDFExtentOffsetToLba(Vcb, ExtInfo->Mapping, (Offset & ~((int64)LBS-1)), NULL, NULL, NULL, &i); 1834 if(i == (ULONG)-1) return STATUS_INVALID_PARAMETER; 1835 #ifdef UDF_DBG 1836 check_size = UDFGetExtentLength(ExtInfo->Mapping); 1837 ASSERT(!(check_size & (LBS-1))); 1838 #endif //UDF_DBG 1839 AdPrint(("Alloc->Rec ExtInfo %x, Extent %x\n", ExtInfo, Extent)); 1840 if((Extent[i].extLength >> 30) == EXTENT_RECORDED_ALLOCATED) return STATUS_SUCCESS; 1841 if((Extent[i].extLength >> 30) == EXTENT_NOT_RECORDED_NOT_ALLOCATED) return STATUS_INVALID_PARAMETER; 1842 ASSERT((((uint32)Offset) & (LBS-1)) + Length <= (Extent[i].extLength & UDF_EXTENT_LENGTH_MASK)); 1843 sLen = (( (((uint32)Offset) & (LBS-1)) + Length+LBS-1) & ~(LBS-1)) >> BSh; 1844 if((Extent[i].extLocation == lba) && (((Extent[i].extLength & UDF_EXTENT_LENGTH_MASK ) >> BSh) == sLen)) { 1845 // xxxxxx -> RRRRRR 1846 Extent[i].extLength &= UDF_EXTENT_LENGTH_MASK; 1847 // Extent[i].extLength |= (EXTENT_RECORDED_ALLOCATED << 30); // = 0; 1848 ExtInfo->Modified = TRUE; 1849 if(i && 1850 ((Extent[i-1].extLength >> 30) == EXTENT_RECORDED_ALLOCATED) && 1851 (lba == (Extent[i-1].extLocation + ((len = Extent[i-1].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh))) && 1852 ((len + (Extent[i].extLength & UDF_EXTENT_LENGTH_MASK)) <= UDF_MAX_EXTENT_LENGTH) && 1853 (i == ((UDFGetMappingLength(Extent) / sizeof(EXTENT_MAP)) - 2)) && 1854 TRUE) { 1855 // make optimization for sequentially written files 1856 Extent[i-1].extLength += Extent[i].extLength; 1857 Extent[i].extLocation = 0; 1858 Extent[i].extLength = 0; 1859 } else { 1860 UDFPackMapping(Vcb, ExtInfo); 1861 } 1862 AdPrint(("Alloc->Rec (1) new %x\n", ExtInfo->Mapping)); 1863 #ifdef UDF_DBG 1864 ASSERT(check_size == UDFGetExtentLength(ExtInfo->Mapping)); 1865 #endif 1866 AdPrint(("Alloc->Rec: ExtInfo %x, Extent %x\n", ExtInfo, ExtInfo->Mapping)); 1867 return STATUS_SUCCESS; 1868 } 1869 if(Extent[i].extLocation < lba) { 1870 if( (((Extent[i].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh) - (lba - Extent[i].extLocation)) 1871 > sLen ) { 1872 // xxxxxx -> xxRRxx 1873 NewExtent = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , UDFGetMappingLength(Extent) + sizeof(EXTENT_MAP)*2, 1874 MEM_EXTMAP_TAG); 1875 if(!NewExtent) return STATUS_INSUFFICIENT_RESOURCES; 1876 Extent[i].extLength &= UDF_EXTENT_LENGTH_MASK; 1877 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 1878 RtlCopyMemory((int8*)&(NewExtent[i+3]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 1879 NewExtent[i].extLocation = Extent[i].extLocation; 1880 NewExtent[i].extLength = (lba - Extent[i].extLocation) << BSh; 1881 NewExtent[i+1].extLength = (Length+BS-1) & ~(BS-1); 1882 NewExtent[i+1].extLocation = lba; 1883 NewExtent[i+2].extLength = Extent[i].extLength - NewExtent[i].extLength - NewExtent[i+1].extLength; 1884 NewExtent[i+2].extLocation = lba + ((Length+BS-1) >> BSh); 1885 ASSERT(!(NewExtent[i].extLength >> 30)); 1886 ASSERT(!(NewExtent[i+2].extLength >> 30)); 1887 NewExtent[i].extLength |= (EXTENT_NOT_RECORDED_ALLOCATED << 30); 1888 NewExtent[i+2].extLength |= (EXTENT_NOT_RECORDED_ALLOCATED << 30); 1889 TryPack = FALSE; 1890 AdPrint(("Alloc->Rec (2) new %x\n", NewExtent)); 1891 } else { 1892 // xxxxxx -> xxRRRR 1893 NewExtent = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , UDFGetMappingLength(Extent) + sizeof(EXTENT_MAP), 1894 MEM_EXTMAP_TAG); 1895 if(!NewExtent) return STATUS_INSUFFICIENT_RESOURCES; 1896 Extent[i].extLength &= UDF_EXTENT_LENGTH_MASK; 1897 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 1898 RtlCopyMemory((int8*)&(NewExtent[i+2]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 1899 NewExtent[i].extLocation = Extent[i].extLocation; 1900 NewExtent[i].extLength = (lba - Extent[i].extLocation) << BSh; 1901 NewExtent[i+1].extLength = Extent[i].extLength - NewExtent[i].extLength; 1902 NewExtent[i+1].extLocation = lba; 1903 ASSERT(!(NewExtent[i].extLength >> 30)); 1904 NewExtent[i].extLength |= (EXTENT_NOT_RECORDED_ALLOCATED << 30); 1905 AdPrint(("Alloc->Rec (3) new %x\n", NewExtent)); 1906 } 1907 } else { 1908 // xxxxxx -> RRRRxx 1909 NewExtent = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , UDFGetMappingLength(Extent) + sizeof(EXTENT_MAP), 1910 MEM_EXTMAP_TAG); 1911 if(!NewExtent) return STATUS_INSUFFICIENT_RESOURCES; 1912 Extent[i].extLength &= UDF_EXTENT_LENGTH_MASK; 1913 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 1914 RtlCopyMemory((int8*)&(NewExtent[i+2]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 1915 NewExtent[i].extLocation = Extent[i].extLocation; 1916 NewExtent[i].extLength = (Length+BS-1) & ~(BS-1); 1917 NewExtent[i+1].extLength = Extent[i].extLength - NewExtent[i].extLength; 1918 NewExtent[i+1].extLocation = Extent[i].extLocation + (NewExtent[i].extLength >> BSh); 1919 ASSERT(!(NewExtent[i+1].extLength >> 30)); 1920 NewExtent[i+1].extLength |= (EXTENT_NOT_RECORDED_ALLOCATED << 30); 1921 AdPrint(("Alloc->Rec (4) new %x\n", NewExtent)); 1922 } 1923 1924 //ASSERT(check_size == UDFGetExtentLength(Extent)); 1925 //ASSERT(!(check_size & (LBS-1))); 1926 1927 AdPrint(("Free Extent %x (new %x)\n", Extent, NewExtent)); 1928 MyFreePool__(Extent); 1929 ExtInfo->Modified = TRUE; 1930 ExtInfo->Mapping = NewExtent; 1931 if(TryPack) 1932 UDFPackMapping(Vcb, ExtInfo); 1933 #ifdef UDF_DBG 1934 ASSERT(check_size == UDFGetExtentLength(ExtInfo->Mapping)); 1935 ASSERT(!(check_size & (LBS-1))); 1936 #endif 1937 1938 AdPrint(("Alloc->Rec: ExtInfo %x, Extent %x\n", ExtInfo, ExtInfo->Mapping)); 1939 1940 return STATUS_SUCCESS; 1941 } // end UDFMarkAllocatedAsRecorded() 1942 1943 /* 1944 This routine rebuilds mapping on write attempts to Not-Alloc-Not-Rec area. 1945 Here we assume that required area lays in a single frag. 1946 */ 1947 OSSTATUS 1948 UDFMarkNotAllocatedAsAllocated( 1949 IN PVCB Vcb, 1950 IN int64 Offset, 1951 IN uint32 Length, 1952 IN PEXTENT_INFO ExtInfo // Extent array 1953 ) 1954 { 1955 uint32 i, len, /*lba,*/ d, l, BOffs, j; 1956 PEXTENT_MAP Extent = ExtInfo->Mapping; // Extent array 1957 PEXTENT_MAP NewExtent; 1958 // uint32 BS = Vcb->BlockSize; 1959 uint32 BSh = Vcb->BlockSizeBits; 1960 OSSTATUS status; 1961 EXTENT_INFO TmpExtInf; 1962 SIZE_T aLen, sLen; 1963 SIZE_T LBS = Vcb->LBlockSize; 1964 // I don't know what else comment can be added here. 1965 // Just belive that it works 1966 /*lba = */ 1967 #ifndef ALLOW_SPARSE 1968 BrutePoint(); 1969 #endif 1970 AdPrint(("Not->Alloc ExtInfo %x, Extent %x\n", ExtInfo, Extent)); 1971 UDFExtentOffsetToLba(Vcb, ExtInfo->Mapping, Offset, NULL, NULL, NULL, &i); 1972 if(i == (ULONG)-1) return STATUS_INVALID_PARAMETER; 1973 if((Extent[i].extLength >> 30) != EXTENT_NOT_RECORDED_NOT_ALLOCATED) return STATUS_SUCCESS; 1974 1975 uint32 PartNum = UDFGetPartNumByPhysLba(Vcb, Extent[0].extLocation); 1976 BOffs = (uint32)(Offset >> BSh); 1977 // length of existing Not-Alloc-Not-Rec frag 1978 sLen = (( (((uint32)Offset) & (LBS-1)) + Length+LBS-1) & ~(LBS-1)) >> BSh; 1979 // required allocation length increment (in bytes) 1980 aLen = (uint32)( ((Offset+Length+LBS-1) & ~(LBS-1)) - (Offset & ~(LBS-1))); 1981 1982 // try to extend previous frag or allocate space _after_ it to 1983 // avoid backward seeks, if previous frag is not Not-Rec-Not-Alloc 1984 if(i && ((Extent[i-1].extLength >> 30) != EXTENT_NOT_RECORDED_NOT_ALLOCATED) ) { 1985 status = UDFAllocFreeExtent(Vcb, aLen, 1986 Extent[i-1].extLocation + ((Extent[i-1].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh), 1987 min(UDFPartEnd(Vcb, PartNum), Extent[i-1].extLocation + ((Extent[i-1].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh) + sLen ), 1988 &TmpExtInf, ExtInfo->Flags /*& EXTENT_FLAG_ALLOC_MASK*/); 1989 if(status == STATUS_DISK_FULL) 1990 // if there are not enough free blocks after that frag... 1991 goto try_alloc_anywhere; 1992 } else { 1993 try_alloc_anywhere: 1994 // ... try to alloc required disk space anywhere 1995 status = UDFAllocFreeExtent(Vcb, aLen, 1996 UDFPartStart(Vcb, PartNum), 1997 UDFPartEnd(Vcb, PartNum), 1998 &TmpExtInf, ExtInfo->Flags /*& EXTENT_FLAG_ALLOC_MASK*/); 1999 } 2000 // check for successfull allocation 2001 if(!OS_SUCCESS(status)) { 2002 AdPrint(("Not->Alloc no free\n")); 2003 return status; 2004 } 2005 // get number of frags in allocated block 2006 d = (UDFGetMappingLength(TmpExtInf.Mapping) / sizeof(EXTENT_MAP)) - 1; 2007 // calculate number of existing blocks before the frag to be changed 2008 l=0; 2009 for(j=0; j<i; j++) { 2010 l += (uint32)((Extent[j].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh); 2011 } 2012 // and now just update mapping... 2013 if( (l == BOffs) && (((Extent[j].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh) == sLen) ) { 2014 // xxxxxx -> RRRRRR 2015 // (d-1) - since we have to raplace last frag of Extent with 1 or more frags of TmpExtInf.Mapping 2016 NewExtent = (PEXTENT_AD)MyAllocatePool__(NonPagedPool, UDFGetMappingLength(Extent) + (d-1)*sizeof(EXTENT_MAP) ); 2017 if(!NewExtent) { 2018 MyFreePool__(TmpExtInf.Mapping); 2019 return STATUS_INSUFFICIENT_RESOURCES; 2020 } 2021 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 2022 RtlCopyMemory((int8*)&(NewExtent[i]), (int8*)(TmpExtInf.Mapping), d*sizeof(EXTENT_MAP) ); 2023 RtlCopyMemory((int8*)&(NewExtent[i+d]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 2024 AdPrint(("Not->Alloc (1) new %x\n", NewExtent)); 2025 } else 2026 if(l < BOffs) { 2027 // .ExtLength, BOffs & l are already aligned... 2028 if( (((Extent[i].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh) - (BOffs-l)) > sLen ) { 2029 // xxxxxx -> xxRRxx 2030 NewExtent = (PEXTENT_AD)MyAllocatePool__(NonPagedPool, UDFGetMappingLength(Extent) + (d+1)*sizeof(EXTENT_MAP) ); 2031 if(!NewExtent) { 2032 MyFreePool__(TmpExtInf.Mapping); 2033 return STATUS_INSUFFICIENT_RESOURCES; 2034 } 2035 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 2036 RtlCopyMemory((int8*)&(NewExtent[i+1]), (int8*)(TmpExtInf.Mapping), d*sizeof(EXTENT_MAP) ); 2037 RtlCopyMemory((int8*)&(NewExtent[i+d+2]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 2038 NewExtent[i].extLocation = 0; 2039 NewExtent[i].extLength = (BOffs - l) << BSh; 2040 NewExtent[i+d+1].extLength = Extent[i].extLength - NewExtent[i].extLength - aLen; 2041 NewExtent[i+d+1].extLocation = 0; 2042 NewExtent[i].extLength |= (EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30); 2043 NewExtent[i+d+1].extLength |= (EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30); 2044 AdPrint(("Not->Alloc (2) new %x\n", NewExtent)); 2045 } else { 2046 // xxxxxx -> xxRRRR 2047 NewExtent = (PEXTENT_AD)MyAllocatePool__(NonPagedPool, UDFGetMappingLength(Extent) + d*sizeof(EXTENT_MAP) ); 2048 if(!NewExtent) { 2049 MyFreePool__(TmpExtInf.Mapping); 2050 return STATUS_INSUFFICIENT_RESOURCES; 2051 } 2052 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 2053 RtlCopyMemory((int8*)&(NewExtent[i+1]), (int8*)(TmpExtInf.Mapping), d*sizeof(EXTENT_MAP) ); 2054 RtlCopyMemory((int8*)&(NewExtent[i+d+1]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 2055 NewExtent[i].extLocation = 0; 2056 NewExtent[i].extLength = (BOffs - l) << BSh; 2057 NewExtent[i].extLength |= (EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30); 2058 AdPrint(("Not->Alloc (3) new %x\n", NewExtent)); 2059 } 2060 } else { 2061 // xxxxxx -> RRRRxx 2062 NewExtent = (PEXTENT_AD)MyAllocatePool__(NonPagedPool, UDFGetMappingLength(Extent) + d*sizeof(EXTENT_MAP) ); 2063 if(!NewExtent) { 2064 MyFreePool__(TmpExtInf.Mapping); 2065 return STATUS_INSUFFICIENT_RESOURCES; 2066 } 2067 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 2068 RtlCopyMemory((int8*)&(NewExtent[i]), (int8*)(TmpExtInf.Mapping), d*sizeof(EXTENT_MAP) ); 2069 RtlCopyMemory((int8*)&(NewExtent[i+d+1]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 2070 NewExtent[i+d].extLength = (Extent[i].extLength & UDF_EXTENT_LENGTH_MASK) - aLen; 2071 NewExtent[i+d].extLocation = 0; 2072 NewExtent[i+d].extLength |= (EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30); 2073 AdPrint(("Not->Alloc (4) new %x\n", NewExtent)); 2074 } 2075 2076 AdPrint(("Free Extent %x, TmpExtInf.Mapping, (new %x)\n", Extent, TmpExtInf.Mapping, NewExtent)); 2077 MyFreePool__(Extent); 2078 MyFreePool__(TmpExtInf.Mapping); 2079 ExtInfo->Modified = TRUE; 2080 ExtInfo->Mapping = NewExtent; 2081 2082 AdPrint(("Not->Alloc: ExtInfo %x, Extent %x\n", ExtInfo, ExtInfo->Mapping)); 2083 2084 return STATUS_SUCCESS; 2085 } // end UDFMarkNotAllocatedAsAllocated() 2086 2087 //#if 0 2088 /* 2089 This routine rebuilds mapping on write zero attempts to 2090 Alloc-Not-Rec area. 2091 Here we assume that required area lays in a single frag. 2092 */ 2093 OSSTATUS 2094 UDFMarkAllocatedAsNotXXX( 2095 IN PVCB Vcb, 2096 IN int64 Offset, 2097 IN uint32 Length, 2098 IN PEXTENT_INFO ExtInfo, // Extent array 2099 IN BOOLEAN Deallocate 2100 ) 2101 { 2102 uint32 i, len, /*lba, d,*/ l, BOffs, j; 2103 PEXTENT_MAP Extent = ExtInfo->Mapping; // Extent array 2104 PEXTENT_MAP NewExtent; 2105 // EXTENT_MAP TmpExtent; 2106 // uint32 BS = Vcb->BlockSize; 2107 uint32 BSh = Vcb->BlockSizeBits; 2108 // OSSTATUS status; 2109 EXTENT_INFO TmpExtInf; 2110 uint32 aLen, sLen; 2111 uint32 flags; 2112 uint32 target_flags = Deallocate ? 2113 EXTENT_NOT_RECORDED_NOT_ALLOCATED : 2114 EXTENT_NOT_RECORDED_ALLOCATED; 2115 SIZE_T LBS = Vcb->LBlockSize; 2116 EXTENT_MAP DeadMapping[2]; 2117 // I don't know what else comment can be added here. 2118 // Just belive that it works 2119 /*lba = */ 2120 #ifndef ALLOW_SPARSE 2121 if(Deallocate) { 2122 BrutePoint(); 2123 } 2124 #endif 2125 2126 AdPrint(("Alloc->Not ExtInfo %x, Extent %x\n", ExtInfo, Extent)); 2127 2128 DeadMapping[0].extLocation = 2129 UDFExtentOffsetToLba(Vcb, ExtInfo->Mapping, Offset, NULL, NULL, NULL, &i); 2130 if(i == (ULONG)-1) { 2131 BrutePoint(); 2132 return STATUS_INVALID_PARAMETER; 2133 } 2134 DeadMapping[0].extLength = Extent[i].extLength; 2135 DeadMapping[1].extLocation = 2136 DeadMapping[1].extLength = 0; 2137 TmpExtInf.Mapping = (PEXTENT_MAP)&DeadMapping; 2138 TmpExtInf.Offset = 0; 2139 TmpExtInf.Length = Extent[i].extLength & UDF_EXTENT_LENGTH_MASK; 2140 2141 flags = Extent[i].extLength >> 30; 2142 if(flags == target_flags) return STATUS_SUCCESS; 2143 2144 // uint32 PartNum = UDFGetPartNumByPhysLba(Vcb, Extent[0].extLocation); 2145 BOffs = (uint32)(Offset >> BSh); 2146 // length of existing Alloc-(Not-)Rec frag (in sectors) 2147 sLen = (( (((uint32)Offset) & (LBS-1)) + Length+LBS-1) & ~(LBS-1)) >> BSh; 2148 // required deallocation length increment (in bytes) 2149 aLen = (uint32)( ((Offset+Length+LBS-1) & ~(LBS-1)) - (Offset & ~(LBS-1)) ); 2150 2151 l=0; 2152 for(j=0; j<i; j++) { 2153 l += (uint32)((Extent[j].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh); 2154 } 2155 flags <<= 30; 2156 if( (l == BOffs) && (((Extent[j].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh) == sLen) ) { 2157 // xxxxxx -> RRRRRR 2158 Extent[i].extLocation = 0; 2159 Extent[i].extLength = (Extent[i].extLength & UDF_EXTENT_LENGTH_MASK) | flags; 2160 NewExtent = Extent; 2161 Extent = NULL; 2162 AdPrint(("Alloc->Not (1) NewExtent = Extent = %x\n", NewExtent)); 2163 } else 2164 if(l < BOffs) { 2165 // .ExtLength, BOffs & l are already aligned... 2166 if( (((Extent[i].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh) - (BOffs-l)) > sLen ) { 2167 // xxxxxx -> xxRRxx 2168 NewExtent = (PEXTENT_MAP)MyAllocatePool__(NonPagedPool, UDFGetMappingLength(Extent) + 2*sizeof(EXTENT_MAP) ); 2169 if(!NewExtent) { 2170 return STATUS_INSUFFICIENT_RESOURCES; 2171 } 2172 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 2173 RtlCopyMemory((int8*)&(NewExtent[i+3]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 2174 NewExtent[i].extLength = (BOffs - l) << BSh; 2175 NewExtent[i].extLength |= flags; 2176 NewExtent[i+1].extLocation = 0; 2177 NewExtent[i+1].extLength = aLen | (target_flags << 30); 2178 NewExtent[i+2].extLength = (Extent[i].extLength & UDF_EXTENT_LENGTH_MASK) - 2179 (NewExtent[i].extLength & UDF_EXTENT_LENGTH_MASK) - aLen ; 2180 NewExtent[i+2].extLocation = Extent[i].extLocation + 2181 (NewExtent[i+2].extLength >> BSh); 2182 NewExtent[i+2].extLength |= flags; 2183 AdPrint(("Alloc->Not (2) new %x\n", NewExtent)); 2184 } else { 2185 // xxxxxx -> xxRRRR 2186 NewExtent = (PEXTENT_MAP)MyAllocatePool__(NonPagedPool, UDFGetMappingLength(Extent) + sizeof(EXTENT_MAP) ); 2187 if(!NewExtent) { 2188 return STATUS_INSUFFICIENT_RESOURCES; 2189 } 2190 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 2191 RtlCopyMemory((int8*)&(NewExtent[i+2]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 2192 NewExtent[i].extLength = ((BOffs - l) << BSh) | flags; 2193 NewExtent[i+1].extLocation = 0; 2194 NewExtent[i+1].extLength = aLen | (target_flags << 30); 2195 AdPrint(("Alloc->Not (3) new %x\n", NewExtent)); 2196 } 2197 } else { 2198 // xxxxxx -> RRRRxx 2199 NewExtent = (PEXTENT_MAP)MyAllocatePool__(NonPagedPool, UDFGetMappingLength(Extent) + sizeof(EXTENT_MAP) ); 2200 if(!NewExtent) { 2201 return STATUS_INSUFFICIENT_RESOURCES; 2202 } 2203 RtlCopyMemory((int8*)NewExtent, (int8*)Extent, i*sizeof(EXTENT_MAP)); 2204 RtlCopyMemory((int8*)&(NewExtent[i+2]), (int8*)&(Extent[i+1]), len = UDFGetMappingLength(&(Extent[i+1])) ); 2205 NewExtent[i+1].extLength = (Extent[i].extLength & UDF_EXTENT_LENGTH_MASK) - aLen; 2206 NewExtent[i+1].extLength |= flags; 2207 NewExtent[i].extLocation = 0; 2208 NewExtent[i].extLength = aLen | (target_flags << 30); 2209 AdPrint(("Alloc->Not (4) new %x\n", NewExtent)); 2210 } 2211 2212 if(Deallocate) 2213 UDFMarkSpaceAsXXX(Vcb, (-1), TmpExtInf.Mapping, AS_DISCARDED); // mark as free 2214 2215 ExtInfo->Modified = TRUE; 2216 ExtInfo->Mapping = NewExtent; 2217 2218 AdPrint(("Alloc->Not: ExtInfo %x, Extent %x\n", ExtInfo, ExtInfo->Mapping)); 2219 2220 if(Extent) { 2221 AdPrint(("Alloc->Not kill %x\n", Extent)); 2222 MyFreePool__(Extent); 2223 } else { 2224 AdPrint(("Alloc->Not keep %x\n", Extent)); 2225 } 2226 2227 return STATUS_SUCCESS; 2228 } // end UDFMarkAllocatedAsNotXXX() 2229 //#endif //0 2230 2231 /* 2232 This routine resizes extent & updates associated mapping 2233 */ 2234 OSSTATUS 2235 UDFResizeExtent( 2236 IN PVCB Vcb, 2237 IN uint32 PartNum, 2238 IN int64 Length, // Required Length 2239 IN BOOLEAN AlwaysInIcb, // must be TRUE for AllocDescs 2240 OUT PEXTENT_INFO ExtInfo 2241 ) 2242 { 2243 uint32 i, flags, lba; 2244 SIZE_T lim; 2245 int64 l; 2246 OSSTATUS status; 2247 EXTENT_INFO TmpExtInf; 2248 EXTENT_MAP TmpMapping[2]; 2249 uint32 s, pe, BSh, PS; 2250 SIZE_T req_s; 2251 SIZE_T LBS = Vcb->LBlockSize; 2252 BSh = Vcb->BlockSizeBits; 2253 PS = Vcb->WriteBlockSize >> Vcb->BlockSizeBits; 2254 uint32 MaxGrow = (UDF_MAX_EXTENT_LENGTH & ~(LBS-1)); 2255 BOOLEAN Sequential = FALSE; 2256 2257 ASSERT(PartNum < 3); 2258 2259 ExtPrint(("Resize ExtInfo %x, %I64x -> %I64x\n", ExtInfo, ExtInfo->Length, Length)); 2260 2261 if(ExtInfo->Flags & EXTENT_FLAG_CUT_PREALLOCATED) { 2262 AdPrint((" cut preallocated\n")); 2263 } else 2264 if(ExtInfo->Length == Length) { 2265 return STATUS_SUCCESS; 2266 } 2267 if((ExtInfo->Flags & EXTENT_FLAG_ALLOC_MASK) == EXTENT_FLAG_ALLOC_SEQUENTIAL) { 2268 MaxGrow &= ~(Vcb->WriteBlockSize-1); 2269 Sequential = TRUE; 2270 } 2271 2272 UDFCheckSpaceAllocation(Vcb, 0, ExtInfo->Mapping, AS_USED); // check if used 2273 if(ExtInfo->Offset) { 2274 if(ExtInfo->Offset + Length <= LBS) { 2275 ExtPrint(("Resize IN-ICB\n")); 2276 ExtInfo->Length = Length; 2277 return STATUS_SUCCESS; 2278 } 2279 if(!AlwaysInIcb) // simulate unused 1st sector in extent 2280 ExtInfo->Offset = LBS; // it'll be truncated later 2281 Length += ExtInfo->Offset; // convert to real offset in extent 2282 } 2283 lba = UDFExtentOffsetToLba(Vcb, ExtInfo->Mapping, Length, NULL, NULL, &flags, &i); 2284 if(ExtInfo->Length < Length) { 2285 // increase extent 2286 if(OS_SUCCESS(UDFGetCachedAllocation(Vcb, ExtInfo->Mapping[0].extLocation, 2287 &TmpExtInf, NULL, UDF_PREALLOC_CLASS_DIR))) { 2288 AdPrint(("Resize found cached(1)\n")); 2289 ExtInfo->Mapping = UDFMergeMappings(ExtInfo->Mapping, TmpExtInf.Mapping); 2290 MyFreePool__(TmpExtInf.Mapping); 2291 } 2292 if((l = UDFGetExtentLength(ExtInfo->Mapping)) >= Length) { 2293 // we have enough space inside extent 2294 ExtInfo->Length = Length; 2295 AdPrint(("Resize do nothing (1)\n")); 2296 } else /*if(lba == LBA_OUT_OF_EXTENT)*/ { 2297 2298 Length -= ExtInfo->Offset; 2299 if(/*Length && l &&*/ (l % MaxGrow) && 2300 (Length-1)/MaxGrow != (l-1)/MaxGrow) { 2301 AdPrint(("Crossing MAX_FRAG boundary...\n")); 2302 int64 l2 = ((l-1)/MaxGrow + 1)*MaxGrow; 2303 status = UDFResizeExtent(Vcb, PartNum, l2, AlwaysInIcb, ExtInfo); 2304 if(!OS_SUCCESS(status)) { 2305 UDFPrint(("Sub-call to UDFResizeExtent() failed (%x)\n", status)); 2306 return status; 2307 } 2308 l = ExtInfo->Length; 2309 ASSERT(l == l2); 2310 } 2311 while((Length - l) > MaxGrow) { 2312 status = UDFResizeExtent(Vcb, PartNum, l+MaxGrow, AlwaysInIcb, ExtInfo); 2313 if(!OS_SUCCESS(status)) { 2314 UDFPrint(("Sub-call (2) to UDFResizeExtent() failed (%x)\n", status)); 2315 return status; 2316 } 2317 l = ExtInfo->Length; 2318 } 2319 Length += ExtInfo->Offset; 2320 // at first, try to resize existing frag 2321 #ifndef UDF_ALLOW_FRAG_AD 2322 i = UDFGetMappingLength(ExtInfo->Mapping); 2323 if(i > (LBS-sizeof(EXTENDED_FILE_ENTRY))) { 2324 // this is very important check since we will not 2325 // be able to _record_ too long AllocDesc because of 2326 // some DEMO limitations in UDFBuildXXXAllocDescs() 2327 AdPrint((" DISK_FULL\n")); 2328 return STATUS_DISK_FULL; 2329 } 2330 i /= sizeof(EXTENT_MAP); 2331 #else //UDF_ALLOW_FRAG_AD 2332 i = UDFGetMappingLength(ExtInfo->Mapping) / sizeof(EXTENT_MAP); 2333 #endif //UDF_ALLOW_FRAG_AD 2334 #ifdef ALLOW_SPARSE 2335 if(!AlwaysInIcb && !(ExtInfo->Offset) && 2336 (Length - l >= (Vcb->SparseThreshold << BSh))) { 2337 // last frag will be Not-Alloc-Not-Rec... 2338 AdPrint(("Resize sparse (2)\n")); 2339 RtlZeroMemory(&TmpExtInf, sizeof(EXTENT_INFO)); 2340 TmpExtInf.Mapping = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , sizeof(EXTENT_MAP)*2, 2341 MEM_EXTMAP_TAG); 2342 if(!TmpExtInf.Mapping) return STATUS_INSUFFICIENT_RESOURCES; 2343 TmpExtInf.Mapping[0].extLength = (((uint32)(Length - l) + LBS-1) & ~(LBS-1)) | (EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30); 2344 TmpExtInf.Mapping[0].extLocation =// 0; 2345 TmpExtInf.Mapping[1].extLength = 2346 TmpExtInf.Mapping[1].extLocation = 0; 2347 l = Length; 2348 ExtInfo->Mapping = UDFMergeMappings(ExtInfo->Mapping, TmpExtInf.Mapping); 2349 MyFreePool__(TmpExtInf.Mapping); 2350 } else 2351 #endif //ALLOW_SPARSE 2352 // allocate some sectors 2353 if(i>1 && !(ExtInfo->Offset)) { 2354 i-=2; 2355 // check if Not-Alloc-Not-Rec at the end of mapping 2356 if((uint32)Length - (uint32)l + (ExtInfo->Mapping[i].extLength & UDF_EXTENT_LENGTH_MASK) > MaxGrow) { 2357 // do nothing, but jump directly to allocator 2358 } else 2359 if((ExtInfo->Mapping[i].extLength >> 30) == EXTENT_NOT_RECORDED_NOT_ALLOCATED) { 2360 AdPrint(("Resize grow sparse (3)\n")); 2361 ExtInfo->Mapping[i].extLength += 2362 (((uint32)Length-(uint32)l+LBS-1) & ~(LBS-1)) ; 2363 l = Length; 2364 // check if Alloc-Not-Rec at the end of mapping 2365 } else if((ExtInfo->Mapping[i].extLength >> 30) == EXTENT_NOT_RECORDED_ALLOCATED) { 2366 AdPrint(("Resize grow Not-Rec (3)\n")); 2367 // current length of last frag 2368 s = ((ExtInfo->Mapping[i].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh); 2369 // prefered location of the next frag 2370 lba = ExtInfo->Mapping[i].extLocation + s; 2371 pe=UDFPartEnd(Vcb,PartNum); 2372 // maximum frag length 2373 if(Sequential) { 2374 lim = (((uint32)UDF_MAX_EXTENT_LENGTH) >> BSh) & ~(PS-1); 2375 } else { 2376 lim = (((uint32)UDF_MAX_EXTENT_LENGTH) >> BSh) & ~(LBS-1); 2377 } 2378 // required last extent length 2379 req_s = s + (uint32)( (((Length + LBS - 1) & ~(LBS-1)) - 2380 ((l + LBS - 1) & ~(LBS-1)) ) >> BSh); 2381 if(lim > req_s) { 2382 lim = req_s; 2383 } 2384 UDFAcquireResourceExclusive(&(Vcb->BitMapResource1),TRUE); 2385 /* if((ExtInfo->Flags & EXTENT_FLAG_SEQUENTIAL) && 2386 ((Length & ~(PS-1)) > (l & ~(PS-1))) && 2387 TRUE) { 2388 status = UDFResizeExtent(Vcb, PartNum, l+MaxGrow, AlwaysInIcb, ExtInfo); 2389 }*/ 2390 // how many sectors we should add 2391 req_s = lim - s; 2392 ASSERT(req_s); 2393 if((lba < pe) && UDFGetFreeBit(Vcb->FSBM_Bitmap, lba)) { 2394 s += UDFGetBitmapLen((uint32*)(Vcb->FSBM_Bitmap), lba, min(pe, lba+req_s-1)); 2395 } 2396 /* for(s1=lba; (s<lim) && (s1<pe) && UDFGetFreeBit(Vcb->FSBM_Bitmap, s1); s1++) { 2397 s++; 2398 }*/ 2399 if(s==lim) { 2400 // we can just increase the last frag 2401 AdPrint(("Resize grow last Not-Rec (4)\n")); 2402 ExtInfo->Mapping[i].extLength = (lim << BSh) | (EXTENT_NOT_RECORDED_ALLOCATED << 30); 2403 l = Length; 2404 UDFMarkSpaceAsXXXNoProtect(Vcb, 0, &(ExtInfo->Mapping[i]), AS_USED); // mark as used 2405 } else { 2406 // we get here if simple increasing of last frag failed 2407 // it worth truncating last frag and try to allocate 2408 // all required data as a single frag 2409 2410 /* if(Sequential && s>=PS) { 2411 s &= ~(PS-1); 2412 AdPrint(("Resize grow last Not-Rec (4/2)\n")); 2413 ExtInfo->Mapping[i].extLength = (s << BSh) | (EXTENT_NOT_RECORDED_ALLOCATED << 30); 2414 l += (s << BSh); 2415 UDFMarkSpaceAsXXXNoProtect(Vcb, 0, &(ExtInfo->Mapping[i]), AS_USED); // mark as used 2416 }*/ 2417 AdPrint(("Resize reloc last Not-Rec (5)\n")); 2418 TmpExtInf.Mapping = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , (i+1)*sizeof(EXTENT_MAP), 2419 MEM_EXTMAP_TAG); 2420 if(!TmpExtInf.Mapping) { 2421 UDFPrint(("UDFResizeExtent: !TmpExtInf.Mapping\n")); 2422 UDFReleaseResource(&(Vcb->BitMapResource1)); 2423 return STATUS_INSUFFICIENT_RESOURCES; 2424 } 2425 RtlCopyMemory(TmpExtInf.Mapping, ExtInfo->Mapping, i*sizeof(EXTENT_MAP)); 2426 TmpExtInf.Mapping[i].extLength = 2427 TmpExtInf.Mapping[i].extLocation = 0; 2428 TmpExtInf.Offset = ExtInfo->Offset; 2429 l -= (ExtInfo->Mapping[i].extLength & UDF_EXTENT_LENGTH_MASK); 2430 TmpExtInf.Length = l; 2431 ASSERT(i || !ExtInfo->Offset); 2432 UDFMarkSpaceAsXXXNoProtect(Vcb, 0, &(ExtInfo->Mapping[i]), AS_DISCARDED); // mark as free 2433 MyFreePool__(ExtInfo->Mapping); 2434 (*ExtInfo) = TmpExtInf; 2435 } 2436 UDFCheckSpaceAllocation(Vcb, 0, ExtInfo->Mapping, AS_USED); // check if used 2437 UDFReleaseResource(&(Vcb->BitMapResource1)); 2438 // check if Alloc-Rec 2439 } else { 2440 // current length of last frag 2441 s = ((ExtInfo->Mapping[i].extLength & UDF_EXTENT_LENGTH_MASK) >> BSh); 2442 // prefered location of the next frag 2443 lba = ExtInfo->Mapping[i].extLocation + s; 2444 pe=UDFPartEnd(Vcb,PartNum); 2445 // maximum frag length 2446 if(Sequential) { 2447 lim = (((uint32)UDF_MAX_EXTENT_LENGTH) >> BSh) & ~(PS-1); 2448 } else { 2449 lim = (((uint32)UDF_MAX_EXTENT_LENGTH) >> BSh) & ~(LBS-1); 2450 } 2451 // required last extent length 2452 req_s = s + (uint32)( (((Length + LBS - 1) & ~(LBS-1)) - 2453 ((l + LBS - 1) & ~(LBS-1)) ) >> BSh); 2454 if(lim > req_s) { 2455 lim = req_s; 2456 } 2457 // s=0; 2458 // how many sectors we should add 2459 req_s = lim - s; 2460 if(req_s) { 2461 uint32 d=0; 2462 2463 UDFAcquireResourceExclusive(&(Vcb->BitMapResource1),TRUE); 2464 //ASSERT(req_s); 2465 if((lba < pe) && UDFGetFreeBit(Vcb->FSBM_Bitmap, lba)) { 2466 s += (d = UDFGetBitmapLen((uint32*)(Vcb->FSBM_Bitmap), lba, min(pe, lba+req_s-1))); 2467 } 2468 /* for(s1=lba; (s<lim) && (s1<pe) && UDFGetFreeBit(Vcb->FSBM_Bitmap, s1); s1++) { 2469 s++; 2470 }*/ 2471 2472 if(s==lim) { 2473 AdPrint(("Resize grow last Rec (6)\n")); 2474 // we can just increase last frag 2475 TmpMapping[0].extLength = req_s << BSh; 2476 TmpMapping[0].extLocation = lba; 2477 TmpMapping[1].extLength = 2478 TmpMapping[1].extLocation = 0; 2479 UDFMarkSpaceAsXXXNoProtect(Vcb, 0, &TmpMapping[0], AS_USED); // mark as used 2480 l += (s << BSh) - (ExtInfo->Mapping[i].extLength & UDF_EXTENT_LENGTH_MASK); 2481 ExtInfo->Mapping[i].extLength = (ExtInfo->Mapping[i].extLength & UDF_EXTENT_FLAG_MASK) | (s << BSh); 2482 } else if(d) { 2483 AdPrint(("Resize part-grow last Rec (6)\n")); 2484 // increase last frag, then alloc rest 2485 TmpMapping[0].extLength = d << BSh; 2486 TmpMapping[0].extLocation = lba; 2487 TmpMapping[1].extLength = 2488 TmpMapping[1].extLocation = 0; 2489 UDFMarkSpaceAsXXXNoProtect(Vcb, 0, &TmpMapping[0], AS_USED); // mark as used 2490 l += (s << BSh) - (ExtInfo->Mapping[i].extLength & UDF_EXTENT_LENGTH_MASK); 2491 ExtInfo->Mapping[i].extLength = (ExtInfo->Mapping[i].extLength & UDF_EXTENT_FLAG_MASK) | (s << BSh); 2492 } else { 2493 AdPrint(("Can't grow last Rec (6)\n")); 2494 } 2495 UDFReleaseResource(&(Vcb->BitMapResource1)); 2496 } else { 2497 AdPrint(("Max frag length reached (6)\n")); 2498 } 2499 } 2500 } 2501 if(l < Length) { 2502 // we get here if simple increasing of the last frag failed 2503 AdPrint(("Resize add new frag (7)\n")); 2504 if(l < LBS && Length >= LBS && 2505 (ExtInfo->Flags & EXTENT_FLAG_ALLOC_MASK) == EXTENT_FLAG_ALLOC_SEQUENTIAL) { 2506 AdPrint(("Resize tune for SEQUENTIAL i/o\n")); 2507 } 2508 status = UDFAllocFreeExtent(Vcb, Length - l, 2509 UDFPartStart(Vcb, PartNum), 2510 UDFPartEnd(Vcb, PartNum), 2511 &TmpExtInf, 2512 ExtInfo->Flags /*& EXTENT_FLAG_ALLOC_MASK*/); 2513 if(!OS_SUCCESS(status)) { 2514 UDFPrint(("UDFResizeExtent: UDFAllocFreeExtent() failed (%x)\n", status)); 2515 return status; 2516 } 2517 ExtInfo->Mapping = UDFMergeMappings(ExtInfo->Mapping, TmpExtInf.Mapping); 2518 MyFreePool__(TmpExtInf.Mapping); 2519 } 2520 UDFPackMapping(Vcb, ExtInfo); 2521 } 2522 } else 2523 if(Length) { 2524 // decrease extent 2525 AdPrint(("Resize cut (8)\n")); 2526 lba = UDFExtentOffsetToLba(Vcb, ExtInfo->Mapping, Length-1, NULL, &lim, &flags, &i); 2527 i++; 2528 ASSERT(lba != LBA_OUT_OF_EXTENT); 2529 ASSERT(lba != LBA_NOT_ALLOCATED); 2530 ASSERT(i); 2531 if(ExtInfo->Mapping[i].extLength) { 2532 UDFCheckSpaceAllocation(Vcb, 0, &(ExtInfo->Mapping[i]), AS_USED); // check if used 2533 if(!ExtInfo->Offset && (ExtInfo->Flags & EXTENT_FLAG_PREALLOCATED)) { 2534 2535 AdPrint(("Resize try save cutted (8)\n")); 2536 RtlZeroMemory(&TmpExtInf, sizeof(EXTENT_INFO)); 2537 s = UDFGetMappingLength(&(ExtInfo->Mapping[i])); 2538 2539 TmpExtInf.Mapping = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , s, MEM_EXTMAP_TAG); 2540 if(TmpExtInf.Mapping) { 2541 RtlCopyMemory(TmpExtInf.Mapping, &(ExtInfo->Mapping[i]), s); 2542 AdPrint(("Resize save cutted (8)\n")); 2543 if(OS_SUCCESS(UDFStoreCachedAllocation(Vcb, ExtInfo->Mapping[0].extLocation, 2544 &TmpExtInf, 0, UDF_PREALLOC_CLASS_DIR))) { 2545 ExtInfo->Mapping[i].extLength = 0; 2546 ExtInfo->Mapping[i].extLocation = 0; 2547 goto tail_cached; 2548 } 2549 } 2550 } 2551 UDFMarkSpaceAsXXX(Vcb, 0, &(ExtInfo->Mapping[i]), AS_DISCARDED); // mark as free 2552 tail_cached:; 2553 } 2554 if((lim-1 >= LBS) && 2555 (flags != EXTENT_NOT_RECORDED_NOT_ALLOCATED)) { 2556 AdPrint(("i=%x, lba=%x, len=%x\n",i,lba,lim)); 2557 ASSERT(lim); 2558 // BrutePoint(); 2559 EXTENT_MAP ClrMap[2]; 2560 ClrMap[0].extLength = lim & ~(LBS-1); 2561 s = (ExtInfo->Mapping[i-1].extLength - ClrMap[0].extLength) & UDF_EXTENT_LENGTH_MASK; 2562 ClrMap[0].extLocation = ExtInfo->Mapping[i-1].extLocation + 2563 (s >> BSh); 2564 ClrMap[1].extLength = 2565 ClrMap[1].extLocation = 0; 2566 ASSERT((ExtInfo->Mapping[i].extLocation < ClrMap[0].extLocation) || 2567 (ExtInfo->Mapping[i].extLocation >= (ClrMap[0].extLocation + (ClrMap[0].extLength >> BSh)))); 2568 UDFCheckSpaceAllocation(Vcb, 0, (PEXTENT_MAP)(&ClrMap), AS_USED); // check if used 2569 UDFMarkSpaceAsXXX(Vcb, 0, (PEXTENT_MAP)(&ClrMap), AS_DISCARDED); // mark as free 2570 ExtInfo->Mapping[i-1].extLength = s | (flags << 30); 2571 } 2572 2573 s = UDFGetMappingLength(ExtInfo->Mapping); 2574 if(!MyReallocPool__((int8*)(ExtInfo->Mapping), s, (int8**)&(ExtInfo->Mapping), (i+1)*sizeof(EXTENT_MAP))) { 2575 // This must never happen on truncate !!! 2576 AdPrint(("ResizeExtent: MyReallocPool__(8) failed\n")); 2577 } 2578 ExtInfo->Mapping[i].extLength = 2579 ExtInfo->Mapping[i].extLocation = 0; 2580 } else { 2581 AdPrint(("Resize zero (9)\n")); 2582 ASSERT(!ExtInfo->Offset); 2583 UDFMarkSpaceAsXXX(Vcb, 0, ExtInfo->Mapping, AS_DISCARDED); // mark as free 2584 s = UDFGetMappingLength(ExtInfo->Mapping); 2585 if(!MyReallocPool__((int8*)(ExtInfo->Mapping), s, (int8**)&(ExtInfo->Mapping), 2*sizeof(EXTENT_MAP))) { 2586 // This must never happen on truncate !!! 2587 AdPrint(("ResizeExtent: MyReallocPool__(9) failed\n")); 2588 } 2589 ExtInfo->Mapping[0].extLength = LBS | (EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30); 2590 ExtInfo->Mapping[0].extLocation = 2591 ExtInfo->Mapping[1].extLength = 2592 ExtInfo->Mapping[1].extLocation = 0; 2593 } 2594 if(ExtInfo->Offset) { 2595 if(!AlwaysInIcb) { 2596 // remove 1st entry pointing to FileEntry 2597 s = UDFGetMappingLength(ExtInfo->Mapping); 2598 RtlMoveMemory(&(ExtInfo->Mapping[0]), &(ExtInfo->Mapping[1]), s - sizeof(EXTENT_MAP)); 2599 if(!MyReallocPool__((int8*)(ExtInfo->Mapping), s, 2600 (int8**)&(ExtInfo->Mapping), s - sizeof(EXTENT_MAP) )) { 2601 // This must never happen on truncate !!! 2602 AdPrint(("ResizeExtent: MyReallocPool__(10) failed\n")); 2603 } 2604 Length -= ExtInfo->Offset; 2605 ExtInfo->Offset = 0; 2606 } else { 2607 Length -= ExtInfo->Offset; // back to in-icb 2608 } 2609 } 2610 ExtInfo->Length = Length; 2611 UDFCheckSpaceAllocation(Vcb, 0, ExtInfo->Mapping, AS_USED); // check if used 2612 2613 for(i=0; (ExtInfo->Mapping[i].extLength & UDF_EXTENT_LENGTH_MASK); i++) { 2614 ExtPrint(("Resized Ext: type %x, loc %x, len %x\n", 2615 ExtInfo->Mapping[i].extLength >> 30, ExtInfo->Mapping[i].extLocation, ExtInfo->Mapping[i].extLength & UDF_EXTENT_LENGTH_MASK)); 2616 } 2617 2618 return STATUS_SUCCESS; 2619 } // end UDFResizeExtent() 2620 2621 /* 2622 This routine (re)builds AllocDescs data for all allocation modes except 2623 in-ICB & resizes associated extent (FileInfo->Dloc->AllocLoc) for 2624 already allocated user data extent (FileInfo->Dloc->DataLoc). 2625 AllocMode in FileEntry pointed by FileInfo must be already initialized. 2626 */ 2627 OSSTATUS 2628 UDFBuildAllocDescs( 2629 IN PVCB Vcb, 2630 IN uint32 PartNum, 2631 IN OUT PUDF_FILE_INFO FileInfo, 2632 OUT int8** AllocData 2633 ) 2634 { 2635 // PEXTENT_MAP InMap; 2636 // uint32 i=0; 2637 int8* Allocs; 2638 uint16 AllocMode; 2639 uint32 InitSz; 2640 OSSTATUS status; 2641 2642 ValidateFileInfo(FileInfo); 2643 AdPrint(("BuildAllocDesc\n")); 2644 // get space available in the 1st LBlock after FE 2645 InitSz = Vcb->LBlockSize - FileInfo->Dloc->FileEntryLen; 2646 Allocs = (int8*)MyAllocatePool__(NonPagedPool, InitSz); 2647 if(!Allocs) { 2648 *AllocData = NULL; 2649 AdPrint(("BuildAllocDesc: cant alloc %x bytes for Allocs\n", InitSz)); 2650 return STATUS_INSUFFICIENT_RESOURCES; 2651 } 2652 RtlZeroMemory(Allocs, InitSz); 2653 // InMap = FileInfo->Dloc->DataLoc.Mapping; 2654 UDFCheckSpaceAllocation(Vcb, 0, InMap, AS_USED); // check if used 2655 2656 // TODO: move data from mapped locations here 2657 2658 AllocMode = ((PFILE_ENTRY)(FileInfo->Dloc->FileEntry))->icbTag.flags & ICB_FLAG_ALLOC_MASK; 2659 switch(AllocMode) { 2660 case ICB_FLAG_AD_IN_ICB: { 2661 MyFreePool__(Allocs); 2662 ASSERT(!FileInfo->Dloc->AllocLoc.Mapping); 2663 Allocs = NULL; 2664 status = STATUS_SUCCESS; 2665 break; 2666 } 2667 case ICB_FLAG_AD_SHORT: { 2668 status = UDFBuildShortAllocDescs(Vcb, PartNum, &Allocs, InitSz, FileInfo); 2669 break; 2670 } 2671 case ICB_FLAG_AD_LONG: { 2672 status = UDFBuildLongAllocDescs(Vcb, PartNum, &Allocs, InitSz, FileInfo); 2673 break; 2674 } 2675 /* case ICB_FLAG_AD_EXTENDED: { 2676 status = UDFBuildExtAllocDescs(Vcb, PartNum, &Allocs, InitSz, FileInfo); 2677 break; 2678 }*/ 2679 default: { 2680 MyFreePool__(Allocs); 2681 Allocs = NULL; 2682 status = STATUS_INVALID_PARAMETER; 2683 } 2684 } 2685 2686 *AllocData = Allocs; 2687 UDFCheckSpaceAllocation(Vcb, 0, FileInfo->Dloc->DataLoc.Mapping, AS_USED); // check if used 2688 2689 return status; 2690 } // end UDFBuildAllocDescs() 2691 2692 /* 2693 This routine discards file's allocation 2694 */ 2695 void 2696 UDFFreeFileAllocation( 2697 IN PVCB Vcb, 2698 IN PUDF_FILE_INFO DirInfo, 2699 IN PUDF_FILE_INFO FileInfo 2700 ) 2701 { 2702 if(FileInfo->Dloc->DataLoc.Offset) { 2703 // in-ICB data 2704 if(FileInfo->Dloc->DataLoc.Mapping) { 2705 ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation == 2706 FileInfo->Dloc->DataLoc.Mapping[0].extLocation); 2707 UDFMarkSpaceAsXXX(Vcb, FileInfo->Dloc, &(FileInfo->Dloc->DataLoc.Mapping[1]), AS_DISCARDED); // free 2708 FileInfo->Dloc->DataLoc.Mapping[1].extLocation = 2709 FileInfo->Dloc->DataLoc.Mapping[1].extLength = 0; 2710 FileInfo->Dloc->DataLoc.Mapping[0].extLocation = 0; 2711 FileInfo->Dloc->DataLoc.Mapping[0].extLength = EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30; 2712 } 2713 if(FileInfo->Dloc->AllocLoc.Mapping) { 2714 ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation == 2715 FileInfo->Dloc->AllocLoc.Mapping[0].extLocation); 2716 UDFMarkSpaceAsXXX(Vcb, FileInfo->Dloc, &(FileInfo->Dloc->AllocLoc.Mapping[1]), AS_DISCARDED); // free 2717 FileInfo->Dloc->AllocLoc.Mapping[1].extLocation = 2718 FileInfo->Dloc->AllocLoc.Mapping[1].extLength = 0; 2719 FileInfo->Dloc->AllocLoc.Mapping[0].extLocation = 0; 2720 FileInfo->Dloc->AllocLoc.Mapping[0].extLength = EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30; 2721 } 2722 UDFFreeFESpace(Vcb, DirInfo, &(FileInfo->Dloc->FELoc)); 2723 } else { 2724 if(FileInfo->Dloc->AllocLoc.Mapping) { 2725 ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation == 2726 FileInfo->Dloc->AllocLoc.Mapping[0].extLocation); 2727 UDFMarkSpaceAsXXX(Vcb, FileInfo->Dloc, &(FileInfo->Dloc->AllocLoc.Mapping[1]), AS_DISCARDED); // free 2728 FileInfo->Dloc->AllocLoc.Mapping[1].extLocation = 2729 FileInfo->Dloc->AllocLoc.Mapping[1].extLength = 0; 2730 FileInfo->Dloc->AllocLoc.Mapping[0].extLocation = 0; 2731 FileInfo->Dloc->AllocLoc.Mapping[0].extLength = EXTENT_NOT_RECORDED_NOT_ALLOCATED << 30; 2732 } 2733 UDFFreeFESpace(Vcb, DirInfo, &(FileInfo->Dloc->FELoc)); 2734 UDFMarkSpaceAsXXX(Vcb, FileInfo->Dloc, FileInfo->Dloc->DataLoc.Mapping, AS_DISCARDED); // free 2735 } 2736 FileInfo->Dloc->DataLoc.Modified = 2737 FileInfo->Dloc->AllocLoc.Modified = 2738 FileInfo->Dloc->FELoc.Modified = FALSE; 2739 } // end UDFFreeFileAllocation() 2740 #endif //UDF_READ_ONLY_BUILD 2741 2742 /* 2743 This routine packs physically sequential extents into single one 2744 */ 2745 void 2746 __fastcall 2747 UDFPackMapping( 2748 IN PVCB Vcb, 2749 IN PEXTENT_INFO ExtInfo // Extent array 2750 ) 2751 { 2752 PEXTENT_MAP NewMap, OldMap; 2753 uint32 i, j, l; 2754 uint32 LastLba, LastType, OldLen; 2755 uint32 OldSize, NewSize; 2756 #ifdef UDF_DBG 2757 int64 check_size; 2758 #endif //UDF_DBG 2759 2760 AdPrint(("Pack ExtInfo %x, Mapping %x\n", ExtInfo, ExtInfo->Mapping)); 2761 AdPrint((" Length %x\n", ExtInfo->Length)); 2762 2763 OldMap = ExtInfo->Mapping; 2764 LastLba = OldMap[0].extLocation; 2765 OldLen = (OldMap[0].extLength & UDF_EXTENT_LENGTH_MASK) >> Vcb->BlockSizeBits; 2766 LastType = OldMap[0].extLength >> 30; 2767 OldSize = 2768 NewSize = UDFGetMappingLength(OldMap); 2769 #ifdef UDF_DBG 2770 check_size = UDFGetExtentLength(ExtInfo->Mapping); 2771 ASSERT(!(check_size & (2048-1))); 2772 #endif //UDF_DBG 2773 2774 l=OldMap[0].extLength & UDF_EXTENT_LENGTH_MASK; 2775 // calculate required length 2776 for(i=1; OldMap[i].extLength; i++) { 2777 if((LastType == (OldMap[i].extLength >> 30)) 2778 && 2779 ((OldMap[i].extLocation == LastLba + OldLen) || 2780 (!OldMap[i].extLocation && !LastLba && (LastType == EXTENT_NOT_RECORDED_NOT_ALLOCATED))) 2781 && 2782 (l + (OldMap[i].extLength & UDF_EXTENT_LENGTH_MASK) <= UDF_MAX_EXTENT_LENGTH)) { 2783 // we can pack two blocks in one 2784 l += OldMap[i].extLength & UDF_EXTENT_LENGTH_MASK; 2785 NewSize -= sizeof(EXTENT_MAP); 2786 } else { 2787 l = OldMap[i].extLength & UDF_EXTENT_LENGTH_MASK; 2788 } 2789 LastLba = OldMap[i].extLocation; 2790 LastType = OldMap[i].extLength >> 30; 2791 OldLen = (OldMap[i].extLength & UDF_EXTENT_LENGTH_MASK) >> Vcb->BlockSizeBits; 2792 } 2793 // no changes ? 2794 if(OldSize <= (NewSize + PACK_MAPPING_THRESHOLD)) { 2795 if(OldSize == NewSize) 2796 return; 2797 if(NewSize >= PACK_MAPPING_THRESHOLD) 2798 return; 2799 } 2800 AdPrint(("Pack ExtInfo %x, Mapping %x, realloc\n", ExtInfo, ExtInfo->Mapping)); 2801 NewMap = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , NewSize, 2802 MEM_EXTMAP_TAG); 2803 // can't alloc ? 2804 if(!NewMap) return; 2805 // Ok, lets pack it... 2806 j=0; 2807 NewMap[0] = OldMap[0]; 2808 LastLba = OldMap[0].extLocation; 2809 OldLen = (OldMap[0].extLength & UDF_EXTENT_LENGTH_MASK) >> Vcb->BlockSizeBits; 2810 LastType = OldMap[0].extLength >> 30; 2811 for(i=1; OldMap[i].extLength; i++) { 2812 2813 ExtPrint(("oShExt: type %x, loc %x, len %x\n", 2814 OldMap[i].extLength >> 30, OldMap[i].extLocation, OldMap[i].extLength & UDF_EXTENT_LENGTH_MASK)); 2815 2816 if((LastType == (OldMap[i].extLength >> 30)) 2817 && 2818 ((OldMap[i].extLocation == LastLba + OldLen) || 2819 (!OldMap[i].extLocation && !LastLba && (LastType == EXTENT_NOT_RECORDED_NOT_ALLOCATED))) 2820 && 2821 ((NewMap[j].extLength & UDF_EXTENT_LENGTH_MASK) + (OldMap[i].extLength & UDF_EXTENT_LENGTH_MASK) <= UDF_MAX_EXTENT_LENGTH)) { 2822 NewMap[j].extLength += OldMap[i].extLength & UDF_EXTENT_LENGTH_MASK; 2823 } else { 2824 j++; 2825 NewMap[j] = OldMap[i]; 2826 } 2827 2828 ExtPrint(("nShExt: type %x, loc %x, len %x\n", 2829 NewMap[j].extLength >> 30, NewMap[j].extLocation, NewMap[j].extLength & UDF_EXTENT_LENGTH_MASK)); 2830 2831 LastLba = OldMap[i].extLocation; 2832 LastType = OldMap[i].extLength >> 30; 2833 OldLen = (OldMap[i].extLength & UDF_EXTENT_LENGTH_MASK) >> Vcb->BlockSizeBits; 2834 } 2835 // write terminator 2836 j++; 2837 ASSERT(NewSize == (j+1)*sizeof(EXTENT_MAP)); 2838 NewMap[j].extLength = 2839 NewMap[j].extLocation = 0; 2840 2841 #ifdef UDF_DBG 2842 ASSERT(check_size == UDFGetExtentLength(ExtInfo->Mapping)); 2843 ASSERT(check_size == UDFGetExtentLength(NewMap)); 2844 #endif 2845 2846 AdPrint(("Pack ExtInfo %x, NewMap %x, OldMap %x\n", ExtInfo, NewMap, OldMap)); 2847 2848 ExtInfo->Mapping = NewMap; 2849 MyFreePool__(OldMap); 2850 2851 AdPrint(("Pack ExtInfo %x, Mapping %x\n", ExtInfo, ExtInfo->Mapping)); 2852 AdPrint((" Length %x\n", ExtInfo->Length)); 2853 } // end UDFPackMapping() 2854 2855 /* 2856 This routine expands mapping to 'frag-per-LBlock' state 2857 */ 2858 OSSTATUS 2859 __fastcall 2860 UDFUnPackMapping( 2861 IN PVCB Vcb, 2862 IN PEXTENT_INFO ExtInfo // Extent array 2863 ) 2864 { 2865 PEXTENT_MAP NewMapping; 2866 PEXTENT_MAP Mapping = ExtInfo->Mapping; 2867 uint32 LBS = Vcb->LBlockSize; 2868 uint32 len = (uint32)(UDFGetExtentLength(Mapping) >> Vcb->LBlockSizeBits); 2869 uint32 i,j, type, base, d; 2870 LONG l; 2871 2872 NewMapping = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , (len+1)*sizeof(EXTENT_MAP), 2873 MEM_EXTMAP_TAG); 2874 if(!NewMapping) return STATUS_INSUFFICIENT_RESOURCES; 2875 2876 j=0; 2877 d = LBS >> Vcb->BlockSizeBits; 2878 for(i=0; (l = (Mapping[i].extLength & UDF_EXTENT_LENGTH_MASK)); i++) { 2879 base = Mapping[i].extLocation; 2880 type = Mapping[i].extLength & UDF_EXTENT_FLAG_MASK; 2881 for(; l>=(LONG)LBS; j++) { 2882 NewMapping[j].extLength = LBS | type; 2883 NewMapping[j].extLocation = base; 2884 base+=d; 2885 l-=LBS; 2886 } 2887 } 2888 // record terminator 2889 ASSERT(NewMapping); 2890 RtlZeroMemory(&(NewMapping[j]), sizeof(EXTENT_MAP)); 2891 MyFreePool__(Mapping); 2892 ExtInfo->Mapping = NewMapping; 2893 2894 return STATUS_SUCCESS; 2895 } // end UDFUnPackMapping() 2896 2897 /* 2898 Relocate a part of extent that starts from relative (inside extent) 2899 block number 'ExtBlock' and has length of 'BC' blocks to continuous 2900 run which starts at block 'Lba' 2901 */ 2902 OSSTATUS 2903 UDFRelocateExtent( 2904 IN PVCB Vcb, 2905 IN PEXTENT_INFO ExtInfo, 2906 IN uint32 ExtBlock, 2907 IN uint32 Lba, 2908 IN uint32 BC 2909 ) 2910 { 2911 return STATUS_ACCESS_DENIED; 2912 } 2913 2914 /* 2915 This routine checks if all the data required is in cache. 2916 */ 2917 BOOLEAN 2918 UDFIsExtentCached( 2919 IN PVCB Vcb, 2920 IN PEXTENT_INFO ExtInfo, // Extent array 2921 IN int64 Offset, // offset in extent 2922 IN uint32 Length, 2923 IN BOOLEAN ForWrite 2924 ) 2925 { 2926 BOOLEAN retstat = FALSE; 2927 PEXTENT_MAP Extent = ExtInfo->Mapping; // Extent array 2928 SIZE_T to_read; 2929 uint32 Lba, sect_offs, flags, i; 2930 2931 WCacheStartDirect__(&(Vcb->FastCache), Vcb, TRUE/*FALSE*//*ForWrite*/); 2932 if(!ExtInfo || !ExtInfo->Mapping) goto EO_IsCached; 2933 if(!Length) { 2934 retstat = TRUE; 2935 goto EO_IsCached; 2936 } 2937 2938 // prevent reading out of data space 2939 if(Offset > ExtInfo->Length) goto EO_IsCached; 2940 if(Offset+Length > ExtInfo->Length) goto EO_IsCached; 2941 Offset += ExtInfo->Offset; // used for in-ICB data 2942 // read maximal possible part of each frag of extent 2943 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_read, &flags, &i); 2944 while(((LONG)Length) > 0) { 2945 // EOF check 2946 if(Lba == LBA_OUT_OF_EXTENT) goto EO_IsCached; 2947 Extent += (i + 1); 2948 // check for reading tail 2949 to_read = min(to_read, Length); 2950 if(flags == EXTENT_RECORDED_ALLOCATED) { 2951 retstat = UDFIsDataCached(Vcb, Lba, (to_read+sect_offs+Vcb->BlockSize-1)>>Vcb->BlockSizeBits); 2952 if(!retstat) goto EO_IsCached; 2953 } else if(ForWrite) { 2954 goto EO_IsCached; 2955 } 2956 Offset += to_read; 2957 Length -= to_read; 2958 Lba = UDFNextExtentToLba(Vcb, Extent, &to_read, &flags, &i); 2959 } 2960 retstat = TRUE; 2961 EO_IsCached: 2962 if(!retstat) { 2963 WCacheEODirect__(&(Vcb->FastCache), Vcb); 2964 } 2965 return retstat; 2966 } // end UDFIsExtentCached() 2967 2968 /* 2969 This routine reads cached data only. 2970 */ 2971 /*OSSTATUS 2972 UDFReadExtentCached( 2973 IN PVCB Vcb, 2974 IN PEXTENT_INFO ExtInfo, // Extent array 2975 IN int64 Offset, // offset in extent 2976 IN uint32 Length, 2977 OUT int8* Buffer, 2978 OUT PSIZE_T ReadBytes 2979 ) 2980 { 2981 (*ReadBytes) = 0; 2982 if(!ExtInfo || !ExtInfo->Mapping) return STATUS_INVALID_PARAMETER; 2983 2984 PEXTENT_MAP Extent = ExtInfo->Mapping; // Extent array 2985 uint32 to_read, Lba, sect_offs, flags; 2986 SIZE_T _ReadBytes; 2987 OSSTATUS status; 2988 // prevent reading out of data space 2989 if(Offset > ExtInfo->Length) return STATUS_END_OF_FILE; 2990 if(Offset+Length > ExtInfo->Length) Length = (uint32)(ExtInfo->Length - Offset); 2991 Offset += ExtInfo->Offset; // used for in-ICB data 2992 // read maximal possible part of each frag of extent 2993 while(((LONG)Length) > 0) { 2994 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_read, &flags, NULL); 2995 // EOF check 2996 if(Lba == LBA_OUT_OF_EXTENT) return STATUS_END_OF_FILE; 2997 // check for reading tail 2998 to_read = (to_read < Length) ? 2999 to_read : Length; 3000 if(flags == EXTENT_RECORDED_ALLOCATED) { 3001 status = UDFReadDataCached(Vcb, TRUE, ( ((uint64)Lba) << Vcb->BlockSizeBits) + sect_offs, to_read, Buffer, &_ReadBytes); 3002 (*ReadBytes) += _ReadBytes; 3003 } else { 3004 RtlZeroMemory(Buffer, to_read); 3005 (*ReadBytes) += to_read; 3006 status = STATUS_SUCCESS; 3007 } 3008 if(!OS_SUCCESS(status)) return status; 3009 // prepare for reading next frag... 3010 Buffer += to_read; 3011 Offset += to_read; 3012 Length -= to_read; 3013 } 3014 return STATUS_SUCCESS; 3015 } // end UDFReadExtentCached()*/ 3016 3017 /* 3018 This routine reads data at any offset from specified extent. 3019 */ 3020 OSSTATUS 3021 UDFReadExtent( 3022 IN PVCB Vcb, 3023 IN PEXTENT_INFO ExtInfo, // Extent array 3024 IN int64 Offset, // offset in extent 3025 IN SIZE_T Length, 3026 IN BOOLEAN Direct, 3027 OUT int8* Buffer, 3028 OUT PSIZE_T ReadBytes 3029 ) 3030 { 3031 (*ReadBytes) = 0; 3032 if(!ExtInfo || !ExtInfo->Mapping) return STATUS_INVALID_PARAMETER; 3033 ASSERT((uintptr_t)Buffer > 0x1000); 3034 3035 AdPrint(("Read ExtInfo %x, Mapping %x\n", ExtInfo, ExtInfo->Mapping)); 3036 3037 PEXTENT_MAP Extent = ExtInfo->Mapping; // Extent array 3038 SIZE_T to_read, _ReadBytes; 3039 uint32 Lba, sect_offs, flags; 3040 uint32 index; 3041 OSSTATUS status; 3042 // prevent reading out of data space 3043 if(Offset > ExtInfo->Length) return STATUS_END_OF_FILE; 3044 if(Offset+Length > ExtInfo->Length) Length = (uint32)(ExtInfo->Length - Offset); 3045 Offset += ExtInfo->Offset; // used for in-ICB data 3046 // read maximal possible part of each frag of extent 3047 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_read, &flags, &index); 3048 _ReadBytes = index; 3049 while(Length) { 3050 // EOF check 3051 if(Lba == LBA_OUT_OF_EXTENT) return STATUS_END_OF_FILE; 3052 Extent += (_ReadBytes + 1); 3053 // check for reading tail 3054 to_read = min(to_read, Length); 3055 if(flags == EXTENT_RECORDED_ALLOCATED) { 3056 status = UDFReadData(Vcb, TRUE, ( ((uint64)Lba) << Vcb->BlockSizeBits) + sect_offs, to_read, Direct, Buffer, &_ReadBytes); 3057 (*ReadBytes) += _ReadBytes; 3058 if(!OS_SUCCESS(status)) return status; 3059 } else { 3060 RtlZeroMemory(Buffer, to_read); 3061 (*ReadBytes) += to_read; 3062 } 3063 // prepare for reading next frag... 3064 Length -= to_read; 3065 if(!Length) 3066 break; 3067 ASSERT(to_read); 3068 Buffer += to_read; 3069 // Offset += to_read; 3070 Lba = UDFNextExtentToLba(Vcb, Extent, &to_read, &flags, &index); 3071 _ReadBytes = index; 3072 sect_offs = 0; 3073 } 3074 return STATUS_SUCCESS; 3075 } // end UDFReadExtent() 3076 3077 /* 3078 This routine reads and builds mapping for 3079 specified amount of data at any offset from specified extent. 3080 Size of output buffer is limited by *_SubExtInfoSz 3081 */ 3082 OSSTATUS 3083 UDFReadExtentLocation( 3084 IN PVCB Vcb, 3085 IN PEXTENT_INFO ExtInfo, // Extent array 3086 IN int64 Offset, // offset in extent to start SubExtent from 3087 OUT PEXTENT_MAP* _SubExtInfo, // SubExtent mapping array 3088 IN OUT uint32* _SubExtInfoSz, // IN: maximum number of fragments to get 3089 // OUT: actually obtained fragments 3090 OUT int64* _NextOffset // offset, caller can start from to continue 3091 ) 3092 { 3093 if(!ExtInfo || !ExtInfo->Mapping) 3094 return STATUS_INVALID_PARAMETER; 3095 3096 PEXTENT_MAP Extent = ExtInfo->Mapping; // Extent array 3097 PEXTENT_MAP SubExtInfo; 3098 SIZE_T to_read; 3099 uint32 Lba, sect_offs, flags, Skip_MapEntries; 3100 int32 SubExtInfoSz = *_SubExtInfoSz; 3101 int64 Length; 3102 int64 NextOffset; 3103 // OSSTATUS status = STATUS_BUFFER_OVERFLOW; 3104 3105 (*_SubExtInfo) = NULL; 3106 (*_SubExtInfoSz) = 0; 3107 NextOffset = Offset; 3108 // prevent reading out of data space 3109 if(Offset >= ExtInfo->Length) 3110 return STATUS_END_OF_FILE; 3111 Length = ExtInfo->Length - Offset; 3112 Offset += ExtInfo->Offset; // used for in-ICB data 3113 // read maximal possible part of each frag of extent 3114 SubExtInfo = (PEXTENT_MAP)MyAllocatePoolTag__(NonPagedPool , SubExtInfoSz*sizeof(EXTENT_MAP), 3115 MEM_EXTMAP_TAG); 3116 (*_SubExtInfo) = SubExtInfo; 3117 if(!SubExtInfo) 3118 return STATUS_INSUFFICIENT_RESOURCES; 3119 3120 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_read, &flags, &Skip_MapEntries); 3121 while(Length && SubExtInfoSz) { 3122 // EOF check 3123 if(Lba == LBA_OUT_OF_EXTENT) { 3124 BrutePoint(); 3125 return STATUS_END_OF_FILE; 3126 } 3127 Extent += (Skip_MapEntries + 1); 3128 // check for reading tail 3129 to_read = (int32)min((int64)to_read, Length); 3130 SubExtInfo->extLength = to_read; 3131 if(flags == EXTENT_NOT_RECORDED_NOT_ALLOCATED) { 3132 SubExtInfo->extLocation = LBA_NOT_ALLOCATED; 3133 } else 3134 if(flags == EXTENT_NOT_RECORDED_ALLOCATED) { 3135 ASSERT(!(Lba & 0x80000000)); 3136 SubExtInfo->extLocation = Lba | 0x80000000; 3137 } else { 3138 SubExtInfo->extLocation = Lba; 3139 } 3140 (*_SubExtInfoSz)++; 3141 SubExtInfoSz--; 3142 NextOffset += to_read; 3143 // prepare for reading next frag... 3144 Length -= to_read; 3145 if(!Length) { 3146 // status = STATUS_SUCCESS; 3147 break; 3148 } 3149 ASSERT(to_read); 3150 Lba = UDFNextExtentToLba(Vcb, Extent, &to_read, &flags, &Skip_MapEntries); 3151 sect_offs = 0; 3152 } 3153 (*_NextOffset) = NextOffset; 3154 return STATUS_SUCCESS; 3155 } // end UDFReadExtentLocation() 3156 3157 #ifdef _MSC_VER 3158 #pragma warning(push) 3159 #pragma warning(disable:4035) // re-enable below 3160 #endif 3161 3162 uint32 3163 UDFGetZeroLength( 3164 IN int8* Buffer, 3165 IN uint32 Length 3166 ) 3167 { 3168 uint32 i; 3169 Length /= sizeof(uint32); 3170 for(i=0; i<Length; i++) { 3171 if( ((uint32*)Buffer)[i] ) 3172 break; 3173 } 3174 return Length*sizeof(uint32); 3175 } 3176 3177 #ifdef _MSC_VER 3178 #pragma warning(pop) // re-enable warning #4035 3179 #endif 3180 3181 #ifndef UDF_READ_ONLY_BUILD 3182 /* 3183 This routine writes data at any offset to specified extent. 3184 */ 3185 OSSTATUS 3186 UDFWriteExtent( 3187 IN PVCB Vcb, 3188 IN PEXTENT_INFO ExtInfo, // Extent array 3189 IN int64 Offset, // offset in extent 3190 IN SIZE_T Length, 3191 IN BOOLEAN Direct, // setting this flag delays flushing of given 3192 // data to indefinite term 3193 IN int8* Buffer, 3194 OUT PSIZE_T WrittenBytes 3195 ) 3196 { 3197 if(!ExtInfo || !ExtInfo->Mapping) 3198 return STATUS_INVALID_PARAMETER; 3199 3200 PEXTENT_MAP Extent = ExtInfo->Mapping; // Extent array 3201 uint32 Lba, sect_offs, flags; 3202 OSSTATUS status; 3203 SIZE_T to_write, _WrittenBytes; 3204 BOOLEAN reread_lba; 3205 // BOOLEAN already_prepared = FALSE; 3206 // BOOLEAN prepare = !Buffer; 3207 3208 AdPrint(("Write ExtInfo %x, Mapping %x\n", ExtInfo, ExtInfo->Mapping)); 3209 3210 Offset += ExtInfo->Offset; // used for in-ICB data 3211 // write maximal possible part of each frag of extent 3212 while(((LONG)Length) > 0) { 3213 UDFCheckSpaceAllocation(Vcb, 0, Extent, AS_USED); // check if used 3214 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_write, &flags, NULL); 3215 // EOF check 3216 if(Lba == LBA_OUT_OF_EXTENT) { 3217 return STATUS_END_OF_FILE; 3218 } 3219 /* if((to_write < Length) && 3220 !Direct && !prepare && !already_prepared) { 3221 // rebuild mapping, allocate space, etc. 3222 // to indicate this, set Buffer to NULL 3223 AdPrint(("UDFWriteExtent: Prepare\n")); 3224 BrutePoint(); 3225 _WrittenBytes = 0; 3226 status = UDFWriteExtent(Vcb, ExtInfo, Offset, Length, *//*Direct*//*FALSE, NULL, &_WrittenBytes); 3227 if(!OS_SUCCESS(status)) { 3228 return status; 3229 } 3230 Extent = ExtInfo->Mapping; 3231 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_write, &flags, NULL); 3232 already_prepared = TRUE; 3233 }*/ 3234 if(flags == EXTENT_NOT_RECORDED_NOT_ALLOCATED) { 3235 // here we should allocate space for this extent 3236 if(!OS_SUCCESS(status = UDFMarkNotAllocatedAsAllocated(Vcb, Offset, to_write, ExtInfo))) 3237 return status; 3238 Extent = ExtInfo->Mapping; 3239 UDFCheckSpaceAllocation(Vcb, 0, Extent, AS_USED); // check if used 3240 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_write, &flags, NULL); 3241 if(Lba == LBA_OUT_OF_EXTENT) { 3242 return STATUS_END_OF_FILE; 3243 } 3244 // we have already re-read Lba 3245 reread_lba = FALSE; 3246 } else { 3247 // we may need to re-read Lba if some changes are 3248 // made while converting from Alloc-Not-Rec 3249 reread_lba = TRUE; 3250 } 3251 // check if writing to not recorded allocated 3252 // in this case we must pad blocks with zeros around 3253 // modified area 3254 // 3255 // ...|xxxxxxxx|xxxxxxxx|xxxxxxxx|... 3256 // . . 3257 // . || . 3258 // . \/ . 3259 // . . 3260 // ...|000ddddd|dddddddd|dd000000|... 3261 // . . 3262 // ^ ^ 3263 // sect_offs sect_offs+to_write 3264 // . . 3265 // .<-- to_write -->. 3266 // 3267 to_write = min(to_write, Length); 3268 if(flags == EXTENT_NOT_RECORDED_ALLOCATED) { 3269 if(!OS_SUCCESS(status = UDFMarkAllocatedAsRecorded(Vcb, Offset, to_write, ExtInfo))) 3270 return status; 3271 Extent = ExtInfo->Mapping; 3272 UDFCheckSpaceAllocation(Vcb, 0, Extent, AS_USED); // check if used 3273 if(reread_lba) { 3274 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_write, &flags, NULL); 3275 to_write = min(to_write, Length); 3276 } 3277 /* 3278 we must fill 1st block with zeros in 1 of 2 cases: 3279 1) start offset is not aligned on LBlock boundary 3280 OR 3281 2) end offset is not aligned on LBlock boundary and lays in 3282 the same LBlock 3283 3284 we must fill last block with zeros if both 3285 1) end offset is not aligned on LBlock boundary 3286 AND 3287 2) end offset DOESN'T lay in the 1st LBlock 3288 */ 3289 3290 // if(!prepare) { 3291 // pad 1st logical block 3292 if((sect_offs || (sect_offs + to_write < Vcb->LBlockSize) ) 3293 && 3294 !Vcb->CDR_Mode) { 3295 status = UDFWriteData(Vcb, TRUE, 3296 ( ((uint64)Lba) << Vcb->BlockSizeBits), 3297 Vcb->LBlockSize, Direct, Vcb->ZBuffer, &_WrittenBytes); 3298 if(!OS_SUCCESS(status)) 3299 return status; 3300 } 3301 // pad last logical block 3302 if((sect_offs + to_write > Vcb->LBlockSize) && 3303 (sect_offs + to_write) & (Vcb->LBlockSize - 1)) { 3304 status = UDFWriteData(Vcb, TRUE, 3305 (( ((uint64)Lba) << Vcb->BlockSizeBits) + sect_offs + to_write) & ~((int64)(Vcb->LBlockSize)-1), 3306 Vcb->LBlockSize, Direct, Vcb->ZBuffer, &_WrittenBytes); 3307 } 3308 if(!OS_SUCCESS(status)) 3309 return status; 3310 /* } else { 3311 status = STATUS_SUCCESS; 3312 }*/ 3313 } 3314 ASSERT(to_write); 3315 // if(!prepare) { 3316 status = UDFWriteData(Vcb, TRUE, ( ((uint64)Lba) << Vcb->BlockSizeBits) + sect_offs, to_write, Direct, Buffer, &_WrittenBytes); 3317 *WrittenBytes += _WrittenBytes; 3318 if(!OS_SUCCESS(status)) return status; 3319 /* } else { 3320 status = STATUS_SUCCESS; 3321 *WrittenBytes += to_write; 3322 }*/ 3323 // prepare for writing next frag... 3324 Buffer += to_write; 3325 Offset += to_write; 3326 Length -= to_write; 3327 } 3328 AdPrint(("Write: ExtInfo %x, Mapping %x\n", ExtInfo, ExtInfo->Mapping)); 3329 return STATUS_SUCCESS; 3330 } // end UDFWriteExtent() 3331 3332 //#if 0 3333 /* 3334 This routine zeroes/deallocates data at any offset to specified extent. 3335 */ 3336 OSSTATUS 3337 UDFZeroExtent( 3338 IN PVCB Vcb, 3339 IN PEXTENT_INFO ExtInfo, // Extent array 3340 IN int64 Offset, // offset in extent 3341 IN SIZE_T Length, 3342 IN BOOLEAN Deallocate, // deallocate frag or just mark as unrecorded 3343 IN BOOLEAN Direct, // setting this flag delays flushing of given 3344 // data to indefinite term 3345 OUT PSIZE_T WrittenBytes 3346 ) 3347 { 3348 if(!ExtInfo || !ExtInfo->Mapping) 3349 return STATUS_INVALID_PARAMETER; 3350 3351 PEXTENT_MAP Extent = ExtInfo->Mapping; // Extent array 3352 uint32 Lba, sect_offs, flags; 3353 OSSTATUS status; 3354 SIZE_T to_write, _WrittenBytes; 3355 SIZE_T LBS = Vcb->LBlockSize; 3356 3357 AdPrint(("Zero ExtInfo %x, Mapping %x\n", ExtInfo, ExtInfo->Mapping)); 3358 3359 Offset += ExtInfo->Offset; // used for in-ICB data 3360 // fill/deallocate maximal possible part of each frag of extent 3361 while(((LONG)Length) > 0) { 3362 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_write, &flags, NULL); 3363 // EOF check 3364 if(Lba == LBA_OUT_OF_EXTENT) { 3365 return STATUS_END_OF_FILE; 3366 } 3367 // check for writing tail 3368 to_write = min(to_write, Length); 3369 3370 if(flags == EXTENT_NOT_RECORDED_NOT_ALLOCATED) { 3371 // here we should do nothing 3372 *WrittenBytes += to_write; 3373 } else 3374 if(flags == EXTENT_NOT_RECORDED_ALLOCATED) { 3375 // we should just deallocate this frag 3376 if(Deallocate) { 3377 if(!OS_SUCCESS(status = UDFMarkAllocatedAsNotAllocated(Vcb, Offset, to_write, ExtInfo))) 3378 return status; 3379 } 3380 Extent = ExtInfo->Mapping; 3381 *WrittenBytes += to_write; 3382 } else { 3383 // fill tail of the 1st Block with ZEROs 3384 if(sect_offs) { 3385 status = UDFWriteData(Vcb, TRUE, ( ((uint64)Lba) << Vcb->BlockSizeBits) + sect_offs, 3386 min(to_write, LBS-sect_offs), 3387 Direct, Vcb->ZBuffer, &_WrittenBytes); 3388 *WrittenBytes += _WrittenBytes; 3389 if(!OS_SUCCESS(status)) 3390 return status; 3391 Offset += _WrittenBytes; 3392 Length -= _WrittenBytes; 3393 to_write -= _WrittenBytes; 3394 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_write, &flags, NULL); 3395 ASSERT(flags != EXTENT_NOT_RECORDED_NOT_ALLOCATED); 3396 ASSERT(flags != EXTENT_NOT_RECORDED_ALLOCATED); 3397 ASSERT(!sect_offs); 3398 } 3399 // deallocate Blocks 3400 if(to_write >= LBS) { 3401 // use 'sect_offs' as length of extent to be deallocated 3402 sect_offs = to_write & ~(LBS - 1); 3403 if(Deallocate) { 3404 status = UDFMarkAllocatedAsNotAllocated(Vcb, Offset, sect_offs, ExtInfo); 3405 } else { 3406 status = UDFMarkRecordedAsAllocated(Vcb, Offset, sect_offs, ExtInfo); 3407 } 3408 if(!OS_SUCCESS(status)) 3409 return status; 3410 // reload extent mapping 3411 Extent = ExtInfo->Mapping; 3412 Offset += sect_offs; 3413 Length -= sect_offs; 3414 *WrittenBytes += sect_offs; 3415 to_write -= sect_offs; 3416 Lba = UDFExtentOffsetToLba(Vcb, Extent, Offset, §_offs, &to_write, &flags, NULL); 3417 ASSERT(flags != EXTENT_NOT_RECORDED_NOT_ALLOCATED); 3418 ASSERT(flags != EXTENT_NOT_RECORDED_ALLOCATED); 3419 ASSERT(!sect_offs); 3420 } 3421 // fill beginning of the last Block with ZEROs 3422 if(to_write) { 3423 status = UDFWriteData(Vcb, TRUE, ( ((uint64)Lba) << Vcb->BlockSizeBits), to_write, Direct, Vcb->ZBuffer, &_WrittenBytes); 3424 *WrittenBytes += _WrittenBytes; 3425 if(!OS_SUCCESS(status)) 3426 return status; 3427 ASSERT(to_write == _WrittenBytes); 3428 } 3429 } 3430 AdPrint(("Zero... ExtInfo %x, Mapping %x\n", ExtInfo, ExtInfo->Mapping)); 3431 // prepare for filling next frag... 3432 Offset += to_write; 3433 Length -= to_write; 3434 } 3435 AdPrint(("Zero: ExtInfo %x, Mapping %x\n", ExtInfo, ExtInfo->Mapping)); 3436 return STATUS_SUCCESS; 3437 } // end UDFZeroExtent() 3438 //#endif //0 3439 #endif //UDF_READ_ONLY_BUILD 3440