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 * 8 * File: Sys_Spec.cpp 9 * 10 * Module: UDF File System Driver 11 * (both User and Kernel mode execution) 12 * 13 * Description: 14 * Contains system-secific code 15 * 16 *************************************************************************/ 17 18 19 /* 20 This routine converts UDF timestamp to NT time 21 */ 22 LONGLONG 23 UDFTimeToNT( 24 IN PUDF_TIME_STAMP UdfTime 25 ) 26 { 27 LONGLONG NtTime; 28 TIME_FIELDS TimeFields; 29 30 TimeFields.Milliseconds = (USHORT)(UdfTime->centiseconds * 10 + UdfTime->hundredsOfMicroseconds / 100); 31 TimeFields.Second = (USHORT)(UdfTime->second); 32 TimeFields.Minute = (USHORT)(UdfTime->minute); 33 TimeFields.Hour = (USHORT)(UdfTime->hour); 34 TimeFields.Day = (USHORT)(UdfTime->day); 35 TimeFields.Month = (USHORT)(UdfTime->month); 36 TimeFields.Year = (USHORT)((UdfTime->year < 1601) ? 1601 : UdfTime->year); 37 38 if (!RtlTimeFieldsToTime(&TimeFields, (PLARGE_INTEGER)&NtTime)) { 39 NtTime = 0; 40 } else { 41 ExLocalTimeToSystemTime( (PLARGE_INTEGER)&NtTime, (PLARGE_INTEGER)&NtTime ); 42 } 43 44 return NtTime; 45 } // end UDFTimeToNT() 46 47 48 /* 49 This routine converts NT time to UDF timestamp 50 */ 51 VOID 52 UDFTimeToUDF( 53 IN LONGLONG NtTime, 54 OUT PUDF_TIME_STAMP UdfTime 55 ) 56 { 57 if(!NtTime) return; 58 LONGLONG LocalTime; 59 60 TIME_FIELDS TimeFields; 61 62 ExSystemTimeToLocalTime( (PLARGE_INTEGER)&NtTime, (PLARGE_INTEGER)&LocalTime ); 63 RtlTimeToTimeFields( (PLARGE_INTEGER)&LocalTime, &TimeFields ); 64 65 LocalTime /= 10; // microseconds 66 UdfTime->microseconds = (UCHAR)(NtTime % 100); 67 LocalTime /= 100; // hundreds of microseconds 68 UdfTime->hundredsOfMicroseconds = (UCHAR)(NtTime % 100); 69 LocalTime /= 100; // centiseconds 70 UdfTime->centiseconds = (UCHAR)(TimeFields.Milliseconds / 10); 71 UdfTime->second = (UCHAR)(TimeFields.Second); 72 UdfTime->minute = (UCHAR)(TimeFields.Minute); 73 UdfTime->hour = (UCHAR)(TimeFields.Hour); 74 UdfTime->day = (UCHAR)(TimeFields.Day); 75 UdfTime->month = (UCHAR)(TimeFields.Month); 76 UdfTime->year = (USHORT)(TimeFields.Year); 77 UdfTime->typeAndTimezone = (TIMESTAMP_TYPE_LOCAL << 14); 78 } // end UDFTimeToUDF() 79 80 /* 81 */ 82 ULONG 83 UDFAttributesToNT( 84 IN PDIR_INDEX_ITEM FileDirNdx, 85 IN tag* FileEntry 86 ) 87 { 88 ASSERT(FileDirNdx); 89 if( (FileDirNdx->FI_Flags & UDF_FI_FLAG_SYS_ATTR) && 90 !(FileDirNdx->FI_Flags & UDF_FI_FLAG_LINKED)) 91 return FileDirNdx->SysAttr; 92 93 ULONG NTAttr = 0; 94 ULONG attr = 0; //permissions 95 USHORT Flags = 0; 96 USHORT Type = 0; 97 UCHAR FCharact = 0; 98 99 if(!FileEntry) { 100 if(!FileDirNdx->FileInfo) 101 return 0; 102 ValidateFileInfo(FileDirNdx->FileInfo); 103 FileEntry = FileDirNdx->FileInfo->Dloc->FileEntry; 104 } 105 if(FileEntry->tagIdent == TID_FILE_ENTRY) { 106 attr = ((PFILE_ENTRY)FileEntry)->permissions; 107 Flags = ((PFILE_ENTRY)FileEntry)->icbTag.flags; 108 Type = ((PFILE_ENTRY)FileEntry)->icbTag.fileType; 109 if(((PFILE_ENTRY)FileEntry)->fileLinkCount > 1) 110 FileDirNdx->FI_Flags |= UDF_FI_FLAG_LINKED; 111 } else { 112 attr = ((PEXTENDED_FILE_ENTRY)FileEntry)->permissions; 113 Flags = ((PEXTENDED_FILE_ENTRY)FileEntry)->icbTag.flags; 114 Type = ((PEXTENDED_FILE_ENTRY)FileEntry)->icbTag.fileType; 115 if(((PEXTENDED_FILE_ENTRY)FileEntry)->fileLinkCount > 1) 116 FileDirNdx->FI_Flags |= UDF_FI_FLAG_LINKED; 117 } 118 FCharact = FileDirNdx->FileCharacteristics; 119 120 if(Flags & ICB_FLAG_SYSTEM) NTAttr |= FILE_ATTRIBUTE_SYSTEM; 121 if(Flags & ICB_FLAG_ARCHIVE) NTAttr |= FILE_ATTRIBUTE_ARCHIVE; 122 if((Type == UDF_FILE_TYPE_DIRECTORY) || 123 (Type == UDF_FILE_TYPE_STREAMDIR) || 124 (FCharact & FILE_DIRECTORY)) { 125 NTAttr |= FILE_ATTRIBUTE_DIRECTORY; 126 #ifdef UDF_DBG 127 } else { 128 //NTAttr |= FILE_ATTRIBUTE_NORMAL; 129 #endif 130 } 131 if(FCharact & FILE_HIDDEN) NTAttr |= FILE_ATTRIBUTE_HIDDEN; 132 if( !(attr & PERM_O_WRITE) && 133 !(attr & PERM_G_WRITE) && 134 !(attr & PERM_U_WRITE) && 135 !(attr & PERM_O_DELETE) && 136 !(attr & PERM_G_DELETE) && 137 !(attr & PERM_U_DELETE) ) { 138 NTAttr |= FILE_ATTRIBUTE_READONLY; 139 } 140 FileDirNdx->SysAttr = NTAttr; 141 return NTAttr; 142 } // end UDFAttributesToNT() 143 144 /* 145 */ 146 VOID 147 UDFAttributesToUDF( 148 IN PDIR_INDEX_ITEM FileDirNdx, 149 IN tag* FileEntry, 150 IN ULONG NTAttr 151 ) 152 { 153 PULONG attr; //permissions 154 PUSHORT Flags; 155 PUCHAR Type; 156 PUCHAR FCharact; 157 158 NTAttr &= UDF_VALID_FILE_ATTRIBUTES; 159 160 if(!FileEntry) { 161 ASSERT(FileDirNdx); 162 if(!FileDirNdx->FileInfo) 163 return; 164 ValidateFileInfo(FileDirNdx->FileInfo); 165 FileEntry = FileDirNdx->FileInfo->Dloc->FileEntry; 166 FileDirNdx->FileInfo->Dloc->FE_Flags |= UDF_FE_FLAG_FE_MODIFIED; 167 } 168 if(FileEntry->tagIdent == TID_FILE_ENTRY) { 169 attr = &((PFILE_ENTRY)FileEntry)->permissions; 170 Flags = &((PFILE_ENTRY)FileEntry)->icbTag.flags; 171 Type = &((PFILE_ENTRY)FileEntry)->icbTag.fileType; 172 } else { 173 attr = &((PEXTENDED_FILE_ENTRY)FileEntry)->permissions; 174 Flags = &((PEXTENDED_FILE_ENTRY)FileEntry)->icbTag.flags; 175 Type = &((PEXTENDED_FILE_ENTRY)FileEntry)->icbTag.fileType; 176 } 177 FCharact = &(FileDirNdx->FileCharacteristics); 178 179 if((*FCharact & FILE_DIRECTORY) || 180 (*Type == UDF_FILE_TYPE_STREAMDIR) || 181 (*Type == UDF_FILE_TYPE_DIRECTORY)) { 182 *FCharact |= FILE_DIRECTORY; 183 if(*Type != UDF_FILE_TYPE_STREAMDIR) 184 *Type = UDF_FILE_TYPE_DIRECTORY; 185 *attr |= (PERM_O_EXEC | PERM_G_EXEC | PERM_U_EXEC); 186 NTAttr |= FILE_ATTRIBUTE_DIRECTORY; 187 NTAttr &= ~FILE_ATTRIBUTE_NORMAL; 188 } else { 189 *FCharact &= ~FILE_DIRECTORY; 190 *Type = UDF_FILE_TYPE_REGULAR; 191 *attr &= ~(PERM_O_EXEC | PERM_G_EXEC | PERM_U_EXEC); 192 } 193 194 if(NTAttr & FILE_ATTRIBUTE_SYSTEM) { 195 *Flags |= ICB_FLAG_SYSTEM; 196 } else { 197 *Flags &= ~ICB_FLAG_SYSTEM; 198 } 199 if(NTAttr & FILE_ATTRIBUTE_ARCHIVE) { 200 *Flags |= ICB_FLAG_ARCHIVE; 201 } else { 202 *Flags &= ~ICB_FLAG_ARCHIVE; 203 } 204 if(NTAttr & FILE_ATTRIBUTE_HIDDEN) { 205 *FCharact |= FILE_HIDDEN; 206 } else { 207 *FCharact &= ~FILE_HIDDEN; 208 } 209 *attr |= (PERM_O_READ | PERM_G_READ | PERM_U_READ); 210 if(!(NTAttr & FILE_ATTRIBUTE_READONLY)) { 211 *attr |= (PERM_O_WRITE | PERM_G_WRITE | PERM_U_WRITE | 212 PERM_O_DELETE | PERM_G_DELETE | PERM_U_DELETE | 213 PERM_O_CHATTR | PERM_G_CHATTR | PERM_U_CHATTR); 214 } else { 215 *attr &= ~(PERM_O_WRITE | PERM_G_WRITE | PERM_U_WRITE | 216 PERM_O_DELETE | PERM_G_DELETE | PERM_U_DELETE | 217 PERM_O_CHATTR | PERM_G_CHATTR | PERM_U_CHATTR); 218 } 219 FileDirNdx->SysAttr = NTAttr; 220 if(FileDirNdx->FileInfo) 221 FileDirNdx->FileInfo->Dloc->FE_Flags |= UDF_FE_FLAG_FE_MODIFIED; 222 FileDirNdx->FI_Flags |= UDF_FI_FLAG_FI_MODIFIED; 223 return; 224 } // end UDFAttributesToUDF() 225 226 #ifndef _CONSOLE 227 /* 228 This routine fills PFILE_BOTH_DIR_INFORMATION structure (NT) 229 */ 230 NTSTATUS 231 UDFFileDirInfoToNT( 232 IN PVCB Vcb, 233 IN PDIR_INDEX_ITEM FileDirNdx, 234 OUT PFILE_BOTH_DIR_INFORMATION NTFileInfo 235 ) 236 { 237 PFILE_ENTRY FileEntry; 238 UNICODE_STRING UdfName; 239 UNICODE_STRING DosName; 240 PEXTENDED_FILE_ENTRY ExFileEntry; 241 USHORT Ident; 242 BOOLEAN ReadSizes; 243 NTSTATUS status; 244 PtrUDFNTRequiredFCB NtReqFcb; 245 246 UDFPrint(("@=%#x, FileDirNdx %x\n", &Vcb, FileDirNdx)); 247 248 ASSERT((ULONG)NTFileInfo > 0x1000); 249 RtlZeroMemory(NTFileInfo, sizeof(FILE_BOTH_DIR_INFORMATION)); 250 251 DosName.Buffer = (PWCHAR)&(NTFileInfo->ShortName); 252 DosName.MaximumLength = sizeof(NTFileInfo->ShortName); // 12*sizeof(WCHAR) 253 254 _SEH2_TRY { 255 UDFPrint((" DirInfoToNT: %*.*S\n", FileDirNdx->FName.Length/sizeof(WCHAR), FileDirNdx->FName.Length/sizeof(WCHAR), FileDirNdx->FName)); 256 } _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { 257 UDFPrint((" DirInfoToNT: exception when printing file name\n")); 258 } _SEH2_END; 259 260 if(FileDirNdx->FileInfo) { 261 UDFPrint((" FileInfo\n")); 262 // validate FileInfo 263 ValidateFileInfo(FileDirNdx->FileInfo); 264 if(UDFGetFileLinkCount(FileDirNdx->FileInfo) > 1) 265 FileDirNdx->FI_Flags |= UDF_FI_FLAG_LINKED; 266 FileEntry = (PFILE_ENTRY)(FileDirNdx->FileInfo->Dloc->FileEntry); 267 // read required sizes from Fcb (if any) if file is not linked 268 // otherwise we should read them from FileEntry 269 if(FileDirNdx->FileInfo->Fcb) { 270 UDFPrint((" Fcb\n")); 271 NtReqFcb = FileDirNdx->FileInfo->Fcb->NTRequiredFCB; 272 NTFileInfo->CreationTime.QuadPart = NtReqFcb->CreationTime.QuadPart; 273 NTFileInfo->LastWriteTime.QuadPart = NtReqFcb->LastWriteTime.QuadPart; 274 NTFileInfo->LastAccessTime.QuadPart = NtReqFcb->LastAccessTime.QuadPart; 275 NTFileInfo->ChangeTime.QuadPart = NtReqFcb->ChangeTime.QuadPart; 276 // NTFileInfo->AllocationSize.QuadPart = NtReqFcb->CommonFCBHeader.AllocationSize.QuadPart; 277 NTFileInfo->AllocationSize.QuadPart = FileDirNdx->AllocationSize; 278 /* FileDirNdx->FileSize = 279 NTFileInfo->EndOfFile.QuadPart = NtReqFcb->CommonFCBHeader.FileSize.QuadPart;*/ 280 NTFileInfo->EndOfFile.QuadPart = FileDirNdx->FileSize; 281 if(FileDirNdx->FI_Flags & UDF_FI_FLAG_SYS_ATTR) { 282 UDFPrint((" SYS_ATTR\n")); 283 NTFileInfo->FileAttributes = FileDirNdx->SysAttr; 284 goto get_name_only; 285 } 286 FileDirNdx->CreationTime = NTFileInfo->CreationTime.QuadPart; 287 FileDirNdx->LastWriteTime = NTFileInfo->LastWriteTime.QuadPart; 288 FileDirNdx->LastAccessTime = NTFileInfo->LastAccessTime.QuadPart; 289 FileDirNdx->ChangeTime = NTFileInfo->ChangeTime.QuadPart; 290 goto get_attr_only; 291 } 292 ASSERT(FileEntry); 293 } else if(!(FileDirNdx->FI_Flags & UDF_FI_FLAG_SYS_ATTR) || 294 (FileDirNdx->FI_Flags & UDF_FI_FLAG_LINKED)) { 295 LONG_AD feloc; 296 297 UDFPrint((" !SYS_ATTR\n")); 298 FileEntry = (PFILE_ENTRY)MyAllocatePool__(NonPagedPool, Vcb->LBlockSize); 299 if(!FileEntry) return STATUS_INSUFFICIENT_RESOURCES; 300 301 feloc.extLength = Vcb->LBlockSize; 302 feloc.extLocation = FileDirNdx->FileEntryLoc; 303 304 if(!NT_SUCCESS(status = UDFReadFileEntry(Vcb, &feloc, FileEntry, &Ident))) { 305 UDFPrint((" !UDFReadFileEntry\n")); 306 MyFreePool__(FileEntry); 307 FileEntry = NULL; 308 goto get_name_only; 309 } 310 ReadSizes = TRUE; 311 } else { 312 UDFPrint((" FileDirNdx\n")); 313 NTFileInfo->CreationTime.QuadPart = FileDirNdx->CreationTime; 314 NTFileInfo->LastWriteTime.QuadPart = FileDirNdx->LastWriteTime; 315 NTFileInfo->LastAccessTime.QuadPart = FileDirNdx->LastAccessTime; 316 NTFileInfo->ChangeTime.QuadPart = FileDirNdx->ChangeTime; 317 NTFileInfo->FileAttributes = FileDirNdx->SysAttr; 318 NTFileInfo->AllocationSize.QuadPart = FileDirNdx->AllocationSize; 319 NTFileInfo->EndOfFile.QuadPart = FileDirNdx->FileSize; 320 NTFileInfo->EaSize = 0; 321 FileEntry = NULL; 322 goto get_name_only; 323 } 324 325 if(Vcb->VCBFlags & UDF_VCB_FLAGS_RAW_DISK) 326 goto get_name_only; 327 328 UDFPrint((" direct\n")); 329 if(FileEntry->descTag.tagIdent == TID_FILE_ENTRY) { 330 UDFPrint((" TID_FILE_ENTRY\n")); 331 if(ReadSizes) { 332 UDFPrint((" ReadSizes\n")); 333 // Times 334 FileDirNdx->CreationTime = NTFileInfo->CreationTime.QuadPart = 335 FileDirNdx->LastWriteTime = NTFileInfo->LastWriteTime.QuadPart = UDFTimeToNT(&(FileEntry->modificationTime)); 336 FileDirNdx->LastAccessTime = NTFileInfo->LastAccessTime.QuadPart = UDFTimeToNT(&(FileEntry->accessTime)); 337 FileDirNdx->ChangeTime = NTFileInfo->ChangeTime.QuadPart = UDFTimeToNT(&(FileEntry->attrTime)); 338 // FileSize 339 FileDirNdx->FileSize = 340 NTFileInfo->EndOfFile.QuadPart = 341 FileEntry->informationLength; 342 UDFPrint((" informationLength=%I64x, lengthAllocDescs=%I64x\n", 343 FileEntry->informationLength, 344 FileEntry->lengthAllocDescs 345 )); 346 // AllocSize 347 FileDirNdx->AllocationSize = 348 NTFileInfo->AllocationSize.QuadPart = 349 (FileEntry->informationLength + Vcb->LBlockSize - 1) & ~((LONGLONG)(Vcb->LBlockSize) - 1); 350 } 351 // NTFileInfo->EaSize = 0;//FileEntry->lengthExtendedAttr; 352 } else if(FileEntry->descTag.tagIdent == TID_EXTENDED_FILE_ENTRY) { 353 ExFileEntry = (PEXTENDED_FILE_ENTRY)FileEntry; 354 UDFPrint((" PEXTENDED_FILE_ENTRY\n")); 355 if(ReadSizes) { 356 UDFPrint((" ReadSizes\n")); 357 // Times 358 FileDirNdx->CreationTime = NTFileInfo->CreationTime.QuadPart = UDFTimeToNT(&(ExFileEntry->createTime)); 359 FileDirNdx->LastWriteTime = NTFileInfo->LastWriteTime.QuadPart = UDFTimeToNT(&(ExFileEntry->modificationTime)); 360 FileDirNdx->LastAccessTime = NTFileInfo->LastAccessTime.QuadPart = UDFTimeToNT(&(ExFileEntry->accessTime)); 361 FileDirNdx->ChangeTime = NTFileInfo->ChangeTime.QuadPart = UDFTimeToNT(&(ExFileEntry->attrTime)); 362 // FileSize 363 FileDirNdx->FileSize = 364 NTFileInfo->EndOfFile.QuadPart = 365 ExFileEntry->informationLength; 366 UDFPrint((" informationLength=%I64x, lengthAllocDescs=%I64x\n", 367 FileEntry->informationLength, 368 FileEntry->lengthAllocDescs 369 )); 370 // AllocSize 371 FileDirNdx->AllocationSize = 372 NTFileInfo->AllocationSize.QuadPart = 373 (ExFileEntry->informationLength + Vcb->LBlockSize - 1) & ~((LONGLONG)(Vcb->LBlockSize) - 1); 374 } 375 // NTFileInfo->EaSize = 0;//ExFileEntry->lengthExtendedAttr; 376 } else { 377 UDFPrint((" ???\n")); 378 goto get_name_only; 379 } 380 381 get_attr_only: 382 383 UDFPrint((" get_attr")); 384 // do some substitutions 385 if(!FileDirNdx->CreationTime) { 386 FileDirNdx->CreationTime = NTFileInfo->CreationTime.QuadPart = Vcb->VolCreationTime; 387 } 388 if(!FileDirNdx->LastAccessTime) { 389 FileDirNdx->LastAccessTime = NTFileInfo->LastAccessTime.QuadPart = FileDirNdx->CreationTime; 390 } 391 if(!FileDirNdx->LastWriteTime) { 392 FileDirNdx->LastWriteTime = NTFileInfo->LastWriteTime.QuadPart = FileDirNdx->CreationTime; 393 } 394 if(!FileDirNdx->ChangeTime) { 395 FileDirNdx->ChangeTime = NTFileInfo->ChangeTime.QuadPart = FileDirNdx->CreationTime; 396 } 397 398 FileDirNdx->SysAttr = 399 NTFileInfo->FileAttributes = UDFAttributesToNT(FileDirNdx, (tag*)FileEntry); 400 FileDirNdx->FI_Flags |= UDF_FI_FLAG_SYS_ATTR; 401 402 get_name_only: 403 // get filename in standard Unicode format 404 UdfName = FileDirNdx->FName; 405 NTFileInfo->FileNameLength = UdfName.Length; 406 RtlCopyMemory((PCHAR)&(NTFileInfo->FileName), (PCHAR)(UdfName.Buffer), UdfName.MaximumLength); 407 if(!(FileDirNdx->FI_Flags & UDF_FI_FLAG_DOS)) { 408 UDFPrint((" !UDF_FI_FLAG_DOS")); 409 UDFDOSName(Vcb, &DosName, &UdfName, 410 (FileDirNdx->FI_Flags & UDF_FI_FLAG_KEEP_NAME) ? TRUE : FALSE); 411 NTFileInfo->ShortNameLength = (UCHAR)DosName.Length; 412 } 413 // report zero EOF & AllocSize for Dirs 414 if(FileDirNdx->FileCharacteristics & FILE_DIRECTORY) { 415 UDFPrint((" FILE_DIRECTORY")); 416 NTFileInfo->AllocationSize.QuadPart = 417 NTFileInfo->EndOfFile.QuadPart = 0; 418 } 419 UDFPrint((" AllocationSize=%I64x, NTFileInfo->EndOfFile=%I64x", NTFileInfo->AllocationSize.QuadPart, NTFileInfo->EndOfFile.QuadPart)); 420 // free tmp buffer (if any) 421 UDFPrint(("\n")); 422 if(FileEntry && !FileDirNdx->FileInfo) 423 MyFreePool__(FileEntry); 424 return STATUS_SUCCESS; 425 } // end UDFFileDirInfoToNT() 426 427 #endif //_CONSOLE 428 429 #ifndef UDF_READ_ONLY_BUILD 430 /* 431 This routine changes xxxTime field(s) in (Ext)FileEntry 432 */ 433 VOID 434 UDFSetFileXTime( 435 IN PUDF_FILE_INFO FileInfo, 436 IN LONGLONG* CrtTime, 437 IN LONGLONG* AccTime, 438 IN LONGLONG* AttrTime, 439 IN LONGLONG* ChgTime 440 ) 441 { 442 USHORT Ident; 443 PDIR_INDEX_ITEM DirNdx; 444 445 ValidateFileInfo(FileInfo); 446 447 FileInfo->Dloc->FE_Flags |= UDF_FE_FLAG_FE_MODIFIED; 448 DirNdx = UDFDirIndex(UDFGetDirIndexByFileInfo(FileInfo), FileInfo->Index); 449 Ident = FileInfo->Dloc->FileEntry->tagIdent; 450 451 if(Ident == TID_FILE_ENTRY) { 452 PFILE_ENTRY fe = (PFILE_ENTRY)(FileInfo->Dloc->FileEntry); 453 454 if(AccTime) { 455 if(DirNdx && *AccTime) DirNdx->LastAccessTime = *AccTime; 456 UDFTimeToUDF(*AccTime, &(fe->accessTime)); 457 } 458 if(AttrTime) { 459 if(DirNdx && *AttrTime) DirNdx->ChangeTime = *AttrTime; 460 UDFTimeToUDF(*AttrTime, &(fe->attrTime)); 461 } 462 if(ChgTime) { 463 if(DirNdx && *ChgTime) DirNdx->CreationTime = 464 DirNdx->LastWriteTime = *ChgTime; 465 UDFTimeToUDF(*ChgTime, &(fe->modificationTime)); 466 } else 467 if(CrtTime) { 468 if(DirNdx && *CrtTime) DirNdx->CreationTime = 469 DirNdx->LastWriteTime = *CrtTime; 470 UDFTimeToUDF(*CrtTime, &(fe->modificationTime)); 471 } 472 473 } else if(Ident == TID_EXTENDED_FILE_ENTRY) { 474 PEXTENDED_FILE_ENTRY fe = (PEXTENDED_FILE_ENTRY)(FileInfo->Dloc->FileEntry); 475 476 if(AccTime) { 477 if(DirNdx && *AccTime) DirNdx->LastAccessTime = *AccTime; 478 UDFTimeToUDF(*AccTime, &(fe->accessTime)); 479 } 480 if(AttrTime) { 481 if(DirNdx && *AttrTime) DirNdx->ChangeTime = *AttrTime; 482 UDFTimeToUDF(*AttrTime, &(fe->attrTime)); 483 } 484 if(ChgTime) { 485 if(DirNdx && *ChgTime) DirNdx->LastWriteTime = *ChgTime; 486 UDFTimeToUDF(*ChgTime, &(fe->modificationTime)); 487 } 488 if(CrtTime) { 489 if(DirNdx && *CrtTime) DirNdx->CreationTime = *CrtTime; 490 UDFTimeToUDF(*CrtTime, &(fe->createTime)); 491 } 492 493 } 494 } // end UDFSetFileXTime() 495 #endif //UDF_READ_ONLY_BUILD 496 497 /* 498 This routine gets xxxTime field(s) in (Ext)FileEntry 499 */ 500 VOID 501 UDFGetFileXTime( 502 IN PUDF_FILE_INFO FileInfo, 503 OUT LONGLONG* CrtTime, 504 OUT LONGLONG* AccTime, 505 OUT LONGLONG* AttrTime, 506 OUT LONGLONG* ChgTime 507 ) 508 { 509 USHORT Ident; 510 511 ValidateFileInfo(FileInfo); 512 513 Ident = FileInfo->Dloc->FileEntry->tagIdent; 514 515 if(Ident == TID_FILE_ENTRY) { 516 PFILE_ENTRY fe = (PFILE_ENTRY)(FileInfo->Dloc->FileEntry); 517 518 if(AccTime) *AccTime = UDFTimeToNT(&(fe->accessTime)); 519 if(AttrTime) *AttrTime = UDFTimeToNT(&(fe->attrTime)); 520 if(ChgTime) *ChgTime = UDFTimeToNT(&(fe->modificationTime)); 521 if(CrtTime) { 522 (*CrtTime) = *ChgTime; 523 } 524 525 } else if(Ident == TID_EXTENDED_FILE_ENTRY) { 526 PEXTENDED_FILE_ENTRY fe = (PEXTENDED_FILE_ENTRY)(FileInfo->Dloc->FileEntry); 527 528 if(AccTime) *AccTime = UDFTimeToNT(&(fe->accessTime)); 529 if(AttrTime) *AttrTime = UDFTimeToNT(&(fe->attrTime)); 530 if(ChgTime) *ChgTime = UDFTimeToNT(&(fe->modificationTime)); 531 if(CrtTime) *CrtTime = UDFTimeToNT(&(fe->createTime)); 532 533 } 534 if(CrtTime) { 535 if(!(*CrtTime)) 536 KeQuerySystemTime((PLARGE_INTEGER)CrtTime); 537 if(AccTime && !(*AccTime)) (*AccTime) = *CrtTime; 538 if(AttrTime && !(*AttrTime)) (*AttrTime) = *CrtTime; 539 if(AccTime && !(*AccTime)) (*AccTime) = *CrtTime; 540 } 541 } // end UDFGetFileXTime() 542 543 VOID 544 UDFNormalizeFileName( 545 IN PUNICODE_STRING FName, 546 IN USHORT valueCRC 547 ) 548 { 549 PWCHAR buffer; 550 USHORT len; 551 552 len = FName->Length/sizeof(WCHAR); 553 buffer = FName->Buffer; 554 555 // check for '', '.' & '..' 556 if(!len) return; 557 if(!buffer[len-1]) { 558 FName->Length-=sizeof(WCHAR); 559 len--; 560 } 561 if(!len) return; 562 if(buffer[0] == UNICODE_PERIOD) { 563 if(len == 1) return; 564 if((buffer[1] == UNICODE_PERIOD) && (len == 2)) return; 565 } 566 567 // check for trailing '.' 568 for(len--;len;len--) { 569 if( ((buffer[len] == UNICODE_PERIOD) || (buffer[len] == UNICODE_SPACE)) ) { 570 FName->Length-=sizeof(WCHAR); 571 buffer[len] = 0; 572 } else 573 break; 574 } 575 } // end UDFNormalizeFileName() 576 577 #ifndef _CONSOLE 578 579 void 580 __fastcall 581 UDFDOSNameOsNative( 582 IN OUT PUNICODE_STRING DosName, 583 IN PUNICODE_STRING UdfName, 584 IN BOOLEAN KeepIntact 585 ) 586 { 587 PWCHAR dosName = DosName->Buffer; 588 PWCHAR udfName = UdfName->Buffer; 589 uint32 udfLen = UdfName->Length / sizeof(WCHAR); 590 GENERATE_NAME_CONTEXT Ctx; 591 592 if(KeepIntact && 593 (udfLen <= 2) && (udfName[0] == UNICODE_PERIOD)) { 594 if((udfLen != 2) || (udfName[1] == UNICODE_PERIOD)) { 595 RtlCopyMemory(dosName, udfName, UdfName->Length); 596 DosName->Length = UdfName->Length; 597 return; 598 } 599 } 600 RtlZeroMemory(&Ctx, sizeof(GENERATE_NAME_CONTEXT)); 601 RtlGenerate8dot3Name(UdfName, FALSE, &Ctx, DosName); 602 603 } // UDFDOSNameOsNative() 604 605 #endif //_CONSOLE 606 607 /*VOID 608 UDFNormalizeFileName( 609 IN PUNICODE_STRING FName, 610 IN USHORT valueCRC 611 ) 612 { 613 WCHAR _newName[UDF_NAME_LEN+5]; 614 PWCHAR newName = (PWCHAR)(&_newName); 615 PWCHAR udfName = FName->Buffer; 616 LONG udfLen = FName->Length >> 1; 617 618 LONG index, newIndex = 0, extIndex = 0, newExtIndex = 0, trailIndex = 0; 619 BOOLEAN needsCRC = FALSE, hasExt = FALSE; 620 WCHAR ext[UDF_EXT_SIZE], current; 621 622 // handle CurrentDir ('.') and ParentDir ('..') cases 623 if((udfLen <= 2) && (udfName[0] == UNICODE_PERIOD)) { 624 if((udfLen != 2) || (udfName[1] == UNICODE_PERIOD)) 625 return; 626 } 627 628 for (index = 0 ; index < udfLen ; index++) { 629 current = udfName[index]; 630 631 // Look for illegal or unprintable characters. 632 if (UDFIsIllegalChar(current) || !UnicodeIsPrint(current)) { 633 needsCRC = TRUE; 634 current = ILLEGAL_CHAR_MARK; 635 // Skip Illegal characters(even spaces), 636 // but not periods. 637 while(index+1 < udfLen && 638 (UDFIsIllegalChar(udfName[index+1]) || 639 !UnicodeIsPrint(udfName[index+1])) && 640 udfName[index+1] != UNICODE_PERIOD) 641 index++; 642 } 643 644 // Record position of extension, if one is found. 645 if ((current == UNICODE_PERIOD) && ((udfLen - index -1) <= UDF_EXT_SIZE)) { 646 if (udfLen == index + 1) { 647 // A trailing period is NOT an extension. 648 hasExt = FALSE; 649 } else { 650 hasExt = TRUE; 651 extIndex = index; 652 newExtIndex = newIndex; 653 } 654 } else if((current != UNICODE_PERIOD) && (current != UNICODE_SPACE)) { 655 trailIndex = index; 656 } 657 658 // if (newIndex < MAXLEN) // tshi is always TRUE for WINNT 659 newName[newIndex] = current; 660 newIndex++; 661 662 // For OS2, 95 & NT, truncate any trailing periods and\or spaces. 663 if (trailIndex != (newIndex - 1)) { 664 newIndex = trailIndex + 1; 665 needsCRC = TRUE; 666 hasExt = FALSE; // Trailing period does not make an extension. 667 } 668 } 669 670 if (needsCRC) { 671 int localExtIndex = 0; 672 if (hasExt) { 673 int maxFilenameLen; 674 // Translate extension, and store it in ext. 675 for(index = 0; index<UDF_EXT_SIZE && extIndex + index +1 < udfLen; index++ ) { 676 current = udfName[extIndex + index + 1]; 677 if (UDFIsIllegalChar(current) //|| !UnicodeIsPrint(current)) { 678 needsCRC = TRUE; 679 // Replace Illegal and non-displayable chars 680 // with underscore. 681 current = ILLEGAL_CHAR_MARK; 682 // Skip any other illegal or non-displayable 683 // characters. 684 while(index + 1 < UDF_EXT_SIZE && 685 (UDFIsIllegalChar(udfName[extIndex + index + 2]) || 686 !UnicodeIsPrint(udfName[extIndex + index + 2])) ) 687 index++; 688 } 689 ext[localExtIndex++] = current; 690 } 691 // Truncate filename to leave room for extension and CRC. 692 maxFilenameLen = ((UDF_NAME_LEN - 4) - localExtIndex - 1); 693 if (newIndex > maxFilenameLen) { 694 newIndex = maxFilenameLen; 695 } else { 696 newIndex = newExtIndex; 697 } 698 } else if (newIndex > UDF_NAME_LEN - 5) { 699 //If no extension, make sure to leave room for CRC. 700 newIndex = UDF_NAME_LEN - 5; 701 } 702 newName[newIndex++] = UNICODE_CRC_MARK; // Add mark for CRC. 703 //Calculate CRC from original filename from FileIdentifier. 704 // valueCRC = UDFUnicodeCksum(fidName, fidNameLen); 705 // / Convert 16-bits of CRC to hex characters. 706 newName[newIndex++] = hexChar[(valueCRC & 0xf000) >> 12]; 707 newName[newIndex++] = hexChar[(valueCRC & 0x0f00) >> 8]; 708 newName[newIndex++] = hexChar[(valueCRC & 0x00f0) >> 4]; 709 newName[newIndex++] = hexChar[(valueCRC & 0x000f)]; 710 // Place a translated extension at end, if found. 711 if (hasExt) { 712 newName[newIndex++] = UNICODE_PERIOD; 713 for (index = 0;index < localExtIndex ;index++ ) { 714 newName[newIndex++] = ext[index]; 715 } 716 } 717 } 718 719 if(FName->Length == (USHORT)newIndex*sizeof(WCHAR)) { 720 RtlCopyMemory(FName->Buffer, newName, newIndex*sizeof(WCHAR)); 721 return; 722 } 723 MyFreePool__(FName->Buffer); 724 FName->Buffer = (PWCHAR)MyAllocatePool__(UDF_FILENAME_MT, (newIndex+1)*sizeof(WCHAR)); 725 if(FName->Buffer) { 726 FName->Buffer[newIndex] = 0; 727 RtlCopyMemory(FName->Buffer, newName, newIndex*sizeof(WCHAR)); 728 } 729 FName->Length = (USHORT)newIndex*sizeof(WCHAR); 730 FName->MaximumLength = (USHORT)(newIndex+1)*sizeof(WCHAR); 731 }*/ 732 733 /*PUDF_FILE_INFO 734 UDFAllocFileInfo( 735 return ExAllocateFromZone(&(UDFGlobalData.FileInfoZoneHeader)); 736 )*/ 737 738 #define STRING_BUFFER_ALIGNMENT (32) 739 #define STRING_BUFFER_ALIGN(sz) (((sz)+STRING_BUFFER_ALIGNMENT)&(~((ULONG)(STRING_BUFFER_ALIGNMENT-1)))) 740 741 NTSTATUS 742 MyAppendUnicodeStringToString_( 743 IN PUNICODE_STRING Str1, 744 IN PUNICODE_STRING Str2 745 #ifdef UDF_TRACK_UNICODE_STR 746 ,IN PCHAR Tag 747 #endif 748 ) 749 { 750 PWCHAR tmp; 751 USHORT i; 752 753 #ifdef UDF_TRACK_UNICODE_STR 754 #define UDF_UNC_STR_TAG Tag 755 #else 756 #define UDF_UNC_STR_TAG "AppUStr" 757 #endif 758 759 tmp = Str1->Buffer; 760 i = Str1->Length + Str2->Length + sizeof(WCHAR); 761 ASSERT(Str1->MaximumLength); 762 if(i > Str1->MaximumLength) { 763 if(!MyReallocPool__((PCHAR)tmp, Str1->MaximumLength, 764 (PCHAR*)&tmp, STRING_BUFFER_ALIGN(i)*2) ) { 765 return STATUS_INSUFFICIENT_RESOURCES; 766 } 767 Str1->MaximumLength = i*2; 768 Str1->Buffer = tmp; 769 } 770 RtlCopyMemory(((PCHAR)tmp)+Str1->Length, Str2->Buffer, Str2->Length); 771 772 /* tmp = (PWCHAR)MyAllocatePoolTag__(NonPagedPool, i = Str1->Length + Str2->Length + sizeof(WCHAR), UDF_UNC_STR_TAG); 773 if(!tmp) 774 return STATUS_INSUFFICIENT_RESOURCES; 775 RtlCopyMemory(tmp, Str1->Buffer, Str1->Length); 776 RtlCopyMemory(((PCHAR)tmp)+Str1->Length, Str2->Buffer, Str2->Length);*/ 777 tmp[(i / sizeof(WCHAR)) - 1] = 0; 778 Str1->Length = i - sizeof(WCHAR); 779 //MyFreePool__(Str1->Buffer); 780 #ifdef UDF_DBG 781 if(Str1->Buffer && (Str1->Length >= 2*sizeof(WCHAR))) { 782 ASSERT((Str1->Buffer[0] != L'\\') || (Str1->Buffer[1] != L'\\')); 783 } 784 #endif // UDF_DBG 785 return STATUS_SUCCESS; 786 787 #undef UDF_UNC_STR_TAG 788 789 } // end MyAppendUnicodeStringToString() 790 791 NTSTATUS 792 MyAppendUnicodeToString_( 793 IN PUNICODE_STRING Str1, 794 IN PCWSTR Str2 795 #ifdef UDF_TRACK_UNICODE_STR 796 ,IN PCHAR Tag 797 #endif 798 ) 799 { 800 PWCHAR tmp; 801 USHORT i; 802 803 #ifdef UDF_TRACK_UNICODE_STR 804 #define UDF_UNC_STR_TAG Tag 805 #else 806 #define UDF_UNC_STR_TAG "AppStr" 807 #endif 808 809 #if defined (_X86_) && defined (_MSC_VER) 810 811 __asm push ebx 812 __asm push esi 813 814 __asm xor ebx,ebx 815 __asm mov esi,Str2 816 Scan_1: 817 __asm cmp [word ptr esi+ebx],0 818 __asm je EO_Scan 819 __asm add ebx,2 820 __asm jmp Scan_1 821 EO_Scan: 822 __asm mov i,bx 823 824 __asm pop esi 825 __asm pop ebx 826 827 #else // NO X86 optimization, use generic C/C++ 828 829 i=0; 830 while(Str2[i]) { 831 i++; 832 } 833 i *= sizeof(WCHAR); 834 835 #endif // _X86_ 836 837 tmp = Str1->Buffer; 838 ASSERT(Str1->MaximumLength); 839 if((Str1->Length+i+sizeof(WCHAR)) > Str1->MaximumLength) { 840 if(!MyReallocPool__((PCHAR)tmp, Str1->MaximumLength, 841 (PCHAR*)&tmp, STRING_BUFFER_ALIGN(i + Str1->Length + sizeof(WCHAR))*2 ) ) { 842 return STATUS_INSUFFICIENT_RESOURCES; 843 } 844 Str1->MaximumLength = STRING_BUFFER_ALIGN(i + sizeof(WCHAR))*2; 845 Str1->Buffer = tmp; 846 } 847 RtlCopyMemory(((PCHAR)tmp)+Str1->Length, Str2, i); 848 i+=Str1->Length; 849 tmp[(i / sizeof(WCHAR))] = 0; 850 Str1->Length = i; 851 #ifdef UDF_DBG 852 /* if(Str1->Buffer && (Str1->Length >= 2*sizeof(WCHAR))) { 853 ASSERT((Str1->Buffer[0] != L'\\') || (Str1->Buffer[1] != L'\\')); 854 }*/ 855 #endif // UDF_DBG 856 return STATUS_SUCCESS; 857 858 #undef UDF_UNC_STR_TAG 859 860 } // end MyAppendUnicodeToString_() 861 862 NTSTATUS 863 MyInitUnicodeString( 864 IN PUNICODE_STRING Str1, 865 IN PCWSTR Str2 866 ) 867 { 868 869 USHORT i; 870 871 #if defined (_X86_) && defined (_MSC_VER) 872 873 __asm push ebx 874 __asm push esi 875 876 __asm xor ebx,ebx 877 __asm mov esi,Str2 878 Scan_1: 879 __asm cmp [word ptr esi+ebx],0 880 __asm je EO_Scan 881 __asm add ebx,2 882 __asm jmp Scan_1 883 EO_Scan: 884 __asm mov i,bx 885 886 __asm pop esi 887 __asm pop ebx 888 889 #else // NO X86 optimization, use generic C/C++ 890 891 i=0; 892 while(Str2[i]) { 893 i++; 894 } 895 i *= sizeof(WCHAR); 896 897 #endif // _X86_ 898 899 Str1->MaximumLength = STRING_BUFFER_ALIGN((Str1->Length = i) + sizeof(WCHAR)); 900 Str1->Buffer = (PWCHAR)MyAllocatePool__(NonPagedPool, Str1->MaximumLength); 901 if(!Str1->Buffer) 902 return STATUS_INSUFFICIENT_RESOURCES; 903 RtlCopyMemory(Str1->Buffer, Str2, i); 904 Str1->Buffer[i/sizeof(WCHAR)] = 0; 905 return STATUS_SUCCESS; 906 907 } // end MyInitUnicodeString() 908 909 NTSTATUS 910 MyCloneUnicodeString( 911 IN PUNICODE_STRING Str1, 912 IN PUNICODE_STRING Str2 913 ) 914 { 915 Str1->MaximumLength = STRING_BUFFER_ALIGN((Str1->Length = Str2->Length) + sizeof(WCHAR)); 916 Str1->Buffer = (PWCHAR)MyAllocatePool__(NonPagedPool, Str1->MaximumLength); 917 if(!Str1->Buffer) 918 return STATUS_INSUFFICIENT_RESOURCES; 919 ASSERT(Str2->Buffer); 920 RtlCopyMemory(Str1->Buffer, Str2->Buffer, Str2->Length); 921 Str1->Buffer[Str1->Length/sizeof(WCHAR)] = 0; 922 return STATUS_SUCCESS; 923 924 } // end MyCloneUnicodeString() 925 926 /* 927 This routine checks do we needn't read something from disk to 928 obtain Attributes & so on 929 */ 930 BOOLEAN 931 UDFIsDirInfoCached( 932 IN PVCB Vcb, 933 IN PUDF_FILE_INFO DirInfo 934 ) 935 { 936 PDIR_INDEX_HDR hDirNdx = DirInfo->Dloc->DirIndex; 937 PDIR_INDEX_ITEM DirNdx; 938 for(uint_di i=2; (DirNdx = UDFDirIndex(hDirNdx,i)); i++) { 939 if(!(DirNdx->FI_Flags & UDF_FI_FLAG_SYS_ATTR) || 940 (DirNdx->FI_Flags & UDF_FI_FLAG_LINKED)) return FALSE; 941 } 942 return TRUE; 943 } // end UDFIsDirInfoCached() 944 945 #ifndef UDF_READ_ONLY_BUILD 946 NTSTATUS 947 UDFDoesOSAllowFileToBeTargetForRename__( 948 IN PUDF_FILE_INFO FileInfo 949 ) 950 { 951 #ifndef _CONSOLE 952 NTSTATUS RC; 953 #endif //_CONSOLE 954 955 if(UDFIsADirectory(FileInfo)) 956 return STATUS_ACCESS_DENIED; 957 if(!FileInfo->ParentFile) 958 return STATUS_ACCESS_DENIED; 959 960 if(UDFAttributesToNT(UDFDirIndex(UDFGetDirIndexByFileInfo(FileInfo),FileInfo->Index), 961 FileInfo->Dloc->FileEntry) & FILE_ATTRIBUTE_READONLY) 962 return STATUS_ACCESS_DENIED; 963 964 if(!FileInfo->Fcb) 965 return STATUS_SUCCESS; 966 #ifndef _CONSOLE 967 RC = UDFCheckAccessRights(NULL, NULL, FileInfo->Fcb, NULL, DELETE, 0); 968 if(!NT_SUCCESS(RC)) 969 return RC; 970 #endif //_CONSOLE 971 if(!FileInfo->Fcb) 972 return STATUS_SUCCESS; 973 // RC = UDFMarkStreamsForDeletion(FileInfo->Fcb->Vcb, FileInfo->Fcb, TRUE); // Delete 974 /* RC = UDFSetDispositionInformation(FileInfo->Fcb, NULL, 975 FileInfo->Fcb->Vcb, NULL, TRUE); 976 if(NT_SUCCESS(RC)) { 977 FileInfo->Fcb->FCBFlags |= UDF_FCB_DELETED; 978 if(UDFGetFileLinkCount(FileInfo) <= 1) { 979 FileInfo->Fcb->NTRequiredFCB->NtReqFCBFlags |= UDF_NTREQ_FCB_DELETED; 980 } 981 } 982 return RC;*/ 983 return STATUS_ACCESS_DENIED; 984 985 } // end UDFDoesOSAllowFileToBeTargetForRename__() 986 987 NTSTATUS 988 UDFDoesOSAllowFileToBeUnlinked__( 989 IN PUDF_FILE_INFO FileInfo 990 ) 991 { 992 PDIR_INDEX_HDR hCurDirNdx; 993 PDIR_INDEX_ITEM CurDirNdx; 994 uint_di i; 995 // IO_STATUS_BLOCK IoStatus; 996 997 ASSERT(FileInfo->Dloc); 998 999 if(!FileInfo->ParentFile) 1000 return STATUS_CANNOT_DELETE; 1001 if(FileInfo->Dloc->SDirInfo) 1002 return STATUS_CANNOT_DELETE; 1003 if(!UDFIsADirectory(FileInfo)) 1004 return STATUS_SUCCESS; 1005 1006 // UDFFlushAFile(FileInfo->Fcb, NULL, &IoStatus, 0); 1007 hCurDirNdx = FileInfo->Dloc->DirIndex; 1008 // check if we can delete all files 1009 for(i=2; (CurDirNdx = UDFDirIndex(hCurDirNdx,i)); i++) { 1010 // try to open Stream 1011 if(CurDirNdx->FileInfo) 1012 return STATUS_CANNOT_DELETE; 1013 } 1014 // return UDFCheckAccessRights(NULL, NULL, FileInfo->Fcb, NULL, DELETE, 0); 1015 return STATUS_SUCCESS; 1016 } // end UDFDoesOSAllowFileToBeUnlinked__() 1017 1018 NTSTATUS 1019 UDFDoesOSAllowFilePretendDeleted__( 1020 IN PUDF_FILE_INFO FileInfo 1021 ) 1022 { 1023 PDIR_INDEX_HDR hDirNdx = UDFGetDirIndexByFileInfo(FileInfo); 1024 if(!hDirNdx) return STATUS_CANNOT_DELETE; 1025 PDIR_INDEX_ITEM DirNdx = UDFDirIndex(hDirNdx, FileInfo->Index); 1026 if(!DirNdx) return STATUS_CANNOT_DELETE; 1027 // we can't hide file that is not marked as deleted 1028 if(!(DirNdx->FileCharacteristics & FILE_DELETED)) { 1029 BrutePoint(); 1030 1031 #ifndef _CONSOLE 1032 if(!(FileInfo->Fcb->FCBFlags & (UDF_FCB_DELETE_ON_CLOSE | 1033 UDF_FCB_DELETED) )) 1034 #endif //_CONSOLE 1035 1036 return STATUS_CANNOT_DELETE; 1037 } 1038 return STATUS_SUCCESS; 1039 } 1040 #endif //UDF_READ_ONLY_BUILD 1041 1042