reactos/drivers/filesystems/udfs/Include/phys_lib.cpp
2021-06-11 15:33:08 +03:00

4385 lines
165 KiB
C++

////////////////////////////////////////////////////////////////////
// Copyright (C) Alexander Telyatnikov, Ivan Keliukh, Yegor Anchishkin, SKIF Software, 1999-2013. Kiev, Ukraine
// All rights reserved
// This file was released under the GPLv2 on June 2015.
////////////////////////////////////////////////////////////////////
/*
Module Name: Phys_lib.cpp
Execution: Kernel mode only
Description:
Contains code that implement read/write operations for physical device
*/
#include "phys_lib.h"
static const char Signature [16] = {CDRW_SIGNATURE_v1};
// Local functions:
OSSTATUS
UDFSetSpeeds(
IN PVCB Vcb
);
NTSTATUS
UDFSetCaching(
IN PVCB Vcb
);
OSSTATUS
UDFRecoverFromError(
IN PVCB Vcb,
IN BOOLEAN WriteOp,
IN OSSTATUS status,
IN uint32 Lba,
IN uint32 BCount,
IN OUT uint32* retry);
#ifdef _BROWSE_UDF_
uint32
UDFFixFPAddress(
IN PVCB Vcb, // Volume control block from this DevObj
IN uint32 Lba
);
#endif //_BROWSE_UDF_
NTSTATUS
UDFSyncCache(
IN PVCB Vcb
)
{
UDFPrint(("UDFSyncCache:\n"));
OSSTATUS RC;
RC = UDFPhSendIOCTL( IOCTL_CDRW_SYNC_CACHE, Vcb->TargetDeviceObject,
NULL,0, NULL,0, FALSE, NULL);
if(OS_SUCCESS(RC)) {
// clear LAST_WRITE flag
Vcb->VCBFlags &= ~UDF_VCB_LAST_WRITE;
}
return RC;
} // end UDFSyncCache()
OSSTATUS
UDFReallocTrackMap(
IN PVCB Vcb,
IN uint32 TrackNum
)
{
#ifdef _BROWSE_UDF_
if(Vcb->TrackMap) {
MyFreePool__(Vcb->TrackMap);
Vcb->TrackMap = NULL;
}
Vcb->TrackMap = (PUDFTrackMap)
MyAllocatePool__(NonPagedPool, TrackNum*sizeof(UDFTrackMap));
if(!Vcb->TrackMap) {
return STATUS_INSUFFICIENT_RESOURCES;
}
#endif //_BROWSE_UDF_
RtlZeroMemory(Vcb->TrackMap,TrackNum*sizeof(UDFTrackMap));
return STATUS_SUCCESS;
} // end UDFReallocTrackMap()
#ifdef _BROWSE_UDF_
OSSTATUS
__fastcall
UDFTIOVerify(
IN void* _Vcb,
IN void* Buffer, // Target buffer
IN SIZE_T Length,
IN uint32 LBA,
OUT PSIZE_T IOBytes,
IN uint32 Flags
)
{
OSSTATUS RC = STATUS_SUCCESS;
uint32 i, j;
SIZE_T mask;
uint32 lba0, len, lba1;
PUCHAR tmp_buff;
PUCHAR p;
PCHAR cached_block;
SIZE_T tmp_wb;
BOOLEAN need_remap;
OSSTATUS final_RC = STATUS_SUCCESS;
BOOLEAN zero;
BOOLEAN non_zero;
BOOLEAN packet_ok;
BOOLEAN free_tmp = FALSE;
BOOLEAN single_packet = FALSE;
#define Vcb ((PVCB)_Vcb)
// ATTENTION! Do not touch bad block bitmap here, since it describes PHYSICAL addresses WITHOUT remapping,
// while here we work with LOGICAL addresses
if(Vcb->VerifyCtx.ItemCount > UDF_MAX_VERIFY_CACHE) {
UDFVVerify(Vcb, 0/*UFD_VERIFY_FLAG_WAIT*/);
}
UDFAcquireResourceExclusive(&(Vcb->IoResource), TRUE);
Flags |= PH_IO_LOCKED;
tmp_wb = (SIZE_T)_Vcb;
if(Flags & PH_EX_WRITE) {
UDFPrint(("IO-Write-Verify\n"));
RC = UDFTWrite(_Vcb, Buffer, Length, LBA, &tmp_wb, Flags | PH_VCB_IN_RETLEN);
} else {
UDFPrint(("IO-Read-Verify\n"));
RC = UDFTRead(_Vcb, Buffer, Length, LBA, &tmp_wb, Flags | PH_VCB_IN_RETLEN);
}
(*IOBytes) = tmp_wb;
switch(RC) {
default:
UDFReleaseResource(&(Vcb->IoResource));
return RC;
case STATUS_FT_WRITE_RECOVERY:
case STATUS_DEVICE_DATA_ERROR:
case STATUS_IO_DEVICE_ERROR:
break;
/* FALL THROUGH */
} // end switch(RC)
if(!Vcb->SparingCount ||
!Vcb->SparingCountFree ||
Vcb->CDR_Mode) {
UDFPrint(("Can't remap\n"));
UDFReleaseResource(&(Vcb->IoResource));
return RC;
}
if(Flags & PH_EX_WRITE) {
UDFPrint(("Write failed, try relocation\n"));
} else {
if(Vcb->Modified) {
UDFPrint(("Read failed, try relocation\n"));
} else {
UDFPrint(("no remap on not modified volume\n"));
UDFReleaseResource(&(Vcb->IoResource));
return RC;
}
}
if(Flags & PH_LOCK_CACHE) {
UDFReleaseResource(&(Vcb->IoResource));
WCacheStartDirect__(&(Vcb->FastCache), Vcb, TRUE);
UDFAcquireResourceExclusive(&(Vcb->IoResource), TRUE);
}
Flags &= ~PH_KEEP_VERIFY_CACHE;
// NOTE: SparingBlockSize may be not equal to PacketSize
// perform recovery
mask = Vcb->SparingBlockSize-1;
lba0 = LBA & ~mask;
len = ((LBA+(Length>>Vcb->BlockSizeBits)+mask) & ~mask) - lba0;
j=0;
if((lba0 == LBA) && (len == mask+1) && (len == (Length>>Vcb->BlockSizeBits))) {
single_packet = TRUE;
tmp_buff = NULL;
} else {
tmp_buff = (PUCHAR)DbgAllocatePoolWithTag(NonPagedPool, Vcb->SparingBlockSize << Vcb->BlockSizeBits, 'bNWD');
if(!tmp_buff) {
UDFPrint((" can't alloc tmp\n"));
UDFReleaseResource(&(Vcb->IoResource));
return STATUS_DEVICE_DATA_ERROR;
}
free_tmp = TRUE;
}
for(i=0; i<len; i++) {
if(!Vcb->SparingCountFree) {
UDFPrint((" no more free spare blocks, abort verification\n"));
break;
}
UDFPrint((" read LBA %x (%x)\n", lba0+i, j));
if(!j) {
need_remap = FALSE;
lba1 = lba0+i;
non_zero = FALSE;
if(single_packet) {
// single packet requested
tmp_buff = (PUCHAR)Buffer;
if(Flags & PH_EX_WRITE) {
UDFPrint((" remap single write\n"));
UDFPrint((" try del from verify cache @ %x, %x\n", lba0, len));
UDFVForget(Vcb, len, UDFRelocateSector(Vcb, lba0), 0);
goto do_remap;
} else {
UDFPrint((" recover and remap single read\n"));
}
}
}
p = tmp_buff+(j<<Vcb->BlockSizeBits);
// not cached, try to read
// prepare for error, if block cannot be read, assume it is zero-filled
RtlZeroMemory(p, Vcb->BlockSize);
// check if block valid
if(Vcb->BSBM_Bitmap) {
if(UDFGetBit((uint32*)(Vcb->BSBM_Bitmap), UDFRelocateSector(Vcb, lba0+i))) {
UDFPrint((" remap: known BB @ %x, mapped to %x\n", lba0+i, UDFRelocateSector(Vcb, lba0+i)));
need_remap = TRUE;
}
}
zero = FALSE;
if(Vcb->FSBM_Bitmap) {
if(UDFGetFreeBit((uint32*)(Vcb->FSBM_Bitmap), lba0+i)) {
UDFPrint((" unused @ %x\n", lba0+i));
zero = TRUE;
}
}
if(!zero && Vcb->ZSBM_Bitmap) {
if(UDFGetZeroBit((uint32*)(Vcb->ZSBM_Bitmap), lba0+i)) {
UDFPrint((" unused @ %x (Z)\n", lba0+i));
zero = TRUE;
}
}
non_zero |= !zero;
if(!j) {
packet_ok = FALSE;
if(!single_packet) {
// try to read entire packet, this returs error more often then sequential reading of all blocks one by one
tmp_wb = (SIZE_T)_Vcb;
RC = UDFTRead(_Vcb, p, Vcb->SparingBlockSize << Vcb->BlockSizeBits, lba0+i, &tmp_wb,
Flags | PH_READ_VERIFY_CACHE | PH_TMP_BUFFER | PH_VCB_IN_RETLEN);
} else {
// Note: we get here ONLY if original request failed
// do not retry if it was single-packet request
RC = STATUS_UNSUCCESSFUL;
}
if(RC == STATUS_SUCCESS) {
UDFPrint((" packet ok @ %x\n", lba0+i));
packet_ok = TRUE;
i += Vcb->SparingBlockSize-1;
continue;
} else {
need_remap = TRUE;
}
}
if(!zero) {
if(WCacheIsCached__(&(Vcb->FastCache), lba0+i, 1)) {
// even if block is cached, we have to verify if it is readable
if(!packet_ok && !UDFVIsStored(Vcb, lba0+i)) {
tmp_wb = (SIZE_T)_Vcb;
RC = UDFTRead(_Vcb, p, Vcb->BlockSize, lba0+i, &tmp_wb,
Flags | PH_FORGET_VERIFIED | PH_READ_VERIFY_CACHE | PH_TMP_BUFFER | PH_VCB_IN_RETLEN);
if(!OS_SUCCESS(RC)) {
UDFPrint((" Found BB @ %x\n", lba0+i));
}
}
RC = WCacheDirect__(&(Vcb->FastCache), _Vcb, lba0+i, FALSE, &cached_block, TRUE/* cached only */);
} else {
cached_block = NULL;
if(!packet_ok) {
RC = STATUS_UNSUCCESSFUL;
} else {
RC = STATUS_SUCCESS;
}
}
if(OS_SUCCESS(RC)) {
// cached or successfully read
if(cached_block) {
// we can get from cache the most fresh data
RtlCopyMemory(p, cached_block, Vcb->BlockSize);
}
} else {
if(!UDFVIsStored(Vcb, lba0+i)) {
tmp_wb = (SIZE_T)_Vcb;
RC = UDFTRead(_Vcb, p, Vcb->BlockSize, lba0+i, &tmp_wb,
Flags | PH_FORGET_VERIFIED | PH_READ_VERIFY_CACHE | PH_TMP_BUFFER | PH_VCB_IN_RETLEN);
} else {
// get it from verify-cache
RC = STATUS_UNSUCCESSFUL;
}
if(!OS_SUCCESS(RC)) {
/*
UDFPrint((" retry @ %x\n", lba0+i));
tmp_wb = (uint32)_Vcb;
RC = UDFTRead(_Vcb, p, Vcb->BlockSize, lba0+i, &tmp_wb,
Flags | PH_FORGET_VERIFIED | PH_READ_VERIFY_CACHE | PH_TMP_BUFFER | PH_VCB_IN_RETLEN);
*/
UDFPrint((" try get from verify cache @ %x\n", lba0+i));
RC = UDFVRead(Vcb, p, 1, UDFRelocateSector(Vcb, lba0+i),
Flags | PH_FORGET_VERIFIED | PH_READ_VERIFY_CACHE | PH_TMP_BUFFER);
need_remap = TRUE;
}
}
} else {
RtlZeroMemory(p, Vcb->BlockSize);
}
if(!packet_ok) {
UDFPrint((" try del from verify cache @ %x\n", lba0+i));
RC = UDFVForget(Vcb, 1, UDFRelocateSector(Vcb, lba0+i), 0);
}
if(!packet_ok || need_remap) {
UDFPrint((" block in bad packet @ %x\n", lba0+i));
if(Vcb->BSBM_Bitmap) {
UDFSetBit(Vcb->BSBM_Bitmap, lba0+i);
}
if(Vcb->FSBM_Bitmap) {
UDFSetUsedBit(Vcb->FSBM_Bitmap, lba0+i);
}
}
j++;
if(j >= Vcb->SparingBlockSize) {
// remap this packet
if(need_remap) {
ASSERT(!packet_ok);
if(!non_zero) {
UDFPrint((" forget Z packet @ %x\n", lba1));
UDFUnmapRange(Vcb, lba1, Vcb->SparingBlockSize);
RC = STATUS_SUCCESS;
} else {
do_remap:
for(j=0; j<3; j++) {
UDFPrint((" remap packet @ %x\n", lba1));
RC = UDFRemapPacket(Vcb, lba1, FALSE);
if(!OS_SUCCESS(RC)) {
if(RC == STATUS_SHARING_VIOLATION) {
UDFPrint((" remap2\n"));
// remapped location have died
RC = UDFRemapPacket(Vcb, lba1, TRUE);
}
if(!OS_SUCCESS(RC)) {
// packet cannot be remapped :(
RC = STATUS_DEVICE_DATA_ERROR;
}
}
UDFPrint((" remap status %x\n", RC));
if(OS_SUCCESS(RC)) {
// write to remapped area
tmp_wb = (SIZE_T)_Vcb;
RC = UDFTWrite(_Vcb, tmp_buff, Vcb->SparingBlockSize << Vcb->BlockSizeBits, lba1, &tmp_wb,
Flags | PH_FORGET_VERIFIED | PH_READ_VERIFY_CACHE | PH_TMP_BUFFER | PH_VCB_IN_RETLEN);
UDFPrint((" write status %x\n", RC));
if(RC != STATUS_SUCCESS) {
// will be remapped
UDFPrint((" retry remap\n"));
// Note: when remap of already remapped block is requested, verify of
// entire sparing are will be performed.
} else {
UDFPrint((" remap OK\n"));
break;
}
} else {
UDFPrint((" failed remap\n"));
break;
}
} // for
}
if(!OS_SUCCESS(RC) && !OS_SUCCESS(final_RC)) {
final_RC = RC;
}
} else {
UDFPrint((" NO remap for @ %x\n", (lba0+i) & ~mask));
}
j=0;
}
}
if(free_tmp) {
DbgFreePool(tmp_buff);
}
tmp_wb = (SIZE_T)_Vcb;
if(Flags & PH_EX_WRITE) {
UDFPrint(("IO-Write-Verify (2)\n"));
//RC = UDFTWrite(_Vcb, Buffer, Length, LBA, &tmp_wb, Flags | PH_FORGET_VERIFIED | PH_VCB_IN_RETLEN);
} else {
UDFPrint(("IO-Read-Verify (2)\n"));
RC = UDFTRead(_Vcb, Buffer, Length, LBA, &tmp_wb, Flags | PH_FORGET_VERIFIED | PH_VCB_IN_RETLEN);
}
(*IOBytes) = tmp_wb;
UDFPrint(("Final %x\n", RC));
UDFReleaseResource(&(Vcb->IoResource));
if(Flags & PH_LOCK_CACHE) {
WCacheEODirect__(&(Vcb->FastCache), Vcb);
}
return RC;
} // end UDFTIOVerify()
OSSTATUS
UDFTWriteVerify(
IN void* _Vcb,
IN void* Buffer, // Target buffer
IN SIZE_T Length,
IN uint32 LBA,
OUT PSIZE_T WrittenBytes,
IN uint32 Flags
)
{
return UDFTIOVerify(_Vcb, Buffer, Length, LBA, WrittenBytes, Flags | PH_VCB_IN_RETLEN | PH_EX_WRITE | PH_KEEP_VERIFY_CACHE);
} // end UDFTWriteVerify()
OSSTATUS
UDFTReadVerify(
IN void* _Vcb,
IN void* Buffer, // Target buffer
IN SIZE_T Length,
IN uint32 LBA,
OUT PSIZE_T ReadBytes,
IN uint32 Flags
)
{
return UDFTIOVerify(_Vcb, Buffer, Length, LBA, ReadBytes, Flags | PH_VCB_IN_RETLEN | PH_KEEP_VERIFY_CACHE);
} // end UDFTReadVerify()
#endif //_BROWSE_UDF_
/*
This routine performs low-level write
ATTENTION! When we are in Variable-Packet mode (CDR_Mode = TRUE)
LBA is ignored and assumed to be equal to NWA by CD-R(W) driver
*/
OSSTATUS
UDFTWrite(
IN void* _Vcb,
IN void* Buffer, // Target buffer
IN SIZE_T Length,
IN uint32 LBA,
OUT PSIZE_T WrittenBytes,
IN uint32 Flags
)
{
#ifndef UDF_READ_ONLY_BUILD
#define Vcb ((PVCB)_Vcb)
#ifdef _BROWSE_UDF_
PEXTENT_MAP RelocExtent;
PEXTENT_MAP RelocExtent_saved = NULL;
#endif //_BROWSE_UDF_
uint32 retry;
BOOLEAN res_acq = FALSE;
OSSTATUS RC = STATUS_SUCCESS;
uint32 rLba;
uint32 BCount;
uint32 i;
#ifdef DBG
//ASSERT(!(LBA & (32-1)));
#endif //DBG
(*WrittenBytes) = 0;
BCount = Length>>Vcb->BlockSizeBits;
UDFPrint(("TWrite %x (%x)\n", LBA, BCount));
#ifdef _BROWSE_UDF_
if(Vcb->VCBFlags & UDF_VCB_FLAGS_DEAD) {
UDFPrint(("DEAD\n"));
return STATUS_NO_SUCH_DEVICE;
}
Vcb->VCBFlags |= (UDF_VCB_SKIP_EJECT_CHECK | UDF_VCB_LAST_WRITE);
if(!Vcb->CDR_Mode) {
RelocExtent = UDFRelocateSectors(Vcb, LBA, BCount);
if(!RelocExtent) {
UDFPrint(("can't relocate\n"));
return STATUS_INSUFFICIENT_RESOURCES;
}
rLba = LBA;
} else {
RelocExtent = UDF_NO_EXTENT_MAP;
rLba = Vcb->NWA;
}
#else //_BROWSE_UDF_
rLba = LBA;
#endif //_BROWSE_UDF_
#ifdef DBG
//ASSERT(!(rLba & (32-1)));
#endif //DBG
_SEH2_TRY {
#ifdef _BROWSE_UDF_
if(!(Flags & PH_IO_LOCKED)) {
UDFAcquireResourceExclusive(&(Vcb->IoResource), TRUE);
res_acq = TRUE;
}
if(RelocExtent == UDF_NO_EXTENT_MAP) {
#endif //_BROWSE_UDF_
retry = UDF_WRITE_MAX_RETRY;
retry_1:
RC = UDFPrepareForWriteOperation(Vcb, rLba, BCount);
if(!OS_SUCCESS(RC)) {
UDFPrint(("prepare failed\n"));
try_return(RC);
}
if(Flags & PH_VCB_IN_RETLEN) {
(*WrittenBytes) = (ULONG_PTR)Vcb;
}
RC = UDFPhWriteVerifySynchronous(Vcb->TargetDeviceObject, Buffer, Length,
((uint64)rLba) << Vcb->BlockSizeBits, WrittenBytes, Flags);
#ifdef _BROWSE_UDF_
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
#endif //_BROWSE_UDF_
if(!OS_SUCCESS(RC) &&
OS_SUCCESS(RC = UDFRecoverFromError(Vcb, TRUE, RC, rLba, BCount, &retry)) )
goto retry_1;
UDFUpdateNWA((PVCB)_Vcb, rLba, BCount, RC);
try_return(RC);
#ifdef _BROWSE_UDF_
}
// write according to relocation table
RelocExtent_saved = RelocExtent;
for(i=0; RelocExtent->extLength; i++, RelocExtent++) {
SIZE_T _WrittenBytes;
rLba = RelocExtent->extLocation;
BCount = RelocExtent->extLength>>Vcb->BlockSizeBits;
retry = UDF_WRITE_MAX_RETRY;
retry_2:
RC = UDFPrepareForWriteOperation(Vcb, rLba, BCount);
if(!OS_SUCCESS(RC)) {
UDFPrint(("prepare failed (2)\n"));
break;
}
if(Flags & PH_VCB_IN_RETLEN) {
_WrittenBytes = (ULONG_PTR)Vcb;
}
RC = UDFPhWriteVerifySynchronous(Vcb->TargetDeviceObject, Buffer, RelocExtent->extLength,
((uint64)rLba) << Vcb->BlockSizeBits, &_WrittenBytes, Flags);
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
if(!OS_SUCCESS(RC) &&
OS_SUCCESS(RC = UDFRecoverFromError(Vcb, TRUE, RC, rLba, BCount, &retry)) )
goto retry_2;
UDFUpdateNWA((PVCB)_Vcb, rLba, BCount, RC);
LBA += BCount;
(*WrittenBytes) += _WrittenBytes;
if(!OS_SUCCESS(RC)) break;
*((uint32*)&Buffer) += RelocExtent->extLength;
}
#endif //_BROWSE_UDF_
try_exit: NOTHING;
} _SEH2_FINALLY {
if(res_acq) {
UDFReleaseResource(&(Vcb->IoResource));
}
#ifdef _BROWSE_UDF_
if(RelocExtent_saved) {
MyFreePool__(RelocExtent_saved);
}
#endif //_BROWSE_UDF_
} _SEH2_END;
UDFPrint(("TWrite: %x\n", RC));
return RC;
#undef Vcb
#else //UDF_READ_ONLY_BUILD
return STATUS_ACCESS_DENIED;
#endif //UDF_READ_ONLY_BUILD
} // end UDFTWrite()
/*
This routine performs low-level read
*/
OSSTATUS
UDFTRead(
IN void* _Vcb,
IN void* Buffer, // Target buffer
IN SIZE_T Length,
IN uint32 LBA,
OUT PSIZE_T ReadBytes,
IN uint32 Flags
)
{
uint32 rLba;
OSSTATUS RC = STATUS_SUCCESS;
uint32 retry;
PVCB Vcb = (PVCB)_Vcb;
uint32 BCount = Length >> Vcb->BlockSizeBits;
uint32 i;
#ifdef _BROWSE_UDF_
PEXTENT_MAP RelocExtent;
PEXTENT_MAP RelocExtent_saved = NULL;
BOOLEAN res_acq = FALSE;
// LARGE_INTEGER delay;
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
ASSERT(Buffer);
(*ReadBytes) = 0;
if(Vcb->VCBFlags & UDF_VCB_FLAGS_DEAD)
return STATUS_NO_SUCH_DEVICE;
RelocExtent = UDFRelocateSectors(Vcb, LBA, BCount);
if(!RelocExtent) return STATUS_INSUFFICIENT_RESOURCES;
_SEH2_TRY {
if(!(Flags & PH_IO_LOCKED)) {
UDFAcquireResourceExclusive(&(Vcb->IoResource), TRUE);
res_acq = TRUE;
}
if(RelocExtent == UDF_NO_EXTENT_MAP) {
rLba = LBA;
if(rLba >= (Vcb->CDR_Mode ? Vcb->NWA : Vcb->LastLBA + 1)) {
RtlZeroMemory(Buffer, Length);
try_return(RC = STATUS_SUCCESS);
}
retry = UDF_WRITE_MAX_RETRY;
retry_1:
RC = UDFPrepareForReadOperation(Vcb, rLba, Length >> Vcb->BlockSizeBits);
if(!OS_SUCCESS(RC)) try_return(RC);
rLba = UDFFixFPAddress(Vcb, rLba);
#else
rLba = LBA;
retry = UDF_WRITE_MAX_RETRY;
retry_1:
RC = UDFPrepareForReadOperation(Vcb, rLba, Length >> Vcb->BlockSizeBits);
if(!OS_SUCCESS(RC)) return RC; // this is for !_BROWSE_UDF only
#endif //_BROWSE_UDF_
if(Flags & PH_VCB_IN_RETLEN) {
(*ReadBytes) = (SIZE_T)Vcb;
}
RC = UDFPhReadSynchronous(Vcb->TargetDeviceObject, Buffer, Length,
((uint64)rLba) << Vcb->BlockSizeBits, ReadBytes, Flags);
Vcb->VCBFlags &= ~UDF_VCB_LAST_WRITE;
#ifdef _BROWSE_UDF_
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
#endif //_BROWSE_UDF_
if(!OS_SUCCESS(RC) &&
OS_SUCCESS(RC = UDFRecoverFromError(Vcb, FALSE, RC, rLba, BCount, &retry)) ) {
if(RC != STATUS_BUFFER_ALL_ZEROS) {
goto retry_1;
}
RtlZeroMemory(Buffer, Length);
(*ReadBytes) = Length;
RC = STATUS_SUCCESS;
}
#ifdef _BROWSE_UDF_
try_return(RC);
}
// read according to relocation table
RelocExtent_saved = RelocExtent;
for(i=0; RelocExtent->extLength; i++, RelocExtent++) {
SIZE_T _ReadBytes;
rLba = RelocExtent->extLocation;
if(rLba >= (Vcb->CDR_Mode ? Vcb->NWA : Vcb->LastLBA + 1)) {
RtlZeroMemory(Buffer, _ReadBytes = RelocExtent->extLength);
RC = STATUS_SUCCESS;
goto TR_continue;
}
BCount = RelocExtent->extLength>>Vcb->BlockSizeBits;
retry = UDF_WRITE_MAX_RETRY;
retry_2:
RC = UDFPrepareForReadOperation(Vcb, rLba, RelocExtent->extLength >> Vcb->BlockSizeBits);
if(!OS_SUCCESS(RC)) break;
rLba = UDFFixFPAddress(Vcb, rLba);
if(Flags & PH_VCB_IN_RETLEN) {
_ReadBytes = (SIZE_T)Vcb;
}
RC = UDFPhReadSynchronous(Vcb->TargetDeviceObject, Buffer, RelocExtent->extLength,
((uint64)rLba) << Vcb->BlockSizeBits, &_ReadBytes, Flags);
Vcb->VCBFlags &= ~UDF_VCB_LAST_WRITE;
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
if(!OS_SUCCESS(RC) &&
OS_SUCCESS(RC = UDFRecoverFromError(Vcb, FALSE, RC, rLba, BCount, &retry)) ) {
if(RC != STATUS_BUFFER_ALL_ZEROS) {
goto retry_2;
}
RtlZeroMemory(Buffer, RelocExtent->extLength);
_ReadBytes = RelocExtent->extLength;
RC = STATUS_SUCCESS;
}
TR_continue:
(*ReadBytes) += _ReadBytes;
if(!OS_SUCCESS(RC)) break;
*((uint32*)&Buffer) += RelocExtent->extLength;
}
try_exit: NOTHING;
} _SEH2_FINALLY {
if(res_acq) {
UDFReleaseResource(&(Vcb->IoResource));
}
if(RelocExtent_saved) {
MyFreePool__(RelocExtent_saved);
}
} _SEH2_END;
#endif //_BROWSE_UDF_
return RC;
} // end UDFTRead()
#ifdef UDF_ASYNC_IO
/*
This routine performs asynchronous low-level read
Is not used now.
*/
OSSTATUS
UDFTReadAsync(
IN void* _Vcb,
IN void* _WContext,
IN void* Buffer, // Target buffer
IN SIZE_T Length,
IN uint32 LBA,
OUT PSIZE_T ReadBytes
)
{
PEXTENT_MAP RelocExtent;
PEXTENT_MAP RelocExtent_saved;
OSSTATUS RC = STATUS_SUCCESS;
// LARGE_INTEGER delay;
uint32 retry = UDF_READ_MAX_RETRY;
PVCB Vcb = (PVCB)_Vcb;
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
uint32 rLba;
uint32 BCount;
ASSERT(Buffer);
(*ReadBytes) = 0;
RelocExtent = UDFRelocateSectors(Vcb, LBA, BCount = Length >> Vcb->BlockSizeBits);
if(!RelocExtent) return STATUS_INSUFFICIENT_RESOURCES;
if(RelocExtent == UDF_NO_EXTENT_MAP) {
rLba = LBA;
if(rLba >= (Vcb->CDR_Mode ? Vcb->NWA : Vcb->LastLBA + 1)) {
RtlZeroMemory(Buffer, Length);
return STATUS_SUCCESS;
}
retry_1:
RC = UDFPrepareForReadOperation(Vcb, rLba, BCount);
if(!OS_SUCCESS(RC)) return RC;
rLba = UDFFixFPAddress(Vcb, rLba);
RC = UDFPhReadSynchronous(Vcb->TargetDeviceObject, Buffer, Length,
((uint64)rLba) << Vcb->BlockSizeBits, ReadBytes, 0);
Vcb->VCBFlags &= ~UDF_VCB_LAST_WRITE;
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
if(!OS_SUCCESS(RC) &&
OS_SUCCESS(RC = UDFRecoverFromError(Vcb, FALSE, RC, rLba, BCount, &retry)) )
goto retry_1;
return RC;
}
// read according to relocation table
RelocExtent_saved = RelocExtent;
for(uint32 i=0; RelocExtent->extLength; i++, RelocExtent++) {
SIZE_T _ReadBytes;
rLba = RelocExtent->extLocation;
if(rLba >= (Vcb->CDR_Mode ? Vcb->NWA : Vcb->LastLBA + 1)) {
RtlZeroMemory(Buffer, _ReadBytes = RelocExtent->extLength);
RC = STATUS_SUCCESS;
goto TR_continue;
}
BCount = RelocExtent->extLength>>Vcb->BlockSizeBits;
retry_2:
RC = UDFPrepareForReadOperation(Vcb, rLba, RelocExtent->extLength >> Vcb->BlockSizeBits);
if(!OS_SUCCESS(RC)) break;
rLba = UDFFixFPAddress(Vcb, rLba);
RC = UDFPhReadSynchronous(Vcb->TargetDeviceObject, Buffer, RelocExtent->extLength,
((uint64)rLba) << Vcb->BlockSizeBits, &_ReadBytes, 0);
Vcb->VCBFlags &= ~UDF_VCB_LAST_WRITE;
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
if(!OS_SUCCESS(RC) &&
OS_SUCCESS(RC = UDFRecoverFromError(Vcb, FALSE, RC, rLba, BCount, &retry)) )
goto retry_2;
TR_continue:
(*ReadBytes) += _ReadBytes;
if(!OS_SUCCESS(RC)) break;
*((uint32*)&Buffer) += RelocExtent->extLength;
}
MyFreePool__(RelocExtent_saved);
return RC;
} // end UDFTReadAsync()
#endif //UDF_ASYNC_IO
/*
*/
NTSTATUS
UDFSetMRWMode(
IN PVCB Vcb
)
{
GET_MRW_MODE_USER_OUT MRWPage;
OSSTATUS RC;
if(Vcb->MediaClassEx != CdMediaClass_CDRW)
return STATUS_SUCCESS;
//#ifdef _BROWSE_UDF_
if(Vcb->CompatFlags & UDF_VCB_IC_MRW_ADDR_PROBLEM)
return STATUS_SUCCESS;
//#endif //_BROWSE_UDF_
if(!Vcb->MRWStatus) {
UDFPrint(("Non-MRW disk. Skip setting MRW_MODE\n"));
return STATUS_SUCCESS;
}
UDFPrint(("try set MRW_MODE\n"));
RC = UDFPhSendIOCTL(IOCTL_CDRW_GET_MRW_MODE, Vcb->TargetDeviceObject,
NULL,0,
(PVOID)&MRWPage,sizeof(MRWPage),
FALSE, NULL);
if(!NT_SUCCESS(RC)) {
return RC;
}
UDFPrint(("GET_MRW_MODE ok (current %x)\n", MRWPage.AddressMode));
MRWPage.AddressMode = Vcb->MRWStatus ? 0 : MrwPage_use_GAA;
UDFPrint(("SET_MRW_MODE %x\n", MRWPage.AddressMode));
RC = UDFPhSendIOCTL(IOCTL_CDRW_SET_MRW_MODE, Vcb->TargetDeviceObject,
(PVOID)&MRWPage,sizeof(MRWPage),
NULL,0,
FALSE, NULL);
UDFPrint(("SET_MRW_MODE status %x\n", RC));
return STATUS_SUCCESS;
} // end UDFSetMRWMode()
OSSTATUS
UDFDoOPC(
IN PVCB Vcb
)
{
OSSTATUS RC;
if(Vcb->OPCNum && !Vcb->OPCDone) {
UDFPrint(("UDFDoOPC\n"));
if(!Vcb->OPCh) {
Vcb->OPCh =
(PSEND_OPC_INFO_HEADER_USER_IN)MyAllocatePool__(NonPagedPool,
sizeof(SEND_OPC_INFO_HEADER_USER_IN) );
}
if(!Vcb->OPCh)
return STATUS_INSUFFICIENT_RESOURCES;
Vcb->OPCh->DoOpc = TRUE;
Vcb->OPCh->OpcBlocksNumber = 0;
RC = UDFPhSendIOCTL(IOCTL_CDRW_SEND_OPC_INFO, Vcb->TargetDeviceObject,
(void*)(Vcb->OPCh),sizeof(SEND_OPC_INFO_HEADER_USER_IN),
NULL,0,
FALSE, NULL);
if(!OS_SUCCESS(RC)) {
UDFPrint(("UDFDoOPC failed\n"));
Vcb->OPCNum = 0;
// Vcb->VCBFlags |= UDF_VCB_FLAGS_OPC_FAILED;
}
Vcb->OPCDone = TRUE;
}
return RC;
} // end UDFDoOPC()
/*
This routine performs media-type dependent preparations
for write operation.
For CDR/RW it sets WriteParameters according to track parameters,
in some cases issues SYNC_CACHE command.
It can also send OPC info if requered.
If write-requested block is located beyond last formatted LBA
on incompletely formatted DVD media, this routine performs
all neccessary formatting operations in order to satisfy
subsequent write request.
*/
OSSTATUS
UDFPrepareForWriteOperation(
IN PVCB Vcb,
IN uint32 Lba,
IN uint32 BCount
)
{
#ifndef UDF_READ_ONLY_BUILD
#ifdef UDF_FORMAT_MEDIA
PUDFFmtState fms = Vcb->fms;
#else
#define fms FALSE
#endif //UDF_FORMAT_MEDIA
#ifdef _UDF_STRUCTURES_H_
if(Vcb->BSBM_Bitmap) {
ULONG i;
for(i=0; i<BCount; i++) {
if(UDFGetBit((uint32*)(Vcb->BSBM_Bitmap), Lba+i)) {
UDFPrint(("W: Known BB @ %#x\n", Lba));
//return STATUS_FT_WRITE_RECOVERY; // this shall not be treated as error and
// we shall get IO request to BAD block
return STATUS_DEVICE_DATA_ERROR;
}
}
}
#endif //_UDF_STRUCTURES_H_
Vcb->VCBFlags |= UDF_VCB_LAST_WRITE;
if(
#ifdef _BROWSE_UDF_
(((Vcb->FsDeviceType != FILE_DEVICE_CD_ROM_FILE_SYSTEM) ||
!(Vcb->VCBFlags & UDF_VCB_FLAGS_VOLUME_MOUNTED) ||
(Vcb->VCBFlags & UDF_VCB_FLAGS_RAW_DISK))
&& !fms
) ||
#endif //_BROWSE_UDF_
#ifdef UDF_FORMAT_MEDIA
(fms && fms->SkipPrepareW) ||
#endif //UDF_FORMAT_MEDIA
!(Vcb->VCBFlags & UDF_VCB_FLAGS_OUR_DEVICE_DRIVER)
) {
UDFPrint(("Skip prepare for Write @%x\n", Lba));
return STATUS_SUCCESS;
}
// check if the device requires OPC before each write operation
UDFDoOPC(Vcb);
if(Vcb->SyncCacheState == SYNC_CACHE_RECOVERY_ATTEMPT) {
Vcb->SyncCacheState = SYNC_CACHE_RECOVERY_RETRY;
} else {
Vcb->SyncCacheState = SYNC_CACHE_RECOVERY_NONE;
}
if(Vcb->LastModifiedTrack &&
(Vcb->TrackMap[Vcb->LastModifiedTrack].FirstLba <= Lba) &&
(Vcb->TrackMap[Vcb->LastModifiedTrack].LastLba >= Lba) &&
!( (Vcb->MediaClassEx == CdMediaClass_DVDRW ||
Vcb->MediaClassEx == CdMediaClass_DVDpRW ||
Vcb->MediaClassEx == CdMediaClass_DVDRAM ||
Vcb->MRWStatus == DiscInfo_BGF_Interrupted ||
Vcb->MRWStatus == DiscInfo_BGF_InProgress) && (Lba > Vcb->LastLBA))
) {
// Ok, we needn't change Write Parameters
// if(Vcb->TrackMap[Vcb->LastModifiedTrack].Flags & TrackMap_Try_variation)
// Vcb->TrackMap[Vcb->LastModifiedTrack].Flags |= TrackMap_Use_variation;
UDFPrint(("Skip prepare for Write (2) @%x\n", Lba));
return STATUS_SUCCESS;
}
UDFSetMRWMode(Vcb);
if(!UDFIsWriteParamsReq(Vcb)) {
#ifdef UDF_FORMAT_MEDIA
if(fms) {
return STATUS_SUCCESS;
}
#endif //UDF_FORMAT_MEDIA
}
for(uint32 i=Vcb->FirstTrackNum; i<=Vcb->LastTrackNum; i++) {
if((Vcb->TrackMap[i].FirstLba > Lba) ||
(Vcb->TrackMap[i].LastLba < Lba)) {
//UDFPrint(("not in track %d\n"));
continue;
}
OSSTATUS RC;
PGET_WRITE_MODE_USER_OUT WParams;
if(!UDFIsWriteParamsReq(Vcb)) {
RC = STATUS_SUCCESS;
goto check_dvd_bg_format;
}
if(!Vcb->WParams) {
Vcb->WParams =
(PGET_WRITE_MODE_USER_OUT)MyAllocatePool__(NonPagedPool, 512);
}
if(!(WParams = Vcb->WParams)) {
UDFPrint(("!WParams\n"));
return STATUS_INSUFFICIENT_RESOURCES;
}
RC = UDFPhSendIOCTL(IOCTL_CDRW_GET_WRITE_MODE, Vcb->TargetDeviceObject,
NULL,0,
(void*)(Vcb->WParams),sizeof(GET_WRITE_MODE_USER_OUT),
FALSE, NULL);
if(!OS_SUCCESS(RC)) {
#ifdef UDF_FORMAT_MEDIA
if(fms) {
fms->SkipPrepareW = 1;
MyFreePool__(WParams);
return STATUS_SUCCESS;
}
#endif //UDF_FORMAT_MEDIA
UDFPrint(("!get WParams\n"));
return RC;
}
// clear unnecassary flags
WParams->Byte2.Flags &= ~WParam_TestWrite;
WParams->Byte2.Flags &= ~WParam_WType_Mask;
// select packet writing
WParams->Byte2.Flags |= WParam_WType_Packet;
WParams->Byte3.Flags &= ~(WParam_TrkMode_Mask |
WParam_TrkMode_AllowCpy |
WParam_Copy);
WParams->Byte3.Flags |= Vcb->TrackMap[i].TrackParam &
(WParam_TrkMode_Mask |
WParam_TrkMode_AllowCpy |
WParam_Copy);
// set packet type (VP/FP)
// if(opt_partition == PT_VAT15 ||
// opt_blank_vat15)
if(WParams->Byte2.Flags & WParam_LS_V) {
WParams->LinkSize = 7;
}
if(Vcb->TrackMap[i].DataParam & TrkInfo_Packet) {
if((Vcb->TrackMap[i].DataParam & TrkInfo_FP) &&
!Vcb->CDR_Mode) {
WParams->Byte3.Flags |= WParam_FP;
} else {
WParams->Byte3.Flags &= ~WParam_FP;
}
} else {
if(!Vcb->CDR_Mode) {
WParams->Byte3.Flags |= WParam_FP;
} else {
WParams->Byte3.Flags &= ~WParam_FP;
}
}
// select multisession mode
WParams->Byte3.Flags &= ~WParam_MultiSes_Mask;
if((Vcb->DiscStat & DiscInfo_Disk_Mask) == DiscInfo_Disk_Appendable) {
WParams->Byte3.Flags |= WParam_Multises_Multi;
} else
if(Vcb->LastSession > 1) {
WParams->Byte3.Flags |= WParam_Multises_Final;
} else {
WParams->Byte3.Flags |= WParam_Multises_None;
}
// set sector mode (Mode1/XA)
WParams->Byte4.Flags &= ~WParam_BlkType_Mask;
if((Vcb->TrackMap[i].DataParam & TrkInfo_Dat_Mask) == TrkInfo_Dat_XA) {
// XA Mode2
WParams->Byte4.Flags |= WParam_BlkType_M2XAF1_2048;
WParams->SesFmt = WParam_SesFmt_CdRomXa;
} else if((Vcb->TrackMap[i].DataParam & TrkInfo_Dat_Mask) == TrkInfo_Dat_Mode1) {
// Mode1
WParams->Byte4.Flags |= WParam_BlkType_M1_2048;
WParams->SesFmt = WParam_SesFmt_CdRom;
} else {
#ifdef UDF_FORMAT_MEDIA
if(fms) {
fms->SkipPrepareW = 1;
MyFreePool__(WParams);
return STATUS_SUCCESS;
}
#endif //UDF_FORMAT_MEDIA
UDFPrint((" inv sector mode\n"));
return STATUS_INVALID_PARAMETER;
}
// set packet size
*((uint32*)&(WParams->PacketSize)) = BCount;
*((uint32*)&(WParams->SubHeader)) = 0;
// set additional flags for VP
if(Vcb->CDR_Mode) {
// if(opt_partition == PT_VAT15)
WParams->SubHeader.Params.Params1.SubMode = WParam_SubHdr_SubMode1;
}
WParams->PageLength = sizeof(GET_WRITE_MODE_USER_OUT)-2;
WParams->PageCode = MODE_PAGE_WRITE_PARAMS;
// apply write parameters
RC = UDFPhSendIOCTL(IOCTL_CDRW_SET_WRITE_MODE, Vcb->TargetDeviceObject,
(void*)WParams,sizeof(SET_WRITE_MODE_USER_IN),
NULL,0,FALSE,NULL);
#ifdef UDF_FORMAT_MEDIA
if(fms) {
if(!NT_SUCCESS(RC)) {
fms->SkipPrepareW = 1;
MyFreePool__(WParams);
return STATUS_SUCCESS;
}
RC = UDFPhSendIOCTL(IOCTL_CDRW_GET_WRITE_MODE, Vcb->TargetDeviceObject,
NULL,0,
(PVOID)WParams,sizeof(GET_WRITE_MODE_USER_OUT),
FALSE, NULL);
if(!NT_SUCCESS(RC)) {
MyFreePool__(WParams);
return RC;
}
if(fms->opt_partition == PT_VAT15 ||
fms->opt_blank_vat15) {
if(WParams->Byte3.Flags & WParam_FP) {
MyFreePool__(WParams);
return STATUS_INVALID_DEVICE_STATE;
}
} else {
if(!(WParams->Byte3.Flags & WParam_FP)) {
MyFreePool__(WParams);
return STATUS_INVALID_DEVICE_STATE;
}
}
}
#endif //UDF_FORMAT_MEDIA
// switch to random access mode
((PSET_RANDOM_ACCESS_USER_IN)WParams)->RandomAccessMode = Vcb->CDR_Mode ? FALSE : TRUE;
// ((PSET_RANDOM_ACCESS_USER_IN)WParams)->RandomAccessMode = (opt_partition != PT_VAT15) ? TRUE : FALSE;
RC = UDFPhSendIOCTL(IOCTL_CDRW_SET_RANDOM_ACCESS, Vcb->TargetDeviceObject,
(void*)WParams,sizeof(SET_RANDOM_ACCESS_USER_IN),
NULL,0,FALSE, NULL);
check_dvd_bg_format:
UDFPrint((" check BGF\n"));
if(!Vcb->CDR_Mode) {
if(OS_SUCCESS(RC)) {
Vcb->LastModifiedTrack = i;
if(!(Vcb->TrackMap[i].Flags & TrackMap_Use_variation)) {
if(Vcb->TrackMap[i].Flags & TrackMap_Try_variation) {
Vcb->TrackMap[i].Flags |= TrackMap_Use_variation;
} else {
Vcb->TrackMap[i].Flags |= TrackMap_Try_variation;
}
}
}
} else {
Vcb->LastModifiedTrack = 0;
}
// fms->SkipPrepareW = 1;
if((Vcb->MediaClassEx == CdMediaClass_DVDRW ||
Vcb->MediaClassEx == CdMediaClass_DVDpRW ||
Vcb->MediaClassEx == CdMediaClass_DVDRAM ||
Vcb->MRWStatus == DiscInfo_BGF_Interrupted )
&& (Lba > Vcb->LastLBA)) {
ULONG fLba;
SIZE_T WrittenBytes;
ULONG PSz = BCount << Vcb->BlockSizeBits;
#ifdef _BROWSE_UDF_
ULONG retry;
#endif //_BROWSE_UDF_
PFORMAT_CDRW_PARAMETERS_USER_IN ForBuf;
ASSERT((Vcb->LastLBA+1) == Vcb->NWA);
if(Lba+BCount <= (Vcb->LastLBA+1) ) {
UDFPrint(("DVD cont. fmt, LBA+BCount<=NWA, exiting\n"));
return STATUS_SUCCESS;
}
if((Vcb->MRWStatus != DiscInfo_BGF_Interrupted) &&
(Lba <= (Vcb->LastLBA+1)) ) {
UDFPrint(("!PausedBGF + DVD cont. fmt, LBA<=NWA, exiting\n"));
return STATUS_SUCCESS;
}
if(Vcb->MRWStatus == DiscInfo_BGF_Interrupted) {
// This code also can restart background MRW formatting
UDFPrint(("DVD cont. fmt, LastLBA %x, Lba %x\n", Vcb->LastLBA, Lba));
ForBuf = (PFORMAT_CDRW_PARAMETERS_USER_IN)DbgAllocatePoolWithTag(NonPagedPool, sizeof(FORMAT_CDRW_PARAMETERS_USER_IN), 'zNWD');
if(ForBuf) {
RtlZeroMemory(ForBuf, sizeof(FORMAT_CDRW_PARAMETERS_USER_IN));
ForBuf->Flags.FlagsEx = FORMAT_UNIT_RESTART_MRW;
ForBuf->BlockCount = 0xffffffff;
RC = UDFPhSendIOCTL(IOCTL_CDRW_FORMAT_UNIT, Vcb->TargetDeviceObject,
ForBuf,sizeof(FORMAT_CDRW_PARAMETERS_USER_IN),
NULL,0,FALSE, NULL);
DbgFreePool(ForBuf);
if(OS_SUCCESS(RC)) {
UDFPrint(("BGFormat restarted Interrupted->InProgress\n"));
Vcb->MRWStatus = DiscInfo_BGF_InProgress;
} else {
PGET_LAST_ERROR_USER_OUT Error = NULL;
if(!Vcb->Error) {
Vcb->Error = (PGET_LAST_ERROR_USER_OUT)
MyAllocatePool__(NonPagedPool, sizeof(GET_LAST_ERROR_USER_OUT));
}
Error = Vcb->Error;
if(Error) {
UDFPhSendIOCTL( IOCTL_CDRW_GET_LAST_ERROR, Vcb->TargetDeviceObject,
NULL,0,
Error,sizeof(GET_LAST_ERROR_USER_OUT),
TRUE,NULL);
UDFPrint(("SK=%x ASC=%x, ASCQ=%x, IE=%x\n",
Error->SenseKey, Error->AdditionalSenseCode, Error->AdditionalSenseCodeQualifier, Error->LastError));
// check for Long Write In Progress
if( (Error->SenseKey == SCSI_SENSE_NOT_READY) &&
(Error->AdditionalSenseCode == SCSI_ADSENSE_LUN_NOT_READY) &&
((Error->AdditionalSenseCodeQualifier == SCSI_SENSEQ_LONG_WRITE_IN_PROGRESS) ||
(Error->AdditionalSenseCodeQualifier == SCSI_SENSEQ_FORMAT_IN_PROGRESS)) ) {
RC = STATUS_SUCCESS;
UDFPrint(("Seems, BGFormat already restarted\n"));
Vcb->MRWStatus = DiscInfo_BGF_InProgress;
}
}
}
}
} else {
RC = STATUS_SUCCESS;
}
UDFPrint(("DVD cont. write, LastLBA %x, Lba %x\n", Vcb->LastLBA, Lba));
ASSERT(Vcb->MediaClassEx == CdMediaClass_DVDRW);
if(!Vcb->fZBuffer) {
Vcb->fZBuffer = (PCHAR)DbgAllocatePoolWithTag(NonPagedPool, PSz, 'zNWD');
RtlZeroMemory(Vcb->fZBuffer, PSz);
Vcb->fZBufferSize = PSz;
} else
if(Vcb->fZBufferSize < PSz) {
PSz = Vcb->fZBufferSize;
}
if(!Vcb->fZBuffer) {
BrutePoint();
RC = STATUS_INSUFFICIENT_RESOURCES;
} else {
for(fLba = Vcb->NWA; fLba < Lba; fLba+=BCount) {
#ifdef _BROWSE_UDF_
retry = UDF_WRITE_MAX_RETRY;
retry_1:
#endif //_BROWSE_UDF_
RC = UDFPhWriteVerifySynchronous(Vcb->TargetDeviceObject, Vcb->fZBuffer, PSz,
((uint64)fLba) << Vcb->BlockSizeBits, &WrittenBytes, PH_TMP_BUFFER);
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
UDFPrint(("Fmt status: %x\n", RC));
#ifdef _BROWSE_UDF_
if(!OS_SUCCESS(RC) &&
OS_SUCCESS(RC = UDFRecoverFromError(Vcb, TRUE, RC, fLba, BCount, &retry)) ) {
goto retry_1;
UDFPrint(("Fmt retry\n"));
}
#endif //_BROWSE_UDF_
if(!OS_SUCCESS(RC)) {
BrutePoint();
UDFPrint(("Fmt break on ERROR\n"));
break;
}
UDFUpdateNWA(Vcb, fLba, BCount, RC);
}
}
} else {
UDFPrint((" no special processing\n"));
}
return RC;
}
#endif //UDF_READ_ONLY_BUILD
UDFPrint((" no suitable track!\n"));
return STATUS_INVALID_PARAMETER;
} // end UDFPrepareForWriteOperation()
//#ifdef _BROWSE_UDF_
/*
This routine tries to recover from hardware error
Return: STATUS_SUCCESS - retry requst
STATUS_XXX - unrecoverable error
*/
OSSTATUS
UDFRecoverFromError(
IN PVCB Vcb,
IN BOOLEAN WriteOp,
IN OSSTATUS status,
IN uint32 Lba,
IN uint32 BCount,
IN OUT uint32* retry
)
{
PGET_LAST_ERROR_USER_OUT Error = NULL;
LARGE_INTEGER delay;
// OSSTATUS RC;
uint32 i;
BOOLEAN UpdateBB = FALSE;
if(!(*retry) ||
!(Vcb->VCBFlags & UDF_VCB_FLAGS_OUR_DEVICE_DRIVER) ||
(Vcb->FsDeviceType != FILE_DEVICE_CD_ROM_FILE_SYSTEM))
return status;
(*retry)--;
// allocate tmp buffer
_SEH2_TRY {
if(!Vcb->Error) {
if(!(Vcb->Error = (PGET_LAST_ERROR_USER_OUT)
MyAllocatePool__(NonPagedPool, sizeof(GET_LAST_ERROR_USER_OUT))))
try_return(status);
}
if(status == STATUS_NO_SUCH_DEVICE) {
UDFPrint(("Error recovery: STATUS_NO_SUCH_DEVICE, die.....\n"));
Vcb->VCBFlags |= UDF_VCB_FLAGS_UNSAFE_IOCTL | UDF_VCB_FLAGS_DEAD;
try_return(status);
}
#ifdef _UDF_STRUCTURES_H_
if(status == STATUS_NO_MEDIA_IN_DEVICE && !Vcb->EjectWaiter) {
UDFPrint(("Error recovery: STATUS_NO_MEDIA_IN_DEVICE, prevent further remount.....\n"));
// Make sure, that volume will never be quick-remounted
// It is very important for ChkUdf utility and
// some CD-recording libraries
Vcb->SerialNumber--;
try_return(status);
}
#endif //_UDF_STRUCTURES_H_
Error = Vcb->Error;
UDFPhSendIOCTL( IOCTL_CDRW_GET_LAST_ERROR, Vcb->TargetDeviceObject,
NULL,0,
Error,sizeof(GET_LAST_ERROR_USER_OUT),
TRUE,NULL);
UDFPrint(("SK=%x ASC=%x, ASCQ=%x, IE=%x\n",
Error->SenseKey, Error->AdditionalSenseCode, Error->AdditionalSenseCodeQualifier, Error->LastError));
// check for Long Write In Progress
if( ((Error->SenseKey == SCSI_SENSE_NOT_READY) &&
(Error->AdditionalSenseCode == SCSI_ADSENSE_LUN_NOT_READY) &&
(Error->AdditionalSenseCodeQualifier == SCSI_SENSEQ_LONG_WRITE_IN_PROGRESS)) ) {
// we should wait...
if(WriteOp) {
if((*retry) == UDF_WRITE_MAX_RETRY-1) {
UDFPrint(("Error recovery: reserve retry count for write retries\n"));
(*retry) = UDF_WRITE_MAX_RETRY*3;
} else
if((*retry) == UDF_WRITE_MAX_RETRY) {
UDFPrint(("Error recovery: jump over UDF_WRITE_MAX_RETRY\n"));
(*retry)--;
}
delay.QuadPart = -500000; // 0.05 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
if(WriteOp && ((*retry) > UDF_WRITE_MAX_RETRY-1)) {
UDFPrint(("Error recovery: simple write retry with delay\n"));
try_return(status = STATUS_SUCCESS);
}
} else {
delay.QuadPart = -500000; // 0.05 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
if((*retry) == UDF_WRITE_MAX_RETRY-1) {
UDFPrint(("Error recovery: retry read after small delay\n"));
try_return(status = STATUS_SUCCESS);
}
}
UDFPrint(("Error recovery: sync cache\n"));
// ...flush device cache...
UDFSyncCache(Vcb);
// wait again & retry
delay.QuadPart = -1000000; // 0.1 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
#ifdef _UDF_STRUCTURES_H_
if(Vcb->BGWriters) (*retry)++;
#endif //_UDF_STRUCTURES_H_
try_return(status = STATUS_SUCCESS);
} else
// check for Long Write In Progress
if((Error->SenseKey == SCSI_SENSE_NOT_READY) &&
(Error->AdditionalSenseCode == SCSI_ADSENSE_LUN_NOT_READY) &&
((Error->AdditionalSenseCodeQualifier == SCSI_SENSEQ_FORMAT_IN_PROGRESS) ||
(Error->AdditionalSenseCodeQualifier == SCSI_SENSEQ_BECOMING_READY) ||
(Error->AdditionalSenseCodeQualifier == SCSI_SENSEQ_OPERATION_IN_PROGRESS) ) ) {
// we should wait & retry
UDFPrint(("Error recovery: op. in progress, waiting 0.3 sec\n"));
delay.QuadPart = -3000000; // 0.3 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
#ifdef _UDF_STRUCTURES_H_
if(Vcb->BGWriters) (*retry)++;
#endif //_UDF_STRUCTURES_H_
Vcb->SyncCacheState = SYNC_CACHE_RECOVERY_ATTEMPT;
try_return(status = STATUS_SUCCESS);
} else
// check for non empty cache special case
if((Error->SenseKey == SCSI_SENSE_ILLEGAL_REQUEST) &&
(Error->AdditionalSenseCode == SCSI_ADSENSE_INVALID_CMD_SEQUENCE)) {
// we should wait & retry
if(!WriteOp) {
UDFPrint(("Error recovery: invalid command sequence on read\n"));
delay.QuadPart = -1000000; // 0.1 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
UDFPrint(("Error recovery: sync cache\n"));
// ...flush device cache...
UDFSyncCache(Vcb);
// wait again & retry
delay.QuadPart = -1000000; // 0.1 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
#ifdef _UDF_STRUCTURES_H_
if(Vcb->BGWriters) (*retry)++;
#endif //_UDF_STRUCTURES_H_
try_return(status = STATUS_SUCCESS);
}
goto reinit_sector_mode;
} else
// check for Bus Reset (sometimes it happends...)
if((Error->SenseKey == SCSI_SENSE_UNIT_ATTENTION) &&
(Error->AdditionalSenseCode == SCSI_ADSENSE_BUS_RESET) ) {
// we should wait
UDFPrint(("Error recovery: bus reset...\n"));
Vcb->MediaChangeCount = Error->MediaChangeCount;
delay.QuadPart = -1000000; // 0.1 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
// reset driver
UDFResetDeviceDriver(Vcb, Vcb->TargetDeviceObject, FALSE);
delay.QuadPart = -1000000; // 0.1 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
// lock it
/* ((PPREVENT_MEDIA_REMOVAL_USER_IN)(Error))->PreventMediaRemoval = TRUE;
UDFPhSendIOCTL( IOCTL_STORAGE_MEDIA_REMOVAL,
Vcb->TargetDeviceObject,
Error,sizeof(PREVENT_MEDIA_REMOVAL_USER_IN),
NULL,0,
FALSE,NULL);
delay.QuadPart = -1000000; // 0.1 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);*/
// reinit write mode the following is performed inside UDFResetDeviceDriver()
//Vcb->LastModifiedTrack = 0;
//Vcb->OPCDone = FALSE;
reinit_sector_mode:
// reinit sector mode
Vcb->LastModifiedTrack = 0;
UDFPrepareForWriteOperation(Vcb, Lba, BCount);
try_return(status = STATUS_SUCCESS);
} else
// check for Illegal Sector Mode.
// We can get this error 'cause of 2 reasons:
// a) Bus reset occured. We should reinit
// b) CopyProtection settings missmatch
// c) preblems with DNA of firmware developer, some TEACs fall into such state
// after failed streaming read
if((Error->SenseKey == SCSI_SENSE_ILLEGAL_REQUEST) &&
(Error->AdditionalSenseCode == SCSI_ADSENSE_ILLEGAL_MODE_FOR_THIS_TRACK)) {
bad_rw_seek_recovery:
if(WriteOp) {
if((*retry) <= 1) {
// Variate CopyProtection...
for(i=Vcb->FirstTrackNum; i<=Vcb->LastTrackNum; i++) {
if((Vcb->TrackMap[i].FirstLba > Lba) ||
(Vcb->TrackMap[i].LastLba < Lba))
continue;
/* if(Vcb->TrackMap[i].Flags & TrackMap_CopyBit_variated)
// Last chance....
goto reinit_sector_mode;*/
// check if we have successuflly completed WriteOp
// using Variation.
// We should not variate these bits again in this case.
if(Vcb->TrackMap[i].Flags & TrackMap_Use_variation)
break;
Vcb->TrackMap[i].Flags &= ~TrackMap_Try_variation;
/* if((Vcb->TrackMap[i].Flags & TrackMap_Try_variation) &&
(Vcb->TrackMap[i].Flags & (TrackMap_AllowCopyBit_variated |
TrackMap_CopyBit_variated)))
break;*/
/* if(Vcb->TrackMap[i].Flags & TrackMap_Use_variation)
break;*/
Vcb->TrackMap[i].Flags |= TrackMap_Try_variation;
// Try variation.
if(!(Vcb->TrackMap[i].Flags ^= TrackMap_AllowCopyBit_variated))
Vcb->TrackMap[i].Flags ^= TrackMap_CopyBit_variated;
if(Vcb->TrackMap[i].Flags & (TrackMap_AllowCopyBit_variated |
TrackMap_CopyBit_variated) ) {
(*retry) = 1;
} else {
Vcb->TrackMap[i].Flags &= ~TrackMap_Try_variation;
}
// reinit sector mode
Vcb->LastModifiedTrack = 0;
UDFPrepareForWriteOperation(Vcb, Lba, BCount);
break;
}
} else {
// Reinit...
//reinit_sector_mode:
// we should wait
delay.QuadPart = -1000000; // 0.1 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
// reinit sector mode
goto reinit_sector_mode;
/*
Vcb->LastModifiedTrack = 0;
UDFPrepareForWriteOperation(Vcb, Lba, BCount);
try_return(status = STATUS_SUCCESS);
*/
}
} else
if((Vcb->CompatFlags & UDF_VCB_IC_BAD_RW_SEEK) &&
(Vcb->IncrementalSeekState != INCREMENTAL_SEEK_DONE)) {
UDFPrint(("Using incremental seek workaround...\n"));
Vcb->IncrementalSeekState = INCREMENTAL_SEEK_WORKAROUND;
try_return(status = STATUS_SUCCESS);
} else {
UDFPrint(("Seems to be BB @ %x\n", Lba));
UpdateBB = TRUE;
}
} else
if((Error->SenseKey == SCSI_SENSE_ILLEGAL_REQUEST) &&
(Error->AdditionalSenseCode == SCSI_ADSENSE_INVALID_SESSION_MODE)) {
if(WriteOp &&
(Vcb->SavedFeatures & CDRW_FEATURE_STREAMING) &&
Lba+BCount <= Vcb->LastLBA+1) {
UDFPrint(("bad Session in streaming mode. Lba %x, try fix-up\n", Lba));
// ...flush device cache...
UDFSyncCache(Vcb);
// we should wait
delay.QuadPart = -10000000; // 1 sec
KeDelayExecutionThread(KernelMode, FALSE, &delay);
try_return(status = STATUS_SUCCESS);
}
} else
if((Error->LastError == CDRW_ERR_WRITE_IN_PROGRESS_BUSY) ||
(status == STATUS_DEVICE_BUSY)) {
delay.QuadPart = -5000000; // 0.5 sec
UDFPrint(("CDRW_ERR_WRITE_IN_PROGRESS_BUSY || STATUS_DEVICE_BUSY\n"));
KeDelayExecutionThread(KernelMode, FALSE, &delay);
#ifdef _UDF_STRUCTURES_H_
if(Vcb->BGWriters) (*retry)++;
#endif //_UDF_STRUCTURES_H_
try_return(status = STATUS_SUCCESS);
} else
// some devices (SONY) return such a strange sequence....
if( ((Error->SenseKey == SCSI_SENSE_ILLEGAL_REQUEST) &&
(Error->AdditionalSenseCode == SCSI_ADSENSE_INVALID_CDB)) &&
WriteOp) {
// reinit write mode
Vcb->LastModifiedTrack = 0;
UDFPrepareForWriteOperation(Vcb, Lba, BCount);
try_return(status = STATUS_SUCCESS);
} else
// No seek on Read... to morgue, I'm afraid
if((Error->SenseKey == SCSI_SENSE_MEDIUM_ERROR) /*&&
((Error->AdditionalSenseCode == SCSI_ADSENSE_CD_READ_ERROR) ||
(Error->AdditionalSenseCode == SCSI_ADSENSE_NO_SENSE) ||
(Error->AdditionalSenseCode == SCSI_ADSENSE_FORMAT_CORRUPTED) ||
(Error->AdditionalSenseCode == SCSI_ADSENSE_SEEK_ERROR))*/ &&
!WriteOp) {
if(Error->AdditionalSenseCode == SCSI_ADSENSE_SEEK_ERROR) {
UDFPrint(("Seek error\n"));
if(Vcb->CompatFlags & UDF_VCB_IC_BAD_RW_SEEK) {
UDFPrint(("try recovery\n"));
goto bad_rw_seek_recovery;
}
UDFPrint(("map error to STATUS_NONEXISTENT_SECTOR\n"));
status = STATUS_NONEXISTENT_SECTOR;
}
UDFPrint(("Seems to be BB @ %x (read 2)\n", Lba));
UpdateBB = TRUE;
} else
// handle invalid block address
if( ((Error->SenseKey == SCSI_SENSE_ILLEGAL_REQUEST) &&
(Error->AdditionalSenseCode == SCSI_ADSENSE_ILLEGAL_BLOCK)) ) {
if(!WriteOp &&
(Vcb->SavedFeatures & CDRW_FEATURE_STREAMING) &&
Lba+BCount <= Vcb->LastLBA+1) {
UDFPrint(("bad LBA %x in streaming mode, try fix-up\n", Lba));
// ...flush device cache...
UDFSyncCache(Vcb);
try_return(status = STATUS_SUCCESS);
}
if((Lba+BCount >= Vcb->LastLBA) &&
(Vcb->MRWStatus == DiscInfo_BGF_Interrupted)) {
UDFPrint(("stupid drive, cannot read beyond formatted area on DiscInfo_BGF_Interrupted\n"));
UpdateBB = FALSE;
try_return(status = STATUS_BUFFER_ALL_ZEROS);
}
// prevent Bad Block Bitmap modification
}
try_exit: NOTHING;
} _SEH2_FINALLY {
#ifdef UDF_DBG
if(OS_SUCCESS(status)) {
UDFPrint(("Retry\n"));
}
#endif //UDF_DBG
} _SEH2_END;
if(!OS_SUCCESS(status)) {
if((Vcb->MountPhErrorCount != (ULONG)-1) &&
(Vcb->MountPhErrorCount < 0x7fffffff)) {
Vcb->MountPhErrorCount++;
}
//#ifdef _UDF_STRUCTURES_H_
if(UpdateBB && (BCount == 1)) {
uint32* bm;
if(!(bm = (uint32*)(Vcb->BSBM_Bitmap))) {
bm = (uint32*)(Vcb->BSBM_Bitmap = (int8*)DbgAllocatePoolWithTag(NonPagedPool, (i = (Vcb->LastPossibleLBA+1+7)>>3), 'mNWD' ));
if(bm) {
RtlZeroMemory(bm, i);
} else {
UDFPrint(("Can't alloc BSBM for %x blocks\n", Vcb->LastPossibleLBA));
}
}
if(bm) {
UDFSetBit(bm, Lba);
UDFPrint(("Set BB @ %#x\n", Lba));
}
#ifdef _BROWSE_UDF_
bm = (uint32*)(Vcb->FSBM_Bitmap);
if(bm) {
UDFSetUsedBit(bm, Lba);
UDFPrint(("Set BB @ %#x as used\n", Lba));
}
#endif //_BROWSE_UDF_
}
//#endif //_UDF_STRUCTURES_H_
}
return status;
} // end UDFRecoverFromError()
//#endif //_BROWSE_UDF_
/*
This routine attempts to read disk layout using ReadDisk/Track info cmd
*/
OSSTATUS
UDFReadDiscTrackInfo(
PDEVICE_OBJECT DeviceObject, // the target device object
PVCB Vcb // Volume Control Block for ^ DevObj
)
{
OSSTATUS RC = STATUS_SUCCESS;
PDISC_INFO_BLOCK_USER_OUT DiscInfo = (PDISC_INFO_BLOCK_USER_OUT)MyAllocatePool__(NonPagedPool,sizeof(DISC_INFO_BLOCK_USER_OUT) );
PTRACK_INFO_BLOCK_USER_OUT TrackInfoOut = (PTRACK_INFO_BLOCK_USER_OUT)MyAllocatePool__(NonPagedPool,sizeof(TRACK_INFO_BLOCK_USER_OUT) );
PTRACK_INFO_BLOCK_USER_IN TrackInfoIn = (PTRACK_INFO_BLOCK_USER_IN)TrackInfoOut;
READ_CAPACITY_USER_OUT CapacityBuffer;
LONG TrackNumber;
BOOLEAN NotFP = FALSE;
BOOLEAN ForceFP = FALSE;
BOOLEAN PacketTrack = FALSE;
BOOLEAN MRWRetry = FALSE;
// BOOLEAN ReadCapacityOk = FALSE;
#ifdef UDF_FORMAT_MEDIA
PUDFFmtState fms = Vcb->fms;
#endif
_SEH2_TRY {
if(!DiscInfo || !TrackInfoOut)
try_return(RC = STATUS_INSUFFICIENT_RESOURCES);
MRWRetry_label:
RC = UDFPhSendIOCTL(IOCTL_CDRW_READ_DISC_INFO, DeviceObject,
NULL, 0,
DiscInfo,sizeof(DISC_INFO_BLOCK_USER_OUT), TRUE, NULL);
if(!OS_SUCCESS(RC)) {
UDFPrint(("ReadDiskInfo failed. Use default.\n"));
if(Vcb->MediaClassEx == CdMediaClass_DVDRW ||
Vcb->MediaClassEx == CdMediaClass_DVDpRW ||
Vcb->MediaClassEx == CdMediaClass_DVDRAM) {
Vcb->LastPossibleLBA = DEFAULT_LAST_LBA_DVD;
} else
if(Vcb->MediaClassEx == CdMediaClass_BDRE) {
Vcb->LastPossibleLBA = DEFAULT_LAST_LBA_BD;
} else {
Vcb->LastPossibleLBA = DEFAULT_LAST_LBA_FP_CD;
}
try_return(RC);
}
#ifdef UDF_FORMAT_MEDIA
if(fms && fms->opt_disk_info) {
UserPrint(("ReadDiskInfo OK\n"));
}
#endif //UDF_FORMAT_MEDIA
RC = UDFPhSendIOCTL(IOCTL_CDRW_READ_CAPACITY, DeviceObject,
NULL, 0,
&CapacityBuffer,sizeof(READ_CAPACITY_USER_OUT), TRUE, NULL);
if(!OS_SUCCESS(RC)) {
UDFPrint(("ReadCapacity failed.\n"));
if(Vcb->MediaClassEx == CdMediaClass_DVDpRW) {
Vcb->LastPossibleLBA = DEFAULT_LAST_LBA_DVD;
}
} else {
UDFPrint(("ReadCapacity ok.\n"));
UDFPrint(("Last possible LBA %#x.\n", CapacityBuffer.LogicalBlockAddress));
if(!(CapacityBuffer.LogicalBlockAddress & 0xc0000000) &&
(CapacityBuffer.LogicalBlockAddress != 0x7fffffff)) {
// good value from ReadCapacity
UDFPrint(("Update Last possible LBA %#x.\n", CapacityBuffer.LogicalBlockAddress));
Vcb->LastPossibleLBA = CapacityBuffer.LogicalBlockAddress;
// ReadCapacityOk = TRUE;
#ifdef UDF_FORMAT_MEDIA
if(fms && fms->opt_disk_info) {
UserPrint(("ReadCapacity OK\n"));
}
#endif //UDF_FORMAT_MEDIA
}
}
#ifdef _CONSOLE
Vcb->PhDeviceType = FILE_DEVICE_CD_ROM;
#endif //_CONSOLE
Vcb->PhSerialNumber = *((uint32*)&(DiscInfo->DiskId));
Vcb->PhErasable = DiscInfo->DiscStat.Flags & DiscInfo_Disk_Erasable;
Vcb->PhDiskType = DiscInfo->DiskType;
// save OPC info
if(DiscInfo->OPCNum)
Vcb->OPCNum = DiscInfo->OPCNum;
UDFPrint(("DiskInfo: SN %x, OPCn %x(%x), Stat %x, Flg: %x\n",
Vcb->PhSerialNumber, Vcb->OPCNum, DiscInfo->OPCNum, DiscInfo->DiscStat.Flags, DiscInfo->Flags.Flags));
#ifdef UDF_FORMAT_MEDIA
if(fms && fms->opt_disk_info) {
UserPrint(("Media type: "));
switch(Vcb->MediaClassEx) {
case CdMediaClass_CDROM : UserPrint(("CD-ROM \n")); break;
case CdMediaClass_CDR : UserPrint(("CD-R \n")); break;
case CdMediaClass_CDRW : UserPrint(("CD-RW \n")); break;
case CdMediaClass_DVDROM : UserPrint(("DVD-ROM \n")); break;
case CdMediaClass_DVDRAM : UserPrint(("DVD-RAM \n")); break;
case CdMediaClass_DVDR : UserPrint(("DVD-R \n")); break;
case CdMediaClass_DVDRW : UserPrint(("DVD-RW \n")); break;
case CdMediaClass_DVDpR : UserPrint(("DVD+R \n")); break;
case CdMediaClass_DVDpRW : UserPrint(("DVD+RW \n")); break;
case CdMediaClass_DDCDROM : UserPrint(("DDCD-ROM \n")); break;
case CdMediaClass_DDCDR : UserPrint(("DDCD-R \n")); break;
case CdMediaClass_DDCDRW : UserPrint(("DDCD-RW \n")); break;
case CdMediaClass_BDROM : UserPrint(("BD-ROM \n")); break;
case CdMediaClass_BDRE : UserPrint(("BD-RE \n")); break;
case CdMediaClass_BDR : UserPrint(("BD-R \n")); break;
case CdMediaClass_HD_DVDROM : UserPrint(("HD DVD-ROM \n")); break;
case CdMediaClass_HD_DVDRAM : UserPrint(("HD DVD-RAM \n")); break;
case CdMediaClass_HD_DVDR : UserPrint(("HD DVD-R \n")); break;
case CdMediaClass_HD_DVDRW : UserPrint(("HD DVD-RW \n")); break;
default: UserPrint(("Unknown\n")); break;
}
UserPrint(("SN %#x, OPCn %#x\n",
Vcb->PhSerialNumber, Vcb->OPCNum, DiscInfo->OPCNum));
UserPrint(("Disk State: "));
switch(DiscInfo->DiscStat.Flags & DiscInfo_Disk_Mask) {
case DiscInfo_Disk_Empty:
UserPrint(("Empty\n"));
break;
case DiscInfo_Disk_Appendable:
UserPrint(("Appendable\n"));
break;
case DiscInfo_Disk_Complete:
UserPrint(("Complete\n"));
break;
case DiscInfo_Disk_OtherRW:
UserPrint(("RW in unknown state\n"));
break;
}
UserPrint(("Last Session State: "));
switch(DiscInfo->DiscStat.Flags & DiscInfo_Ses_Mask) {
case DiscInfo_Ses_Empty:
UserPrint(("Empty\n"));
break;
case DiscInfo_Ses_Incomplete:
UserPrint(("Incomplete\n"));
break;
case DiscInfo_Ses_Complete:
UserPrint(("Complete\n"));
break;
default:
UserPrint(("unknown state\n"));
break;
}
UserPrint(("Erasable: %s\n",
(DiscInfo->DiscStat.Flags & DiscInfo_Disk_Erasable) ? "yes" : "no"
));
}
#endif //UDF_FORMAT_MEDIA
// Save disk status
Vcb->DiscStat = DiscInfo->DiscStat.Flags;
if((DiscInfo->DiscStat.Flags & DiscInfo_Disk_Mask) == DiscInfo_Disk_Empty) {
UDFPrint(("Blank\n"));
Vcb->BlankCD = TRUE;
}
if( (DiscInfo->DiscStat.Flags & DiscInfo_Disk_Mask) == DiscInfo_Disk_Empty ||
(DiscInfo->DiscStat.Flags & DiscInfo_Ses_Mask) == DiscInfo_Ses_Incomplete) {
// we shall mount empty disk to make it possible for
// external applications to perform format operation
// or something like this
UDFPrint(("Try RAW_MOUNT\n"));
Vcb->VCBFlags |= UDF_VCB_FLAGS_RAW_DISK;
PacketTrack = TRUE;
}
#ifndef _BROWSE_UDF_
// If drive returned reasonable value from ReadCapacity, do not use
// last LeadIn/LeadOut
if(Vcb->MediaClassEx != CdMediaClass_DVDpRW &&
!ReadCapacityOk) {
// +RW returns bad value
UDFPrint(("+RW returns bad value\n"));
Vcb->LastPossibleLBA = (DiscInfo->LastSesLeadOutLBA & 0x80000000) ?
0 : DiscInfo->LastSesLeadOutLBA;
if(!(DiscInfo->LastSesLeadInLBA & 0x80000000)) {
Vcb->LastPossibleLBA = max(DiscInfo->LastSesLeadInLBA, Vcb->LastPossibleLBA);
}
}
#endif // _BROWSE_UDF_
if((DiscInfo->Flags.Flags & DiscInfo_BGF_Mask) != 0) {
UDFPrint(("ForceFP + MRW\n"));
ForceFP = TRUE;
Vcb->MRWStatus = DiscInfo->Flags.Flags & DiscInfo_BGF_Mask;
// update addressing mode
if(!MRWRetry) {
UDFSetMRWMode(Vcb);
MRWRetry = TRUE;
goto MRWRetry_label;
}
}
UDFPrint(("MRW state %x\n", Vcb->MRWStatus));
if(Vcb->MediaClassEx == CdMediaClass_DVDRW) {
if(Vcb->PhMediaCapFlags & CdCapFlags_RandomWritable) {
UDFPrint(("DVD-RW Rewritable\n"));
ForceFP = TRUE;
} else
if((DiscInfo->DiscStat.Flags & DiscInfo_Disk_Mask) == DiscInfo_Disk_Empty) {
UDFPrint(("Blank DVD-RW\n"));
ForceFP = TRUE;
} else {
UDFPrint(("DVD-RW Sequential\n"));
NotFP = TRUE;
}
} else
if(CdrwIsDvdOverwritable(Vcb->MediaClassEx)) {
UDFPrint(("force Rewritable (2)\n"));
ForceFP = TRUE;
}
// We have incomplete last session, so process each track from last to first
// Vcb->LastPossibleLBA = DiscInfo->LastSesLeadInLBA;
Vcb->LastSession = DiscInfo->Status.NumOfSes;
Vcb->LastTrackNum = DiscInfo->Status.LastTrackNumLastSes;
Vcb->FirstTrackNum = DiscInfo->FirstTrackNum;
// some devices report LastTrackNum=0 for full disks
Vcb->LastTrackNum = max(Vcb->LastTrackNum, Vcb->FirstTrackNum);
if(!Vcb->LastTrackNum) {
UDFPrint(("Try read 1st track...\n"));
Vcb->LastTrackNum = 1;
}
UDFPrint(("DiskInfo: 1st trk %x, last trk %x\n", Vcb->FirstTrackNum, Vcb->LastTrackNum));
#ifdef UDF_FORMAT_MEDIA
if(fms && fms->opt_disk_info) {
UserPrint(("First track: %d\n"
"Last track: %d\n", Vcb->FirstTrackNum, Vcb->LastTrackNum));
UserPrint(("------------------------------------------\n"));
}
#endif //UDF_FORMAT_MEDIA
RC = UDFReallocTrackMap(Vcb, Vcb->LastTrackNum+1);
if(!OS_SUCCESS(RC))
try_return(RC);
// Get last LBA from invisible track (if any)
RtlZeroMemory(TrackInfoOut,sizeof(TRACK_INFO_BLOCK_USER_OUT));
TrackInfoIn->LBA_TrkNum = 0; // invisible track
TrackInfoIn->Track = TRUE;
RC = UDFPhSendIOCTL(IOCTL_CDRW_READ_TRACK_INFO, DeviceObject,
TrackInfoIn, sizeof(TRACK_INFO_BLOCK_USER_IN),
TrackInfoOut,sizeof(TRACK_INFO_BLOCK_USER_OUT), TRUE, NULL);
if(OS_SUCCESS(RC)) {
if((Vcb->LastTrackNum < TrackInfoOut->TrackNum) &&
TrackInfoOut->TrackLength &&
(TrackInfoOut->TrackStartLBA != TrackInfoOut->NextWriteLBA)) {
Vcb->LastTrackNum = TrackInfoOut->TrackNum;
if(!(TrackInfoOut->NextWriteLBA & 0x80000000))
Vcb->NWA = TrackInfoOut->NextWriteLBA;
if(TrackInfoOut->TrackLength > 1) {
Vcb->LastPossibleLBA =
TrackInfoOut->TrackStartLBA + TrackInfoOut->TrackLength - (TrackInfoOut->TrackLength ? 1 : 0);
UDFPrint((" set LastPossibleLBA=%x\n", Vcb->LastPossibleLBA));
}
}
UDFPrint(("Ses %d, Track %d (%x, len %x) PckSize %x: \n"
" NWA: %x (%s) DatType:%x, %s %s %s %s TrkType:%x %s %s\n"
" LRA: %x (%s) RC_LBA:%x\n",
TrackInfoOut->SesNum,
0,
TrackInfoOut->TrackStartLBA,
TrackInfoOut->TrackLength,
TrackInfoOut->FixPacketSize,
TrackInfoOut->NextWriteLBA,
TrackInfoOut->NWA_V & TrkInfo_NWA_V ? "vld" : "inv",
TrackInfoOut->DataParam.Flags & TrkInfo_Dat_Mask,
(TrackInfoOut->DataParam.Flags & TrkInfo_Packet) ? "Pck" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_FP) ? "FP" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_Blank) ? "Blank" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_RT) ? "RT" : "",
TrackInfoOut->TrackParam.Flags & TrkInfo_Trk_Mask,
(TrackInfoOut->TrackParam.Flags & TrkInfo_Copy) ? "Cpy" : "",
(TrackInfoOut->TrackParam.Flags & TrkInfo_Damage) ? "Damage" : "",
TrackInfoOut->LastRecordedAddr,
(TrackInfoOut->NWA_V & TrkInfo_LRA_V) ? "vld" : "inv",
TrackInfoOut->ReadCompatLBA
));
#ifdef UDF_FORMAT_MEDIA
if(fms && fms->opt_disk_info) {
UserPrint(("Invisible track: \n"));
UserPrint((" Ses %d, Track %d (%x, len %x) PckSize %x: \n"
" NWA: %x (%s) DatType:%x, %s %s %s %s TrkType:%x %s %s\n"
" LRA: %x (%s) RC_LBA:%x\n",
TrackInfoOut->SesNum,
0,
TrackInfoOut->TrackStartLBA,
TrackInfoOut->TrackLength,
TrackInfoOut->FixPacketSize,
TrackInfoOut->NextWriteLBA,
TrackInfoOut->NWA_V & TrkInfo_NWA_V ? "vld" : "inv",
TrackInfoOut->DataParam.Flags & TrkInfo_Dat_Mask,
(TrackInfoOut->DataParam.Flags & TrkInfo_Packet) ? "Pck" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_FP) ? "FP" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_Blank) ? "Blank" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_RT) ? "RT" : "",
TrackInfoOut->TrackParam.Flags & TrkInfo_Trk_Mask,
(TrackInfoOut->TrackParam.Flags & TrkInfo_Copy) ? "Cpy" : "",
(TrackInfoOut->TrackParam.Flags & TrkInfo_Damage) ? "Damage" : "",
TrackInfoOut->LastRecordedAddr,
(TrackInfoOut->NWA_V & TrkInfo_LRA_V) ? "vld" : "inv",
TrackInfoOut->ReadCompatLBA
));
}
#endif //UDF_FORMAT_MEDIA
}
for (TrackNumber=(LONG)DiscInfo->FirstTrackNum;TrackNumber <= (LONG)Vcb->LastTrackNum;TrackNumber++) {
RtlZeroMemory(TrackInfoOut,sizeof(TRACK_INFO_BLOCK_USER_OUT));
TrackInfoIn->LBA_TrkNum = TrackNumber;
TrackInfoIn->Track = TRUE;
RC = UDFPhSendIOCTL(IOCTL_CDRW_READ_TRACK_INFO, DeviceObject,
TrackInfoIn, sizeof(TRACK_INFO_BLOCK_USER_IN),
TrackInfoOut,sizeof(TRACK_INFO_BLOCK_USER_OUT), TRUE, NULL);
// fill sector type map
if(TrackInfoOut->TrackStartLBA & 0x80000000) {
UDFPrint(("TrkInfo: Bad FirstLba (%x), change to %x\n", TrackInfoOut->TrackStartLBA, 0));
Vcb->TrackMap[TrackNumber].FirstLba = 0;
} else {
Vcb->TrackMap[TrackNumber].FirstLba = TrackInfoOut->TrackStartLBA;
}
if(TrackInfoOut->TrackLength & 0x80000000) {
UDFPrint(("TrkInfo: Bad TrackLength (%x), change to %x\n", TrackInfoOut->TrackLength,
Vcb->LastPossibleLBA - Vcb->TrackMap[TrackNumber].FirstLba + 1));
TrackInfoOut->TrackLength = Vcb->LastPossibleLBA - Vcb->TrackMap[TrackNumber].FirstLba + 1;
}
Vcb->TrackMap[TrackNumber].LastLba = TrackInfoOut->TrackStartLBA +
TrackInfoOut->TrackLength -
(TrackInfoOut->TrackLength ? 1 : 0);
Vcb->TrackMap[TrackNumber].TrackParam = TrackInfoOut->TrackParam.Flags;
Vcb->TrackMap[TrackNumber].DataParam = TrackInfoOut->DataParam.Flags;
Vcb->TrackMap[TrackNumber].NWA_V = TrackInfoOut->NWA_V;
if((TrackInfoOut->NextWriteLBA & 0x80000000) ||
(TrackInfoOut->NextWriteLBA < TrackInfoOut->TrackStartLBA)) {
if(!(Vcb->TrackMap[TrackNumber].LastLba & 0x8000000)) {
UDFPrint(("TrkInfo: set NWA to LastLba (%x)\n", Vcb->TrackMap[TrackNumber].LastLba));
Vcb->TrackMap[TrackNumber].NWA =
Vcb->TrackMap[TrackNumber].LastLba;
} else {
UDFPrint(("TrkInfo: set NWA to INV (1)\n"));
Vcb->TrackMap[TrackNumber].NWA = 0;
Vcb->TrackMap[TrackNumber].NWA_V = 0;
}
} else {
if(!(TrackInfoOut->NextWriteLBA & 0x80000000)) {
UDFPrint(("TrkInfo: Good NWA (%x)\n", TrackInfoOut->NextWriteLBA));
Vcb->TrackMap[TrackNumber].NWA =
TrackInfoOut->NextWriteLBA;
} else {
UDFPrint(("TrkInfo: set NWA to INV (2)\n"));
Vcb->TrackMap[TrackNumber].NWA = 0;
Vcb->TrackMap[TrackNumber].NWA_V = 0;
}
}
Vcb->TrackMap[TrackNumber].Session = TrackInfoOut->SesNum;
// for FP tracks we shall get PacketSize from returned info
// otherwise set to default UDF value (0x20)
if(NotFP) {
UDFPrint(("Apply NotFP\n"));
Vcb->TrackMap[TrackNumber].DataParam &= ~TrkInfo_FP;
#ifdef DBG
TrackInfoOut->DataParam.Flags &= ~TrkInfo_FP;
#endif //DBG
} else
if(ForceFP) {
UDFPrint(("Apply ForceFP\n"));
PacketTrack = TRUE;
Vcb->TrackMap[TrackNumber].DataParam |= TrkInfo_FP;
#ifdef DBG
TrackInfoOut->DataParam.Flags |= TrkInfo_FP;
#endif //DBG
}
if(Vcb->TrackMap[TrackNumber].DataParam & TrkInfo_FP) {
Vcb->TrackMap[TrackNumber].PacketSize = TrackInfoOut->FixPacketSize;
Vcb->VCBFlags |= UDF_VCB_FLAGS_RAW_DISK;
Vcb->FP_disc = TRUE;
} else {
Vcb->TrackMap[TrackNumber].PacketSize = PACKETSIZE_UDF;
}
// presence of Damaged track means, that we should mount this disk in RAW mode
if(Vcb->TrackMap[TrackNumber].TrackParam & TrkInfo_Damage) {
UDFPrint(("TrkInfo_Damage, Try RAW_MOUNT\n"));
Vcb->VCBFlags |= UDF_VCB_FLAGS_RAW_DISK;
}
// presence of track with Unknown data type means, that we should mount
// this disk in RAW mode
if((TrackInfoOut->DataParam.Flags & TrkInfo_Dat_Mask) == TrkInfo_Trk_unknown) {
UDFPrint(("Unknown DatType, Try RAW_MOUNT\n"));
Vcb->VCBFlags |= UDF_VCB_FLAGS_RAW_DISK;
}
PacketTrack |= ((TrackInfoOut->DataParam.Flags & TrkInfo_Packet) != 0);
UDFPrint(("Ses %d, Track %d (%x - %x) PckSize %x: \n"
" NWA: %x (%s) DatType:%x, %s %s %s %s TrkType:%x %s %s\n"
" LRA: %x (%s) RC_LBA:%x\n",
TrackInfoOut->SesNum,
TrackNumber,
Vcb->TrackMap[TrackNumber].FirstLba,
Vcb->TrackMap[TrackNumber].LastLba,
TrackInfoOut->FixPacketSize,
TrackInfoOut->NextWriteLBA,
TrackInfoOut->NWA_V & TrkInfo_NWA_V ? "vld" : "inv",
TrackInfoOut->DataParam.Flags & TrkInfo_Dat_Mask,
(TrackInfoOut->DataParam.Flags & TrkInfo_Packet) ? "Pck" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_FP) ? "FP" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_Blank) ? "Blank" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_RT) ? "RT" : "",
TrackInfoOut->TrackParam.Flags & TrkInfo_Trk_Mask,
(TrackInfoOut->TrackParam.Flags & TrkInfo_Copy) ? "Cpy" : "",
(TrackInfoOut->TrackParam.Flags & TrkInfo_Damage) ? "Damage" : "",
TrackInfoOut->LastRecordedAddr,
(TrackInfoOut->NWA_V & TrkInfo_LRA_V) ? "vld" : "inv",
TrackInfoOut->ReadCompatLBA
));
#ifdef UDF_FORMAT_MEDIA
if(fms && fms->opt_disk_info) {
UserPrint(("Track %d: \n", TrackNumber));
UserPrint((" Ses %d, Track %d (%x, len %x) PckSize %x: \n"
" NWA: %x (%s) DatType:%x, %s %s %s %s TrkType:%x %s %s\n"
" LRA: %x (%s) RC_LBA:%x\n",
TrackInfoOut->SesNum,
TrackNumber,
TrackInfoOut->TrackStartLBA,
TrackInfoOut->TrackLength,
TrackInfoOut->FixPacketSize,
TrackInfoOut->NextWriteLBA,
TrackInfoOut->NWA_V & TrkInfo_NWA_V ? "vld" : "inv",
TrackInfoOut->DataParam.Flags & TrkInfo_Dat_Mask,
(TrackInfoOut->DataParam.Flags & TrkInfo_Packet) ? "Pck" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_FP) ? "FP" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_Blank) ? "Blank" : "",
(TrackInfoOut->DataParam.Flags & TrkInfo_RT) ? "RT" : "",
TrackInfoOut->TrackParam.Flags & TrkInfo_Trk_Mask,
(TrackInfoOut->TrackParam.Flags & TrkInfo_Copy) ? "Cpy" : "",
(TrackInfoOut->TrackParam.Flags & TrkInfo_Damage) ? "Damage" : "",
TrackInfoOut->LastRecordedAddr,
(TrackInfoOut->NWA_V & TrkInfo_LRA_V) ? "vld" : "inv",
TrackInfoOut->ReadCompatLBA
));
}
#endif //UDF_FORMAT_MEDIA
if(TrackNumber == DiscInfo->FirstTrackNum) {
if(!(Vcb->TrackMap[TrackNumber].FirstLba & 0x80000000)) {
UDFPrint(("TrkInfo: Update FirstLBA (%x)\n", Vcb->TrackMap[TrackNumber].FirstLba));
Vcb->FirstLBA = Vcb->TrackMap[TrackNumber].FirstLba;
}
}
if((TrackInfoOut->SesNum == Vcb->LastSession) && !Vcb->FirstTrackNumLastSes) {
if(!(Vcb->TrackMap[TrackNumber].FirstLba & 0x80000000)) {
UDFPrint(("TrkInfo: Update FirstLBALastSes (%x)\n", Vcb->TrackMap[TrackNumber].FirstLba));
Vcb->FirstLBALastSes = Vcb->TrackMap[TrackNumber].FirstLba;
}
Vcb->FirstTrackNumLastSes = TrackNumber;
}
}
if(!(TrackInfoOut->NextWriteLBA & 0x80000000) &&
!(TrackInfoOut->TrackLength & 0x80000000) &&
(Vcb->NWA < TrackInfoOut->NextWriteLBA)
) {
UDFPrint((" set NWA to %x\n", TrackInfoOut->NextWriteLBA));
if(Vcb->MediaClassEx != CdMediaClass_DVDpRW) {
Vcb->NWA = TrackInfoOut->NextWriteLBA;
} else {
Vcb->NWA =
TrackInfoOut->TrackStartLBA + TrackInfoOut->TrackLength - (TrackInfoOut->TrackLength ? 1 : 0);
}
}
if(Vcb->MediaClassEx != CdMediaClass_DVDpRW &&
!(TrackInfoOut->TrackLength & 0x80000000) &&
TrackInfoOut->TrackLength > 1) {
Vcb->LastPossibleLBA =
TrackInfoOut->TrackStartLBA + TrackInfoOut->TrackLength - (TrackInfoOut->TrackLength ? 1 : 0);
UDFPrint((" set LastPossibleLBA=%x\n", Vcb->LastPossibleLBA));
}
TrackNumber = Vcb->LastTrackNum;
// quick formatted +RW returns bogus value
if(Vcb->MediaClassEx == CdMediaClass_DVDpRW) {
UDFPrint((" check quick formatted +RW\n"));
if(Vcb->TrackMap[TrackNumber].LastLba &&
!(Vcb->TrackMap[TrackNumber].LastLba & 0x80000000) &&
Vcb->TrackMap[TrackNumber].LastLba < Vcb->LastPossibleLBA /*&&
Vcb->TrackMap[TrackNumber].LastLba != Vcb->LastPossibleLBA*/
) {
UDFPrint((" track LastLBA %x != LastPossibleLBA %x, verify\n",
Vcb->TrackMap[TrackNumber].LastLba, Vcb->LastPossibleLBA));
if(Vcb->MRWStatus == DiscInfo_BGF_Complete) {
UDFPrint((" complete MRW state\n"));
#ifdef _BROWSE_UDF_
Vcb->LastPossibleLBA =
Vcb->NWA =
Vcb->LastLBA =
Vcb->TrackMap[TrackNumber].LastLba;
goto valid_track_length;
#endif // _BROWSE_UDF_
} else
if(Vcb->MRWStatus) {
uint8* buff;
SIZE_T ReadBytes;
UDFPrint((" MRW state %x\n", Vcb->MRWStatus));
buff = (uint8*)DbgAllocatePoolWithTag(NonPagedPool, Vcb->WriteBlockSize, 'bNWD' );
if(buff) {
RC = UDFTRead(Vcb,
buff,
Vcb->WriteBlockSize,
Vcb->TrackMap[TrackNumber].LastLba+1,
&ReadBytes,
PH_TMP_BUFFER);
DbgFreePool(buff);
if(!OS_SUCCESS(RC)) {
UDFPrint((" Can't read beyond track LastLBA (%x)\n", Vcb->TrackMap[TrackNumber].LastLba+1));
Vcb->LastLBA = Vcb->TrackMap[TrackNumber].LastLba;
Vcb->NWA = Vcb->LastLBA+1;
Vcb->TrackMap[TrackNumber].NWA_V = 1;
Vcb->TrackMap[TrackNumber].NWA = Vcb->NWA;
Vcb->TrackMap[TrackNumber].LastLba = Vcb->LastPossibleLBA;
RC = STATUS_SUCCESS;
goto valid_track_length;
}
}
}
}
UDFPrint((" set track LastLBA %x\n", Vcb->LastPossibleLBA));
Vcb->NWA =
Vcb->LastLBA =
Vcb->TrackMap[TrackNumber].LastLba =
Vcb->LastPossibleLBA;
}
valid_track_length:
// Test for last empty session
if((Vcb->TrackMap[TrackNumber].Session !=
Vcb->TrackMap[TrackNumber-1].Session) &&
(Vcb->LastSession > 1)) {
// Note: some devices return negative track length
if((Vcb->TrackMap[TrackNumber].LastLba <=
Vcb->TrackMap[TrackNumber].FirstLba) ||
(Vcb->TrackMap[TrackNumber].FirstLba ==
Vcb->TrackMap[TrackNumber].NWA)) {
// empty last session...
Vcb->LastTrackNum--;
// TrackNumber--;
/* for(SesNum = Vcb->TrackMap[TrackNumber].Session;
Vcb->TrackMap[TrackNumber].Session == SesNum;
TrackNumber--) {
}*/
if(TrackNumber>1)
Vcb->LastSession = Vcb->TrackMap[TrackNumber-1].Session;
}
}
TrackNumber = Vcb->LastTrackNum;
#ifdef _BROWSE_UDF_
Vcb->LastLBA = min(Vcb->TrackMap[TrackNumber].LastLba, Vcb->TrackMap[TrackNumber].NWA);
#endif //_BROWSE_UDF_
if(Vcb->TrackMap[TrackNumber].NWA_V & TrkInfo_NWA_V) {
UDFPrint((" NWA ok, set LastLBA to min(Last %x, NWA %x\n",
Vcb->TrackMap[TrackNumber].LastLba,
Vcb->TrackMap[TrackNumber].NWA));
Vcb->LastLBA = min(Vcb->TrackMap[TrackNumber].LastLba, Vcb->TrackMap[TrackNumber].NWA);
} else {
UDFPrint((" no NWA, set LastLBA to Last %x\n", Vcb->TrackMap[TrackNumber].LastLba));
Vcb->LastLBA = Vcb->TrackMap[TrackNumber].LastLba;
}
Vcb->VCBFlags |= UDF_VCB_FLAGS_TRACKMAP;
if(!PacketTrack && Vcb->MediaClassEx != CdMediaClass_DVDRAM ) {
UDFPrint((" disable Raw mount\n"));
Vcb->VCBFlags &= ~UDF_VCB_FLAGS_RAW_DISK;
}
try_exit: NOTHING;
} _SEH2_FINALLY {
if(DiscInfo) MyFreePool__(DiscInfo);
if(TrackInfoOut) MyFreePool__(TrackInfoOut);
} _SEH2_END;
return RC;
} // end UDFReadDiscTrackInfo()
/*
This routine attempts to read disk layout using ReadFullTOC cmd
*/
OSSTATUS
UDFReadAndProcessFullToc(
PDEVICE_OBJECT DeviceObject, // the target device object
PVCB Vcb
)
{
OSSTATUS RC = STATUS_SUCCESS;
PREAD_FULL_TOC_USER_OUT toc = (PREAD_FULL_TOC_USER_OUT)MyAllocatePool__(NonPagedPool,sizeof(READ_FULL_TOC_USER_OUT) );
uint32 index;
uint8 POINT;
uint8 CurTrack = 0;
uint32 LastLeadOut = 0;
// BOOLEAN IsMRW = FALSE;
UDFPrint(("UDFReadAndProcessFullToc\n"));
if(!toc) return STATUS_INSUFFICIENT_RESOURCES;
Vcb->FirstTrackNum = 0xFF;
RtlZeroMemory(toc,sizeof(READ_FULL_TOC_USER_OUT));
RC = UDFPhSendIOCTL(IOCTL_CDRW_READ_FULL_TOC,DeviceObject,
NULL,0,
toc,sizeof(READ_FULL_TOC_USER_OUT),
TRUE,NULL);
if(!OS_SUCCESS(RC)) {
MyFreePool__(toc);
return RC;
}
#ifdef _CONSOLE
Vcb->PhDeviceType = FILE_DEVICE_CD_ROM;
#endif //_CONSOLE
Vcb->LastSession = toc->Sessions.Last_TrackSes;
RC = UDFReallocTrackMap(Vcb, 0x100);
if(!OS_SUCCESS(RC)) {
MyFreePool__(toc);
return RC;
}
// get LastPossibleLBA
// Note: some drives return Full TOC items unordered.
// So, LeadOut position may come before Track definition.
// In order to handle such situation, we must initialize
// CurTrack when First or Last Track descriptor comes
for (index=0;(index<MAXIMUM_NUMBER_OF_SESSIONS);index++) {
/* if((toc->SessionData[index].Adr == TOC_ADR_TrackInfo) &&
((toc->SessionData[index].Control == TOC_CTL_MRWTrackInfo) || (toc->SessionData[index].Control == TOC_CTL_MRWLastSes))) {
IsMRW = TRUE;
}*/
if(toc->SessionData[index].Adr == 1) {
switch (POINT = toc->SessionData[index].POINT) {
case POINT_FirstTrackNum: {
Vcb->FirstTrackNum = toc->SessionData[index].Params.FirstTrackNum.FirstTrackNum;
if(!CurTrack)
CurTrack = (uint8)(Vcb->FirstTrackNum);
break;
}
case POINT_LastTrackNum: {
Vcb->LastTrackNum = toc->SessionData[index].Params.LastTrackNum.LastTrackNum;
if(CurTrack < Vcb->LastTrackNum)
CurTrack = (uint8)(Vcb->FirstTrackNum);
break;
}
case POINT_StartPositionOfLeadOut: {
#define TempMSF toc->SessionData[index].Params.StartPositionOfLeadOut.MSF
Vcb->TrackMap[CurTrack].LastLba = MSF_TO_LBA(TempMSF[0],TempMSF[1],TempMSF[2]);
LastLeadOut = max(LastLeadOut, Vcb->TrackMap[CurTrack].LastLba);
#undef TempMSF
break;
}
default: {
if( (Vcb->FirstTrackNum != 0x0FF) &&
(toc->SessionData[index].POINT == Vcb->FirstTrackNum) ) {
#define TempMSF toc->SessionData[index].Params.StartPositionOfTrack.MSF
Vcb->FirstLBA = MSF_TO_LBA(TempMSF[0],TempMSF[1],TempMSF[2]);
if(Vcb->FirstLBA & 0x80000000) {
Vcb->FirstLBA = 0;
}
#undef TempMSF
}
break;
}
}
if((POINT >= POINT_StartPositionOfTrack_Min) &&
(POINT <= POINT_StartPositionOfTrack_Max)) {
#define TempMSF toc->SessionData[index].Params.StartPositionOfTrack.MSF
Vcb->TrackMap[POINT].FirstLba = MSF_TO_LBA(TempMSF[0],TempMSF[1],TempMSF[2])-1;
if(Vcb->TrackMap[POINT].FirstLba & 0x80000000) {
if(POINT == 1) {
Vcb->TrackMap[POINT].FirstLba = 0;
} else {
if(Vcb->TrackMap[POINT-1].LastLba) {
Vcb->TrackMap[POINT].FirstLba = Vcb->TrackMap[POINT-1].LastLba+1;
}
}
}
#undef TempMSF
if(POINT > POINT_StartPositionOfTrack_Min) {
Vcb->TrackMap[POINT-1].LastLba = Vcb->TrackMap[POINT].FirstLba-1;
}
CurTrack = POINT;
}
} else
if(toc->SessionData[index].Adr == 5) {
switch (POINT = toc->SessionData[index].POINT) {
case POINT_StartPositionOfNextProgramArea: {
#define TempMSF toc->SessionData[index].Params.StartPositionOfNextProgramArea.MaxLeadOut_MSF
Vcb->LastPossibleLBA = MSF_TO_LBA(TempMSF[0],TempMSF[1],TempMSF[2]);
#undef TempMSF
break;
}
default: {
break;
}
}
}
}
/* if(!IsMRW) {
UDFPrint(("No MRW\n"));
Vcb->CompatFlags &= ~UDF_VCB_IC_MRW_ADDR_PROBLEM;
}*/
// Vcb->CompatFlags &= ~UDF_VCB_IC_MRW_ADDR_PROBLEM;
// some devices report LastTrackNum=0 for full disks
Vcb->LastTrackNum = max(Vcb->LastTrackNum, Vcb->FirstTrackNum);
Vcb->TrackMap[Vcb->LastTrackNum].LastLba = max(LastLeadOut, Vcb->TrackMap[Vcb->LastTrackNum].LastLba);
Vcb->LastLBA = Vcb->TrackMap[Vcb->LastTrackNum].LastLba;
MyFreePool__(toc);
// Vcb->LastLBA=PacketVariable2Fixed(Vcb->LastLBA)-2;
return STATUS_SUCCESS;
} // end UDFReadAndProcessFullToc()
/*
use standard way to determine disk layout (ReadTOC cmd)
*/
OSSTATUS
UDFUseStandard(
PDEVICE_OBJECT DeviceObject, // the target device object
PVCB Vcb // Volume control block from this DevObj
)
{
OSSTATUS RC = STATUS_SUCCESS;
PREAD_TOC_USER_OUT toc = (PREAD_TOC_USER_OUT)MyAllocatePool__(NonPagedPool,max(Vcb->BlockSize, sizeof(READ_TOC_USER_OUT)) );
PGET_LAST_SESSION_USER_OUT LastSes = (PGET_LAST_SESSION_USER_OUT)MyAllocatePool__(NonPagedPool,sizeof(GET_LAST_SESSION_USER_OUT) );
uint32 LocalTrackCount;
// uint32 LocalTocLength;
uint32 TocEntry;
#ifdef _BROWSE_UDF_
uint32 OldTrkNum;
uint32 TrkNum;
SIZE_T ReadBytes, i, len;
#endif //_BROWSE_UDF_
#ifdef UDF_FORMAT_MEDIA
PUDFFmtState fms = Vcb->fms;
#else
#define fms FALSE
#endif //UDF_FORMAT_MEDIA
UDFPrint(("UDFUseStandard\n"));
_SEH2_TRY {
if(!toc || !LastSes) {
try_return (RC = STATUS_INSUFFICIENT_RESOURCES);
}
RtlZeroMemory(toc,sizeof(READ_TOC_TOC));
Vcb->VCBFlags |= UDF_VCB_FLAGS_USE_STD;
RC = UDFPhSendIOCTL(IOCTL_CDROM_READ_TOC,DeviceObject,
toc,sizeof(READ_TOC_USER_OUT),
toc,sizeof(READ_TOC_USER_OUT),
TRUE,NULL );
if((RC == STATUS_DEVICE_NOT_READY) || (RC == STATUS_NO_MEDIA_IN_DEVICE)) {
try_return(RC);
}
#ifdef UDF_FORMAT_MEDIA
if(fms->opt_media == MT_none) {
try_return(RC = STATUS_NO_MEDIA_IN_DEVICE);
}
#endif //UDF_FORMAT_MEDIA
// If even standard read toc does not work, then use default values
if(!OS_SUCCESS(RC)) {
RC = UDFReallocTrackMap(Vcb, 2);
if(!OS_SUCCESS(RC)) {
try_return(RC);
}
Vcb->LastSession=1;
Vcb->FirstTrackNum=1;
// Vcb->FirstLBA=0;
Vcb->LastTrackNum=1;
Vcb->TrackMap[1].FirstLba = Vcb->FirstLBA;
Vcb->TrackMap[1].LastLba = Vcb->LastLBA;
Vcb->TrackMap[1].PacketSize = PACKETSIZE_UDF;
#ifdef UDF_FORMAT_MEDIA
if(!fms) {
#endif //UDF_FORMAT_MEDIA
#ifdef _BROWSE_UDF_
#ifdef UDF_HDD_SUPPORT
if(UDFGetDevType(DeviceObject) == FILE_DEVICE_DISK) {
try_return(RC = STATUS_SUCCESS);
}
#endif //UDF_HDD_SUPPORT
#endif //_BROWSE_UDF_
#ifdef UDF_FORMAT_MEDIA
} else {
if(fms->opt_media == MT_HD) {
Vcb->LastPossibleLBA = Vcb->LastLBA;
try_return(RC = STATUS_SUCCESS);
}
}
#endif //UDF_FORMAT_MEDIA
Vcb->LastPossibleLBA = max(Vcb->LastLBA, DEFAULT_LAST_LBA_FP_CD);
Vcb->TrackMap[1].DataParam = TrkInfo_Dat_XA | TrkInfo_FP | TrkInfo_Packet;
Vcb->TrackMap[1].TrackParam = TrkInfo_Trk_XA;
Vcb->TrackMap[1].NWA = 0xffffffff;
Vcb->NWA = DEFAULT_LAST_LBA_FP_CD + 7 + 1;
try_return(RC = STATUS_SUCCESS);
}
#ifdef _CONSOLE
Vcb->PhDeviceType = FILE_DEVICE_CD_ROM;
#endif //_CONSOLE
LocalTrackCount = toc->Tracks.Last_TrackSes - toc->Tracks.First_TrackSes + 1;
// LocalTocLength = PtrOffset( toc, &(toc->TrackData[LocalTrackCount + 1]) ); /* FIXME ReactOS Assume PtrOffset is not changing it's arguments? */
// Get out if there is an immediate problem with the TOC.
if(toc->Tracks.First_TrackSes > toc->Tracks.Last_TrackSes) {
try_return(RC = STATUS_DISK_CORRUPT_ERROR);
}
#ifdef _BROWSE_UDF_
Vcb->LastTrackNum=toc->Tracks.Last_TrackSes;
Vcb->FirstTrackNum=toc->Tracks.First_TrackSes;
// some devices report LastTrackNum=0 for full disks
Vcb->LastTrackNum = max(Vcb->LastTrackNum, Vcb->FirstTrackNum);
RC = UDFReallocTrackMap(Vcb, MAXIMUM_NUMBER_OF_TRACKS+1);
/* if(Vcb->TrackMap) {
MyFreePool__(Vcb->TrackMap);
Vcb->TrackMap = NULL;
}
Vcb->TrackMap = (PUDFTrackMap)
MyAllocatePool__(NonPagedPool, (MAXIMUM_NUMBER_OF_TRACKS+1)*sizeof(UDFTrackMap));
if(!Vcb->TrackMap) {
MyFreePool__(toc);
return STATUS_INSUFFICIENT_RESOURCES;
}
RtlZeroMemory(Vcb->TrackMap,(MAXIMUM_NUMBER_OF_TRACKS+1)*sizeof(UDFTrackMap));
*/
if(!OS_SUCCESS(RC)) {
BrutePoint();
try_return(RC);
}
// find 1st and last session
RC = UDFPhSendIOCTL(IOCTL_CDROM_GET_LAST_SESSION,DeviceObject,
LastSes,sizeof(GET_LAST_SESSION_USER_OUT),
LastSes,sizeof(GET_LAST_SESSION_USER_OUT),
TRUE,NULL );
if(OS_SUCCESS(RC)) {
TrkNum = LastSes->LastSes_1stTrack.TrackNum;
Vcb->LastSession = LastSes->Sessions.First_TrackSes;
for(TocEntry=0;TocEntry<LocalTrackCount + 1;TocEntry++) {
if(toc->TrackData[TocEntry].TrackNum == TrkNum) {
Vcb->TrackMap[TrkNum].Session = Vcb->LastSession;
}
}
}
OldTrkNum = 0;
// Scan toc for first & last LBA
for(TocEntry=0;TocEntry<LocalTrackCount + 1;TocEntry++) {
#define TempMSF toc->TrackData[TocEntry].LBA
TrkNum = toc->TrackData[TocEntry].TrackNum;
#ifdef UDF_DBG
if (TrkNum >= MAXIMUM_NUMBER_OF_TRACKS &&
TrkNum != TOC_LastTrack_ID) {
UDFPrint(("UDFUseStandard: Array out of bounds\n"));
BrutePoint();
try_return(RC = STATUS_SUCCESS);
}
UDFPrint(("Track N %d (0x%x) first LBA %ld (%lx) \n",TrkNum,TrkNum,
MSF_TO_LBA(TempMSF[1],TempMSF[2],TempMSF[3]),
MSF_TO_LBA(TempMSF[1],TempMSF[2],TempMSF[3])));
#endif // UDF_DBG
if(Vcb->FirstTrackNum == TrkNum) {
Vcb->FirstLBA = MSF_TO_LBA(TempMSF[1],TempMSF[2],TempMSF[3]);
if(Vcb->FirstLBA & 0x80000000) {
Vcb->FirstLBA = 0;
}
}
if(TOC_LastTrack_ID == TrkNum) {
Vcb->LastLBA = MSF_TO_LBA(TempMSF[1],TempMSF[2],TempMSF[3])-1;
Vcb->TrackMap[OldTrkNum].LastLba = Vcb->LastLBA-1;
UDFPrint(("UDFUseStandard: Last track entry, break TOC scan\n"));
// continue;
break;
} else {
Vcb->TrackMap[TrkNum].FirstLba = MSF_TO_LBA(TempMSF[1],TempMSF[2],TempMSF[3]);
if(Vcb->TrackMap[TrkNum].FirstLba & 0x80000000)
Vcb->TrackMap[TrkNum].FirstLba = 0;
if(TrkNum) {
if (TOC_LastTrack_ID == OldTrkNum) {
UDFPrint(("UDFUseStandard: Wrong previous track number\n"));
BrutePoint();
} else {
Vcb->TrackMap[OldTrkNum].LastLba = Vcb->TrackMap[TrkNum].FirstLba-1;
}
}
}
// check track type
switch(toc->TrackData[TocEntry].Control & TocControl_TrkMode_Mask) {
case TocControl_TrkMode_Data:
case TocControl_TrkMode_IncrData:
Vcb->TrackMap[TrkNum].DataParam = TrkInfo_Dat_XA;
Vcb->TrackMap[TrkNum].TrackParam = TrkInfo_Trk_XA;
break;
default:
Vcb->TrackMap[TrkNum].DataParam = TrkInfo_Dat_unknown;
Vcb->TrackMap[TrkNum].TrackParam = TrkInfo_Trk_unknown;
}
OldTrkNum = TrkNum;
#undef TempMSF
}
TrkNum = Vcb->LastTrackNum;
RC = STATUS_SUCCESS;
// find last _valid_ track
for(;TrkNum;TrkNum--) {
if((Vcb->TrackMap[TrkNum].DataParam != TrkInfo_Dat_unknown) &&
(Vcb->TrackMap[TrkNum].TrackParam != TrkInfo_Trk_unknown)) {
RC = STATUS_UNSUCCESSFUL;
Vcb->LastTrackNum = TrkNum;
break;
}
}
// no valid tracks...
if(!TrkNum) {
UDFPrint(("UDFUseStandard: no valid tracks...\n"));
try_return(RC = STATUS_UNRECOGNIZED_VOLUME);
}
i = 0;
// Check for last VP track. Some last sectors may belong to Link-data &
// be unreadable. We should forget about them, because UDF needs
// last _readable_ sector.
while(!OS_SUCCESS(RC) && (i<8)) {
RC = UDFPhReadSynchronous(Vcb->TargetDeviceObject, (int8*)toc, Vcb->BlockSize,
((uint64)(Vcb->TrackMap[TrkNum].LastLba-i)) << Vcb->BlockSizeBits, &ReadBytes, PH_TMP_BUFFER);
i++;
}
if(OS_SUCCESS(RC)) {
Vcb->LastLBA = Vcb->TrackMap[TrkNum].LastLba-i+1;
/* if(i) {
Vcb->TrackMap[TrkNum].PacketSize = PACKETSIZE_UDF;
Vcb->TrackMap[TrkNum].;
}*/
} else {
// Check for FP track. READ_TOC reports actual track length, but
// Link-data is hidden & unreadable for us. So, available track
// length may be less than actual. Here we assume that Packet-size
// is PACKETSIZE_UDF.
i = 0;
len = Vcb->TrackMap[TrkNum].LastLba - Vcb->TrackMap[TrkNum].FirstLba + 1;
len = (uint32)(((int64)len*PACKETSIZE_UDF) / (PACKETSIZE_UDF+7));
while(!OS_SUCCESS(RC) && (i<9)) {
RC = UDFPhReadSynchronous(Vcb->TargetDeviceObject, (int8*)toc, Vcb->BlockSize,
((uint64)(Vcb->TrackMap[TrkNum].FirstLba-i+len)) << Vcb->BlockSizeBits, &ReadBytes, PH_TMP_BUFFER);
i++;
}
if(OS_SUCCESS(RC)) {
Vcb->LastLBA =
Vcb->TrackMap[TrkNum].LastLba = Vcb->TrackMap[TrkNum].FirstLba-i+len+1;
Vcb->TrackMap[TrkNum].PacketSize = PACKETSIZE_UDF;
// Vcb->TrackMap[TrkNum].;
} else
if(RC == STATUS_INVALID_DEVICE_REQUEST) {
// wrap return code from Audio-disk
RC = STATUS_SUCCESS;
}
}
#ifdef UDF_CDRW_EMULATION_ON_ROM
Vcb->LastPossibleLBA = Vcb->LastLBA+7+1+1024;
Vcb->NWA = Vcb->LastLBA+7+1;
#else
Vcb->LastPossibleLBA =
Vcb->NWA = Vcb->LastLBA+7+1;
#endif //UDF_CDRW_EMULATION_ON_ROM
#else //_BROWSE_UDF_
Vcb->FirstTrackNum=toc->Tracks.Last_TrackSes;
Vcb->LastTrackNum=toc->Tracks.First_TrackSes;
// Scan toc for first & last LBA
for(TocEntry=0;TocEntry<LocalTrackCount + 1;TocEntry++) {
#define TempMSF toc->TrackData[TocEntry].LBA
if(Vcb->FirstTrackNum == toc->TrackData[TocEntry].TrackNum) {
Vcb->FirstLBA = MSF_TO_LBA(TempMSF[1],TempMSF[2],TempMSF[3]);
if(Vcb->FirstLBA & 0x80000000) {
Vcb->FirstLBA = 0;
}
}
if(TOC_LastTrack_ID == toc->TrackData[TocEntry].TrackNum) {
Vcb->LastLBA = MSF_TO_LBA(TempMSF[1],TempMSF[2],TempMSF[3])-1;
}
#undef TempMSF
}
// Vcb->LastLBA=PacketVariable2Fixed(Vcb->LastLBA)-2;
Vcb->LastPossibleLBA = DEFAULT_LAST_LBA_FP_CD;
#endif //_BROWSE_UDF_
try_exit: NOTHING;
} _SEH2_FINALLY {
if(toc) MyFreePool__(toc);
if(LastSes) MyFreePool__(LastSes);
} _SEH2_END;
return RC;
} // end UDFUseStandard()
/*
Get block size (for read operation)
*/
OSSTATUS
UDFGetBlockSize(
IN PDEVICE_OBJECT DeviceObject, // the target device object
IN PVCB Vcb // Volume control block from this DevObj
)
{
OSSTATUS RC = STATUS_SUCCESS;
PDISK_GEOMETRY DiskGeometry = (PDISK_GEOMETRY)MyAllocatePool__(NonPagedPool,sizeof(DISK_GEOMETRY));
PPARTITION_INFORMATION PartitionInfo = (PPARTITION_INFORMATION)MyAllocatePool__(NonPagedPool,sizeof(PARTITION_INFORMATION)*2);
#ifdef UDF_FORMAT_MEDIA
PUDFFmtState fms = Vcb->fms;
#else
#define fms FALSE
#endif //UDF_FORMAT_MEDIA
if(!DiskGeometry || !PartitionInfo)
try_return (RC = STATUS_INSUFFICIENT_RESOURCES);
#ifdef _BROWSE_UDF_
#ifdef UDF_HDD_SUPPORT
if(!fms) {
if(UDFGetDevType(DeviceObject) == FILE_DEVICE_DISK) {
UDFPrint(("UDFGetBlockSize: HDD\n"));
RC = UDFPhSendIOCTL(IOCTL_DISK_GET_DRIVE_GEOMETRY,DeviceObject,
0,NULL,
DiskGeometry,sizeof(DISK_GEOMETRY),
TRUE,NULL );
Vcb->BlockSize = (OS_SUCCESS(RC)) ? DiskGeometry->BytesPerSector : 512;
if(!NT_SUCCESS(RC))
try_return(RC);
RC = UDFPhSendIOCTL(IOCTL_DISK_GET_PARTITION_INFO,DeviceObject,
0,NULL,
PartitionInfo,sizeof(PARTITION_INFORMATION),
TRUE,NULL );
if(!NT_SUCCESS(RC)) {
UDFPrint(("UDFGetBlockSize: IOCTL_DISK_GET_PARTITION_INFO failed\n"));
if(RC == STATUS_INVALID_DEVICE_REQUEST) /* ReactOS Code Change (was =) */
RC = STATUS_UNRECOGNIZED_VOLUME;
try_return(RC);
}
if(PartitionInfo->PartitionType != PARTITION_IFS) {
UDFPrint(("UDFGetBlockSize: PartitionInfo->PartitionType != PARTITION_IFS\n"));
try_return(RC = STATUS_UNRECOGNIZED_VOLUME);
}
} else {
#endif //UDF_HDD_SUPPORT
RC = UDFPhSendIOCTL(IOCTL_CDROM_GET_DRIVE_GEOMETRY,DeviceObject,
DiskGeometry,sizeof(DISK_GEOMETRY),
DiskGeometry,sizeof(DISK_GEOMETRY),
TRUE,NULL );
if(RC == STATUS_DEVICE_NOT_READY) {
// probably, the device is really busy, may be by CD/DVD recording
UserPrint((" busy (0)\n"));
try_return(RC);
}
Vcb->BlockSize = (OS_SUCCESS(RC)) ? DiskGeometry->BytesPerSector : 2048;
#ifdef UDF_HDD_SUPPORT
}
}
#endif //UDF_HDD_SUPPORT
#endif //_BROWSE_UDF_
#ifdef UDF_FORMAT_MEDIA
if(fms) {
RC = UDFPhSendIOCTL(IOCTL_CDROM_GET_DRIVE_GEOMETRY,DeviceObject,
DiskGeometry,sizeof(DISK_GEOMETRY),
DiskGeometry,sizeof(DISK_GEOMETRY),
FALSE, NULL );
if(!NT_SUCCESS(RC)) {
RC = UDFPhSendIOCTL(IOCTL_DISK_GET_DRIVE_GEOMETRY,DeviceObject,
DiskGeometry,sizeof(DISK_GEOMETRY),
DiskGeometry,sizeof(DISK_GEOMETRY),
FALSE, NULL );
if(NT_SUCCESS(RC)) {
fms->opt_media = MT_HD;
RC = UDFPhSendIOCTL(IOCTL_DISK_GET_PARTITION_INFO,DeviceObject,
NULL,0,
PartitionInfo,sizeof(PARTITION_INFORMATION)*2,
FALSE, NULL );
if(!NT_SUCCESS(RC)) {
LONG HiOffs=0;
RC = SetFilePointer(DeviceObject->h,0,&HiOffs,FILE_END);
}
}
}
if(RC == STATUS_DEVICE_NOT_READY) {
// probably, the device is really busy, may be by CD/DVD recording
UserPrint((" busy\n"));
try_return(RC );
}
Vcb->BlockSize = (NT_SUCCESS(RC)) ? DiskGeometry->BytesPerSector : 2048;
}
#endif //UDF_FORMAT_MEDIA
// Block size must be an even multiple of 512
switch (Vcb->BlockSize) {
case 2048: Vcb->BlockSizeBits = 11; break;
#ifdef UDF_HDD_SUPPORT
case 512: Vcb->BlockSizeBits = 9; break;
case 1024: Vcb->BlockSizeBits = 10; break;
case 4096: Vcb->BlockSizeBits = 12; break;
case 8192: Vcb->BlockSizeBits = 13; break;
#endif //UDF_HDD_SUPPORT
default:
{
UserPrint(("UDF: Bad block size (%ld)\n", Vcb->BlockSize));
try_return(RC = STATUS_UNSUCCESSFUL);
}
}
#ifdef UDF_HDD_SUPPORT
if(
#ifdef _BROWSE_UDF_
(!fms && (UDFGetDevType(DeviceObject) == FILE_DEVICE_DISK))
||
#endif //_BROWSE_UDF_
#ifdef UDF_FORMAT_MEDIA
(fms && fms->opt_media == MT_HD)
||
#endif //UDF_FORMAT_MEDIA
FALSE ) {
#ifdef UDF_FORMAT_MEDIA
if(fms && !NT_SUCCESS(RC))
try_return(STATUS_UNSUCCESSFUL);
#endif //UDF_FORMAT_MEDIA
Vcb->FirstLBA=0;//(ULONG)(PartitionInfo->StartingOffset.QuadPart >> Vcb->BlockSizeBits);
Vcb->LastPossibleLBA =
Vcb->LastLBA = (uint32)(PartitionInfo->PartitionLength.QuadPart >> Vcb->BlockSizeBits)/* + Vcb->FirstLBA*/ - 1;
} else {
#endif //UDF_HDD_SUPPORT
Vcb->FirstLBA=0;
if(OS_SUCCESS(RC)) {
Vcb->LastLBA = (uint32)(DiskGeometry->Cylinders.QuadPart *
DiskGeometry->TracksPerCylinder *
DiskGeometry->SectorsPerTrack - 1);
if(Vcb->LastLBA == 0x7fffffff) {
Vcb->LastLBA = UDFIsDvdMedia(Vcb) ? DEFAULT_LAST_LBA_DVD : DEFAULT_LAST_LBA_FP_CD;
}
} else {
Vcb->LastLBA = UDFIsDvdMedia(Vcb) ? DEFAULT_LAST_LBA_DVD : DEFAULT_LAST_LBA_FP_CD;
}
Vcb->LastPossibleLBA = Vcb->LastLBA;
#ifdef UDF_HDD_SUPPORT
}
#endif //UDF_HDD_SUPPORT
#ifdef _BROWSE_UDF_
// if(UDFGetDevType(DeviceObject) == FILE_DEVICE_DISK) {
Vcb->WriteBlockSize = PACKETSIZE_UDF*Vcb->BlockSize;
// } else {
// Vcb->WriteBlockSize = PACKETSIZE_UDF*Vcb->BlockSize;
// }
#else //_BROWSE_UDF_
if(fms->opt_media == MT_HD) {
Vcb->WriteBlockSize = Vcb->BlockSize;
} else {
Vcb->WriteBlockSize = PACKETSIZE_UDF*Vcb->BlockSize;
}
#endif //_BROWSE_UDF_
RC = STATUS_SUCCESS;
try_exit: NOTHING;
UDFPrint(("UDFGetBlockSize:\nBlock size is %x, Block size bits %x, Last LBA is %x\n",
Vcb->BlockSize, Vcb->BlockSizeBits, Vcb->LastLBA));
MyFreePool__(PartitionInfo);
MyFreePool__(DiskGeometry);
return RC;
} // end UDFGetBlockSize()
#ifdef _BROWSE_UDF_
OSSTATUS
UDFCheckTrackFPAddressing(
// IN PDEVICE_OBJECT DeviceObject, // the target device object
IN PVCB Vcb, // Volume control block from this DevObj
IN ULONG TrackNum
)
{
OSSTATUS RC = STATUS_SUCCESS;
// OSSTATUS RC2 = STATUS_UNSUCCESSFUL;
uint32 lba=0;
uint32 i;
uint8* Buffer;
// SIZE_T ReadBytes;
uint8 user_data;
ULONG FirstChunkLen = 0;
ULONG NextChunkLen = 0;
ULONG NextChunkLenCount = 0;
ULONG NextChunkLenOth = 0;
ULONG NextChunkLenOthCount = 0;
// ULONG MRW_Offset = 0;
PLL_READ_USER_IN pLLR_in;
PCD_SECTOR_HEADER pHdr;
/* uint8 cMSF[3] = {0,2,0};
uint8 cMSF1[3] = {0,2,1};*/
if(!Vcb->TrackMap) {
Vcb->CompatFlags &= ~UDF_VCB_IC_FP_ADDR_PROBLEM;
return STATUS_SUCCESS;
}
Buffer = (uint8*)DbgAllocatePoolWithTag(NonPagedPool, max(Vcb->BlockSize,
sizeof(LL_READ_USER_IN)+16), 'pNWD');
if(!Buffer)
return STATUS_INSUFFICIENT_RESOURCES;
pLLR_in = (PLL_READ_USER_IN)Buffer;
pHdr = (PCD_SECTOR_HEADER)(Buffer+sizeof(LL_READ_USER_IN));
/* if(Vcb->CompatFlags & UDF_VCB_IC_MRW_ADDR_PROBLEM) {
MRW_Offset = (MRW_DMA_OFFSET/32)*39;
}*/
user_data = 0;
for(i=0; i<=0x200; i++) {
RtlZeroMemory(pLLR_in, sizeof(pLLR_in)+16);
pLLR_in->ExpectedBlkType = ReadCd_BlkType_Any;
pLLR_in->LBA = i;
pLLR_in->NumOfBlocks = 1;
pLLR_in->Flags.Flags = ReadCd_Header_Hdr;
// pLLR_in->UseMFS = FALSE; // already zero
// MOV_MSF(pLLR_in->Starting_MSF, cMSF);
// MOV_MSF(pLLR_in->Ending_MSF, cMSF1);
RtlZeroMemory(pHdr, sizeof(CD_SECTOR_HEADER));
RC = UDFPhSendIOCTL(IOCTL_CDRW_LL_READ, Vcb->TargetDeviceObject,
pLLR_in, sizeof(LL_READ_USER_IN),
pHdr, sizeof(CD_SECTOR_HEADER),
TRUE, NULL );
/* RC = UDFPhReadSynchronous(Vcb->TargetDeviceObject, Buffer, Vcb->BlockSize,
((uint64)(i+MRW_Offset)) << Vcb->BlockSizeBits, &ReadBytes, 0);*/
// skip unreadable
if(!OS_SUCCESS(RC)) {
UDFPrint((" Read error at lba %x\n", i));
continue;
}
// skip strange (damaged ?) blocks
if((pHdr->Mode.Flags & WParam_SubHdr_Mode_Mask) != WParam_SubHdr_Mode1 &&
(pHdr->Mode.Flags & WParam_SubHdr_Mode_Mask) != WParam_SubHdr_Mode2) {
UDFPrint((" Unexpected data type (%x) at lba %x\n", pHdr->Mode.Flags & WParam_SubHdr_Mode_Mask, i));
continue;
}
if((pHdr->Mode.Flags & WParam_SubHdr_Format_Mask) == WParam_SubHdr_Format_UserData &&
!user_data) {
lba = i;
}
/* if(OS_SUCCESS(RC) && !OS_SUCCESS(RC2)) {
lba = i;
}*/
if((pHdr->Mode.Flags & WParam_SubHdr_Format_Mask) != WParam_SubHdr_Format_UserData &&
user_data) {
// if(!OS_SUCCESS(RC) && OS_SUCCESS(RC2)) {
UDFPrint((" %x - %x (%x sectors)\n", lba, i-1, i-lba));
if(!FirstChunkLen) {
FirstChunkLen = i-lba;
} else {
if(!NextChunkLen) {
NextChunkLen = i-lba;
NextChunkLenCount++;
} else {
if(NextChunkLen == i-lba) {
NextChunkLenCount++;
} else {
if((NextChunkLenOth+1) % (NextChunkLen+1)) {
NextChunkLenOth = i-lba;
NextChunkLenOthCount++;
} else {
NextChunkLenCount++;
}
}
}
}
}
user_data = ((pHdr->Mode.Flags & WParam_SubHdr_Format_Mask) == WParam_SubHdr_Format_UserData);
// RC2 = RC;
}
DbgFreePool(Buffer);
if(!NextChunkLenCount && !NextChunkLenOthCount) {
Vcb->CompatFlags &= ~UDF_VCB_IC_FP_ADDR_PROBLEM;
return STATUS_SUCCESS;
}
if(NextChunkLenOthCount > NextChunkLenCount) {
NextChunkLen = NextChunkLenOth;
}
if(NextChunkLen > PACKETSIZE_UDF+7) {
Vcb->CompatFlags &= ~UDF_VCB_IC_FP_ADDR_PROBLEM;
return STATUS_SUCCESS;
}
Vcb->TrackMap[TrackNum].DataParam &= ~TrkInfo_Dat_Mask;
Vcb->TrackMap[TrackNum].DataParam |= TrkInfo_Dat_XA;
Vcb->TrackMap[TrackNum].Flags |= TrackMap_FixFPAddressing;
Vcb->TrackMap[TrackNum].PacketSize = 1;
while(NextChunkLen >> Vcb->TrackMap[TrackNum].PacketSize) {
Vcb->TrackMap[TrackNum].PacketSize++;
}
Vcb->TrackMap[TrackNum].PacketSize = 1 << (Vcb->TrackMap[TrackNum].PacketSize-1);
Vcb->TrackMap[TrackNum].TrackFPOffset = NextChunkLen - FirstChunkLen; // !!!!!
Vcb->TrackMap[TrackNum].PacketFPOffset = Vcb->TrackMap[TrackNum].TrackFPOffset;//0;//NextChunkLenOth - FirstChunkLen;
Vcb->TrackMap[TrackNum].LastLba = (Vcb->TrackMap[TrackNum].LastLba*Vcb->TrackMap[TrackNum].PacketSize) /
(Vcb->TrackMap[TrackNum].PacketSize + 7);
return STATUS_SUCCESS;
} // end UDFCheckTrackFPAddressing()
uint32
UDFFixFPAddress(
IN PVCB Vcb, // Volume control block from this DevObj
IN uint32 Lba
)
{
uint32 i = Vcb->LastReadTrack;
uint32 pk;
uint32 rel;
// if(Vcb->CompatFlags & UDF_VCB_IC_MRW_ADDR_PROBLEM) {
if(Vcb->TrackMap[i].Flags & TrackMap_FixMRWAddressing) {
pk = Lba / MRW_DA_SIZE;
rel = Lba % MRW_DA_SIZE;
Lba = pk*MRW_DMA_SEGMENT_SIZE + rel;
Lba += MRW_DMA_OFFSET;
}
if(Vcb->TrackMap[i].Flags & TrackMap_FixFPAddressing) {
if(Lba < 0x20)
return Lba;
pk = Lba / Vcb->TrackMap[i].PacketSize;
rel = Lba % Vcb->TrackMap[i].PacketSize;
UDFPrint(("FixFPAddr: %x -> %x\n", Lba, pk*(Vcb->TrackMap[i].PacketSize+7) + rel));
return pk*(Vcb->TrackMap[i].PacketSize+7) + rel /*- Vcb->TrackMap[i].PacketFPOffset*/;
}
return Lba;
} // end UDFFixFPAddress()
#endif //_BROWSE_UDF_
/*
detect device driver & try to read disk layout (use all methods)
*/
OSSTATUS
UDFGetDiskInfo(
IN PDEVICE_OBJECT DeviceObject, // the target device object
IN PVCB Vcb // Volume control block from this DevObj
)
{
OSSTATUS RC = STATUS_UNRECOGNIZED_VOLUME;
int8* ioBuf = (int8*)MyAllocatePool__(NonPagedPool,4096);
uint8 MediaType;
PLUN_WRITE_PERF_DESC_USER WPerfDesc;
uint32 i;
// BOOLEAN MRW_problem = FALSE;
uint32 SavedFeatures = 0;
#ifdef UDF_FORMAT_MEDIA
PUDFFmtState fms = Vcb->fms;
#else
#define fms FALSE
#endif //UDF_FORMAT_MEDIA
UDFPrint(("UDFGetDiskInfo\n"));
if(!ioBuf) {
return STATUS_INSUFFICIENT_RESOURCES;
}
_SEH2_TRY {
RC = UDFGetBlockSize(DeviceObject, Vcb);
if(!OS_SUCCESS(RC)) try_return(RC);
// Get lower driver signature
RC = UDFPhSendIOCTL(IOCTL_CDRW_GET_SIGNATURE,DeviceObject,
ioBuf,sizeof(GET_SIGNATURE_USER_OUT),
ioBuf,sizeof(GET_SIGNATURE_USER_OUT),
TRUE,NULL);
if(!OS_SUCCESS(RC)) {
RC = UDFUseStandard(DeviceObject, Vcb);
#ifdef _BROWSE_UDF_
if(!NT_SUCCESS(RC) || fms)
try_return(RC);
// assume Device Recordable for now
goto GetSignatureFailed;
#endif //_BROWSE_UDF_
}
UDFPrint(("UDF: Signature of low driver is : %s \n",
((PGET_SIGNATURE_USER_OUT)(ioBuf))->VendorId));
if(!strncmp( (const char *)(&( ((PGET_SIGNATURE_USER_OUT)(ioBuf))->VendorId[0]) ),
Signature,strlen(Signature) )) {
UDFPrint(("UDF: *****************************************\n"));
UDFPrint(("UDF: ********* Our Device Driver Found ******\n"));
UDFPrint(("UDF: *****************************************\n"));
(Vcb->VCBFlags) |= UDF_VCB_FLAGS_OUR_DEVICE_DRIVER;
#ifndef _BROWSE_UDF_
// reset driver
#ifdef UDF_FORMAT_MEDIA
if(!fms->opt_probe) {
#endif //UDF_FORMAT_MEDIA
UDFResetDeviceDriver(Vcb, Vcb->TargetDeviceObject, FALSE);
// lock it
((PPREVENT_MEDIA_REMOVAL_USER_IN)(ioBuf))->PreventMediaRemoval = TRUE;
UDFPhSendIOCTL( IOCTL_STORAGE_MEDIA_REMOVAL,
DeviceObject,
ioBuf,sizeof(PREVENT_MEDIA_REMOVAL_USER_IN),
NULL,0,
FALSE, NULL);
#ifdef UDF_FORMAT_MEDIA
}
#endif //UDF_FORMAT_MEDIA
#endif //_BROWSE_UDF_
//#else //_BROWSE_UDF_
// get device features
UDFPhSendIOCTL( IOCTL_CDRW_GET_DEVICE_INFO,
DeviceObject,
NULL,0,
ioBuf,sizeof(GET_DEVICE_INFO_USER_OUT),
FALSE,NULL);
Vcb->SavedFeatures =
SavedFeatures = ((PGET_DEVICE_INFO_USER_OUT)ioBuf)->Features;
if(!(SavedFeatures & CDRW_FEATURE_SYNC_ON_WRITE)) {
UDFPrint(("UDFGetDiskInfo: UDF_VCB_IC_NO_SYNCCACHE_AFTER_WRITE\n"));
Vcb->CompatFlags |= UDF_VCB_IC_NO_SYNCCACHE_AFTER_WRITE;
}
if(!(SavedFeatures & CDRW_FEATURE_FORCE_SYNC_BEFORE_READ)) {
UDFPrint(("UDFGetDiskInfo: UDF_VCB_IC_SYNCCACHE_BEFORE_READ\n"));
Vcb->CompatFlags |= UDF_VCB_IC_SYNCCACHE_BEFORE_READ;
}
if(SavedFeatures & CDRW_FEATURE_BAD_RW_SEEK) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_BAD_RW_SEEK\n"));
Vcb->CompatFlags |= UDF_VCB_IC_BAD_RW_SEEK;
}
// we must check if this is FP-formatted disk in old devices
// independently of MediaType they report
if(SavedFeatures & CDRW_FEATURE_FP_ADDRESSING_PROBLEM) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_FP_ADDRESSING_PROBLEM ?\n"));
Vcb->CompatFlags |= UDF_VCB_IC_FP_ADDR_PROBLEM;
}
if(SavedFeatures & CDRW_FEATURE_MRW_ADDRESSING_PROBLEM) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_MRW_ADDRESSING_PROBLEM ?\n"));
}
if(SavedFeatures & CDRW_FEATURE_FORCE_SYNC_ON_WRITE) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_FORCE_SYNC_ON_WRITE\n"));
Vcb->VCBFlags |= UDF_VCB_FLAGS_FORCE_SYNC_CACHE;
}
if(SavedFeatures & CDRW_FEATURE_BAD_DVD_LAST_LBA) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_BAD_DVD_LAST_LBA\n"));
Vcb->CompatFlags |= UDF_VCB_IC_BAD_DVD_LAST_LBA;
}
if(SavedFeatures & CDRW_FEATURE_STREAMING) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_STREAMING\n"));
}
if(SavedFeatures & CDRW_FEATURE_OPC) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_OPC -> assume OPCNum=1\n"));
Vcb->OPCNum = 1;
}
#ifdef UDF_FORMAT_MEDIA
if(SavedFeatures & CDRW_FEATURE_FULL_BLANK_ON_FORMAT) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_FULL_BLANK_ON_FORMAT\n"));
if((fms->opt_probe || fms->opt_smart_f)/* &&
(fms->format_media && fms->blank_media*/) {
UDFPrint(("UDFGetDiskInfo: force Full Erase\n"));
fms->opt_qblank = FALSE;
}
}
#endif //UDF_FORMAT_MEDIA
#ifdef _BROWSE_UDF_
// get device buffer size
RC = UDFPhSendIOCTL( IOCTL_CDRW_BUFFER_CAPACITY,
DeviceObject,
NULL,0,
ioBuf,sizeof(BUFFER_CAPACITY_BLOCK_USER_OUT),
FALSE,NULL);
if(NT_SUCCESS(RC)) {
Vcb->CdrwBufferSize = ((PBUFFER_CAPACITY_BLOCK_USER_OUT)ioBuf)->BufferLength;
} else {
Vcb->CdrwBufferSize = 0;
}
UDFPrint(("UDFGetDiskInfo: CdrwBufferSize = %dKb\n", Vcb->CdrwBufferSize / 1024));
Vcb->CdrwBufferSizeCounter = 0;
#endif //_BROWSE_UDF_
// get media type
RC = UDFPhSendIOCTL(IOCTL_CDRW_GET_MEDIA_TYPE,DeviceObject,
NULL,0,ioBuf,sizeof(GET_MEDIA_TYPE_USER_OUT),
FALSE, NULL);
if(!OS_SUCCESS(RC)) goto Try_FullToc;
Vcb->MediaType =
MediaType = ((PGET_MEDIA_TYPE_USER_OUT)ioBuf)->MediaType;
UDFPrint(("UDFGetDiskInfo: MediaType %x\n", MediaType));
#ifndef UDF_FORMAT_MEDIA
// we shall ignore audio-disks
switch(MediaType) {
case MediaType_120mm_CDROM_AudioOnly:
case MediaType_80mm_CDROM_AudioOnly:
case MediaType_120mm_CDR_AudioOnly:
case MediaType_80mm_CDR_AudioOnly:
case MediaType_120mm_CDRW_AudioOnly:
case MediaType_80mm_CDRW_AudioOnly:
// case :
UDFPrint(("UDFGetDiskInfo: we shall ignore audio-disks...\n"));
try_return(RC = STATUS_UNRECOGNIZED_VOLUME);
}
#endif //UDF_FORMAT_MEDIA
UDFPrint(("UDFGetDiskInfo: Check DVD-disks...\n"));
RC = UDFPhSendIOCTL(IOCTL_CDRW_GET_MEDIA_TYPE_EX,DeviceObject,
NULL,0,ioBuf,sizeof(GET_MEDIA_TYPE_EX_USER_OUT),
FALSE, NULL);
if(!OS_SUCCESS(RC)) goto Try_FullToc;
Vcb->MediaClassEx =
MediaType = (((PGET_MEDIA_TYPE_EX_USER_OUT)ioBuf)->MediaClass);
UDFPrint(("UDFGetDiskInfo: MediaClassEx %x\n", MediaType));
#ifdef _BROWSE_UDF_
if(!fms) {
switch(MediaType) {
case CdMediaClass_CDR:
case CdMediaClass_DVDR:
case CdMediaClass_DVDpR:
case CdMediaClass_HD_DVDR:
case CdMediaClass_BDR:
UDFPrint(("UDFGetDiskInfo: MediaClass R\n"));
Vcb->MediaType = MediaType_UnknownSize_CDR;
break;
case CdMediaClass_CDRW:
if(SavedFeatures & CDRW_FEATURE_MRW_ADDRESSING_PROBLEM) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_MRW_ADDRESSING_PROBLEM on CD-RW\n"));
Vcb->CompatFlags |= UDF_VCB_IC_MRW_ADDR_PROBLEM;
}
case CdMediaClass_DVDRW:
case CdMediaClass_DVDpRW:
case CdMediaClass_DVDRAM:
case CdMediaClass_HD_DVDRW:
case CdMediaClass_HD_DVDRAM:
case CdMediaClass_BDRE:
UDFPrint(("UDFGetDiskInfo: MediaClass RW\n"));
Vcb->MediaType = MediaType_UnknownSize_CDRW;
break;
case CdMediaClass_CDROM:
case CdMediaClass_DVDROM:
case CdMediaClass_HD_DVDROM:
case CdMediaClass_BDROM:
UDFPrint(("UDFGetDiskInfo: MediaClass ROM\n"));
Vcb->MediaType = MediaType_Unknown;
// Vcb->MediaType = MediaType_UnknownSize_CDROM;
break;
default:
UDFPrint(("UDFGetDiskInfo: MediaClass Unknown\n"));
Vcb->MediaType = MediaType_Unknown;
break;
}
MediaType = Vcb->MediaType;
}
#endif //_BROWSE_UDF_
#ifdef UDF_FORMAT_MEDIA
if(fms) {
switch(MediaType) {
case CdMediaClass_CDR:
UDFPrint(("CdMediaClass_CDR\n"));
MediaType = MediaType_UnknownSize_CDR;
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_CDR;
break;
case CdMediaClass_DVDR:
UDFPrint(("CdMediaClass_DVDR -> MediaType_UnknownSize_CDR\n"));
MediaType = MediaType_UnknownSize_CDR;
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDR;
break;
case CdMediaClass_DVDpR:
UDFPrint(("CdMediaClass_DVDpR -> MediaType_UnknownSize_CDR\n"));
MediaType = MediaType_UnknownSize_CDR;
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDpR;
break;
case CdMediaClass_HD_DVDR:
UDFPrint(("CdMediaClass_HD_DVDR -> MediaType_UnknownSize_CDR\n"));
MediaType = MediaType_UnknownSize_CDR;
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDR;
break;
case CdMediaClass_BDR:
UDFPrint(("CdMediaClass_BDR -> MediaType_UnknownSize_CDR\n"));
MediaType = MediaType_UnknownSize_CDR;
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDR;
break;
case CdMediaClass_CDRW:
UDFPrint(("CdMediaClass_CDRW\n"));
MediaType = MediaType_UnknownSize_CDRW;
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_CDRW;
if(SavedFeatures & CDRW_FEATURE_MRW_ADDRESSING_PROBLEM) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_MRW_ADDRESSING_PROBLEM on CD-RW\n"));
Vcb->CompatFlags |= UDF_VCB_IC_MRW_ADDR_PROBLEM;
}
break;
case CdMediaClass_DVDRW:
UDFPrint((" CdMediaClass_DVDRW -> MediaType_UnknownSize_CDRW\n"));
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDRW;
MediaType = MediaType_UnknownSize_CDRW;
break;
case CdMediaClass_DVDpRW:
UDFPrint((" CdMediaClass_DVDpRW -> MediaType_UnknownSize_CDRW\n"));
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDpRW;
MediaType = MediaType_UnknownSize_CDRW;
break;
case CdMediaClass_DVDRAM:
UDFPrint((" CdMediaClass_DVDRAM -> MediaType_UnknownSize_CDRW\n"));
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDRAM;
MediaType = MediaType_UnknownSize_CDRW;
break;
case CdMediaClass_HD_DVDRW:
UDFPrint((" CdMediaClass_HD_DVDRW -> MediaType_UnknownSize_CDRW\n"));
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDRW;
MediaType = MediaType_UnknownSize_CDRW;
break;
case CdMediaClass_HD_DVDRAM:
UDFPrint((" CdMediaClass_HD_DVDRAM -> MediaType_UnknownSize_CDRW\n"));
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDRAM;
MediaType = MediaType_UnknownSize_CDRW;
break;
case CdMediaClass_BDRE:
UDFPrint((" CdMediaClass_BDRE -> MediaType_UnknownSize_CDRW\n"));
if(fms->opt_media == MT_AUTO)
fms->opt_media = MT_DVDRW;
MediaType = MediaType_UnknownSize_CDRW;
break;
case CdMediaClass_NoDiscPresent:
UDFPrint((" CdMediaClass_NoDiscPresent -> MediaType_NoDiscPresent\n"));
MediaType = MediaType_NoDiscPresent;
fms->opt_media = MT_none;
break;
case CdMediaClass_DoorOpen:
UDFPrint((" CdMediaClass_DoorOpen -> MediaType_DoorOpen\n"));
MediaType = MediaType_DoorOpen;
fms->opt_media = MT_none;
break;
default:
UDFPrint((" MediaType_Unknown\n"));
MediaType = MediaType_Unknown;
break;
}
if(!apply_force_r(fms)) {
my_exit(fms, MKUDF_CANT_APPLY_R);
}
}
#endif //UDF_FORMAT_MEDIA
Vcb->DVD_Mode = (((PGET_MEDIA_TYPE_EX_USER_OUT)ioBuf)->MediaClassEx == CdMediaClassEx_DVD);
Vcb->PhMediaCapFlags = ((PGET_MEDIA_TYPE_EX_USER_OUT)ioBuf)->CapFlags;
Vcb->WriteParamsReq = (Vcb->PhMediaCapFlags & CdCapFlags_WriteParamsReq) ? TRUE : FALSE;
if(Vcb->DVD_Mode &&
!(Vcb->PhMediaCapFlags & CdCapFlags_RandomWritable)) {
UDFPrint(("UDFGetDiskInfo: DVD && !CdCapFlags_RandomWritable\n"));
UDFPrint((" Read-only volume\n"));
// BrutePoint();
#ifndef UDF_CDRW_EMULATION_ON_ROM
Vcb->VCBFlags |= UDF_VCB_FLAGS_VOLUME_READ_ONLY;
#endif
}
#ifdef UDF_FORMAT_MEDIA
if(fms) {
if((MediaType == MediaType_NoDiscPresent) ||
(MediaType == MediaType_DoorOpen)) {
UserPrint(("No media in device\n"));
my_exit(fms, MKUDF_NO_MEDIA_IN_DEVICE);
}
}
#endif //UDF_FORMAT_MEDIA
if(!Vcb->WriteParamsReq) {
UDFPrint(("UDFGetDiskInfo: do not use WriteParams\n"));
}
if(Vcb->PhMediaCapFlags & CdCapFlags_Cav) {
UDFPrint(("UDFGetDiskInfo: Use CAV (1)\n"));
Vcb->VCBFlags |= UDF_VCB_FLAGS_USE_CAV;
}
#ifdef _BROWSE_UDF_
if(!fms) {
// check if this device is capable to write on such media
if(UDFIsDvdMedia(Vcb)) {
//RC =
UDFPrint(("UDFGetDiskInfo: update defaulted LastLBA\n"));
UDFGetBlockSize(DeviceObject,Vcb);
//if(!OS_SUCCESS(RC)) goto Try_FullToc;
} else {
if((SavedFeatures & CDRW_FEATURE_MRW_ADDRESSING_PROBLEM) &&
(SavedFeatures & UDF_VCB_IC_FP_ADDR_PROBLEM)) {
UDFPrint(("UDFGetDiskInfo: CDRW_FEATURE_MRW_ADDRESSING_PROBLEM on old CD-ROM\n"));
Vcb->CompatFlags |= UDF_VCB_IC_MRW_ADDR_PROBLEM;
}
}
}
#endif //_BROWSE_UDF_
/*#ifdef UDF_FORMAT_MEDIA
if(fms) {
if(MediaType == CdMediaClass_DVDRW) {
UserPrint(("Not empty media. Erase required.\n"));
my_exit(fms, MKUDF_BLANK_FORMAT_REQUIRED);
}
}
#endif //UDF_FORMAT_MEDIA*/
#define cap ((PGET_CAPABILITIES_3_USER_OUT)ioBuf)
// get device capabilities
RC = UDFPhSendIOCTL(IOCTL_CDRW_GET_CAPABILITIES,DeviceObject,
NULL,0,ioBuf,sizeof(GET_CAPABILITIES_3_USER_OUT),
FALSE, NULL);
if(!OS_SUCCESS(RC)) goto Try_FullToc;
// check if this device is capable to write on such media
RC = UDFPhSendIOCTL(IOCTL_DISK_IS_WRITABLE,DeviceObject,
NULL,0,NULL,0,FALSE, NULL);
if(RC != STATUS_SUCCESS) {
UDFPrint(("IS_WRITABLE - false, doing additional check...\n"));
if( ((MediaType >= MediaType_UnknownSize_CDRW) && !(cap->WriteCap & DevCap_write_cd_rw)) ||
((MediaType >= MediaType_UnknownSize_CDR) && !(cap->WriteCap & DevCap_write_cd_r)) ||
(MediaType < MediaType_UnknownSize_CDR) ) {
UserPrint(("Hardware Read-only volume\n"));
#ifndef UDF_CDRW_EMULATION_ON_ROM
Vcb->VCBFlags |= UDF_VCB_FLAGS_VOLUME_READ_ONLY;
#endif //UDF_CDRW_EMULATION_ON_ROM
#ifdef UDF_FORMAT_MEDIA
if(fms && !fms->opt_read_iso)
my_exit(fms, MKUDF_HW_READ_ONLY);
#endif //UDF_FORMAT_MEDIA
}
} else {
UDFPrint(("Writable disk\n"));
}
Vcb->MaxWriteSpeed = cap->MaximumWriteSpeedSupported;
Vcb->MaxReadSpeed = cap->MaximumSpeedSupported;
if(cap->PageLength >= (sizeof(GET_CAPABILITIES_3_USER_OUT)-2)) {
Vcb->CurSpeed = max(cap->CurrentSpeed, cap->CurrentWriteSpeed3);
if(cap->LunWPerfDescriptorCount && cap->LunWPerfDescriptorCount != 0xffff) {
ULONG n;
UDFPrint(("Write performance descriptor(s) found: %x\n", cap->LunWPerfDescriptorCount));
n = (4096 - sizeof(GET_CAPABILITIES_3_USER_OUT)) / sizeof(LUN_WRITE_PERF_DESC_USER);
n = min(n, cap->LunWPerfDescriptorCount);
// get device capabilities
RC = UDFPhSendIOCTL(IOCTL_CDRW_GET_CAPABILITIES,DeviceObject,
ioBuf,sizeof(GET_CAPABILITIES_3_USER_OUT)+n*sizeof(LUN_WRITE_PERF_DESC_USER),
ioBuf,sizeof(GET_CAPABILITIES_3_USER_OUT)+n*sizeof(LUN_WRITE_PERF_DESC_USER),
TRUE,NULL);
if(OS_SUCCESS(RC)) {
WPerfDesc = (PLUN_WRITE_PERF_DESC_USER)(ioBuf + sizeof(GET_CAPABILITIES_3_USER_OUT));
n = FALSE;
for(i = 0; i<n; i++) {
if((WPerfDesc[i].RotationControl & LunWPerf_RotCtrl_Mask) == LunWPerf_RotCtrl_CAV) {
Vcb->VCBFlags |= UDF_VCB_FLAGS_USE_CAV;
if(!n) {
Vcb->CurSpeed = WPerfDesc[i].WriteSpeedSupported;
n = TRUE;
UDFPrint(("Use CAV\n"));
} else {
Vcb->CurSpeed = max(WPerfDesc[i].WriteSpeedSupported, Vcb->CurSpeed);
}
UDFPrint(("supports speed %dX\n", Vcb->CurSpeed/176));
//break;
}
}
if(n) {
UDFPrint(("Set r/w speeds to %dX\n", Vcb->CurSpeed/176));
Vcb->MaxWriteSpeed =
Vcb->MaxReadSpeed = Vcb->CurSpeed;
}
}
}
} else {
Vcb->CurSpeed = max(cap->CurrentSpeed, cap->CurrentWriteSpeed);
}
UDFPrint((" Speeds r/w %dX/%dX\n", Vcb->CurSpeed/176, cap->CurrentWriteSpeed/176));
if(Vcb->VCBFlags & UDF_VCB_FLAGS_USE_CAV) {
// limit both read & write speed to last write speed for CAV mode
// some drives damage data when speed is adjusted during recording process
// even in packet mode
UDFSetSpeeds(Vcb);
}
UDFSetCaching(Vcb);
#undef cap
#ifdef UDF_FORMAT_MEDIA
if(fms) {
if( (fms->auto_media || (fms->opt_media == MT_AUTO)) &&
(fms->opt_media < MT_DVDR) ) {
if(MediaType < MediaType_UnknownSize_CDRW) {
fms->opt_media = MT_CDR;
} else {
fms->opt_media = MT_CDRW;
}
}
if(!apply_force_r(fms)) {
my_exit(fms, MKUDF_CANT_APPLY_R);
}
}
#endif //UDF_FORMAT_MEDIA
RC = UDFReadDiscTrackInfo(DeviceObject, Vcb);
if(!OS_SUCCESS(RC)) {
// may be we have a CD-ROM device
Try_FullToc:
UDFPrint(("Hardware Read-only volume (2)\n"));
// BrutePoint();
#ifndef UDF_CDRW_EMULATION_ON_ROM
Vcb->VCBFlags |= UDF_VCB_FLAGS_VOLUME_READ_ONLY;
#endif
RC = UDFReadAndProcessFullToc(DeviceObject, Vcb);
if(!OS_SUCCESS(RC)) {
RC = UDFUseStandard(DeviceObject,Vcb);
if(!OS_SUCCESS(RC)) try_return(RC);
}
}
} else {
#ifdef _BROWSE_UDF_
GetSignatureFailed:
#endif
RC = UDFUseStandard(DeviceObject, Vcb);
if(!OS_SUCCESS(RC)) try_return(RC);
}
try_exit: NOTHING;
} _SEH2_FINALLY {
if(ioBuf) MyFreePool__(ioBuf);
if(UDFIsDvdMedia(Vcb) &&
(Vcb->CompatFlags & UDF_VCB_IC_BAD_DVD_LAST_LBA) &&
(Vcb->VCBFlags & UDF_VCB_FLAGS_VOLUME_READ_ONLY) &&
Vcb->LastLBA &&
(Vcb->LastLBA < DEFAULT_LAST_LBA_DVD)) {
UDFPrint(("UDF: Bad DVD last LBA %x, fixup!\n", Vcb->LastLBA));
Vcb->LastLBA = DEFAULT_LAST_LBA_DVD;
Vcb->NWA = 0;
}
if(UDFIsDvdMedia(Vcb) && !Vcb->FirstLBA && !Vcb->LastPossibleLBA) {
UDFPrint(("UDF: Empty DVD. Use bogus values for now\n"));
Vcb->LastPossibleLBA = DEFAULT_LAST_LBA_DVD;
Vcb->LastLBA = 0;
}
if((Vcb->LastPossibleLBA & 0x80000000) || (Vcb->LastPossibleLBA < Vcb->LastLBA)) {
UDFPrint(("UDF: bad LastPossibleLBA %x -> %x\n", Vcb->LastPossibleLBA, Vcb->LastLBA));
Vcb->LastPossibleLBA = Vcb->LastLBA;
}
if(!Vcb->WriteBlockSize)
Vcb->WriteBlockSize = PACKETSIZE_UDF*Vcb->BlockSize;
#ifdef _BROWSE_UDF_
if(Vcb->TrackMap) {
if(Vcb->TrackMap[Vcb->LastTrackNum].LastLba > Vcb->NWA) {
if(Vcb->NWA) {
if(Vcb->TrackMap[Vcb->LastTrackNum].DataParam & TrkInfo_FP) {
Vcb->LastLBA = Vcb->NWA-1;
} else {
Vcb->LastLBA = Vcb->NWA-7-1;
}
}
} else {
if((Vcb->LastTrackNum > 1) &&
(Vcb->TrackMap[Vcb->LastTrackNum-1].FirstLba >= Vcb->TrackMap[Vcb->LastTrackNum-1].LastLba)) {
Vcb->LastLBA = Vcb->TrackMap[Vcb->LastTrackNum-1].LastLba;
}
}
}
for(i=0; i<32; i++) {
if(!(Vcb->LastPossibleLBA >> i))
break;
}
if(i > 20) {
Vcb->WCacheBlocksPerFrameSh = max(Vcb->WCacheBlocksPerFrameSh, (2*i)/5+2);
Vcb->WCacheBlocksPerFrameSh = min(Vcb->WCacheBlocksPerFrameSh, 16);
}
if(Vcb->CompatFlags & UDF_VCB_IC_FP_ADDR_PROBLEM) {
// Check first 0x200 blocks
UDFCheckTrackFPAddressing(Vcb, Vcb->FirstTrackNum);
// if we really have such a problem, fix LastLBA
if(Vcb->CompatFlags & UDF_VCB_IC_FP_ADDR_PROBLEM) {
UDFPrint(("UDF: Fix LastLBA: %x -> %x\n", Vcb->LastLBA, (Vcb->LastLBA*32) / 39));
Vcb->LastLBA = (Vcb->LastLBA*32) / 39;
}
}
#endif //_BROWSE_UDF_
if(Vcb->VCBFlags & UDF_VCB_FLAGS_VOLUME_READ_ONLY) {
if(!Vcb->BlankCD && Vcb->MediaType != MediaType_UnknownSize_CDRW) {
UDFPrint(("UDFGetDiskInfo: R/O+!Blank+!RW -> !RAW\n"));
Vcb->VCBFlags &= ~UDF_VCB_FLAGS_RAW_DISK;
} else {
UDFPrint(("UDFGetDiskInfo: Blank or RW\n"));
}
}
UDFPrint(("UDF: ------------------------------------------\n"));
UDFPrint(("UDF: Media characteristics\n"));
UDFPrint(("UDF: Last session: %d\n",Vcb->LastSession));
UDFPrint(("UDF: First track in first session: %d\n",Vcb->FirstTrackNum));
UDFPrint(("UDF: First track in last session: %d\n",Vcb->FirstTrackNumLastSes));
UDFPrint(("UDF: Last track in last session: %d\n",Vcb->LastTrackNum));
UDFPrint(("UDF: First LBA in first session: %x\n",Vcb->FirstLBA));
UDFPrint(("UDF: First LBA in last session: %x\n",Vcb->FirstLBALastSes));
UDFPrint(("UDF: Last LBA in last session: %x\n",Vcb->LastLBA));
UDFPrint(("UDF: First writable LBA (NWA) in last session: %x\n",Vcb->NWA));
UDFPrint(("UDF: Last available LBA beyond end of last session: %x\n",Vcb->LastPossibleLBA));
UDFPrint(("UDF: blocks per frame: %x\n",1 << Vcb->WCacheBlocksPerFrameSh));
UDFPrint(("UDF: Flags: %s%s\n",
Vcb->VCBFlags & UDF_VCB_FLAGS_RAW_DISK ? "RAW " : "",
Vcb->VCBFlags & UDF_VCB_FLAGS_VOLUME_READ_ONLY ? "R/O " : "WR "
));
UDFPrint(("UDF: ------------------------------------------\n"));
#ifdef UDF_FORMAT_MEDIA
if(fms && fms->opt_disk_info) {
UserPrint(("------------------------------------------\n"));
UserPrint(("Media characteristics:\n"));
UserPrint((" First writable LBA (NWA) in last session: %x\n",Vcb->NWA));
UserPrint((" Last available LBA beyond end of last session: %x\n",Vcb->LastPossibleLBA));
UserPrint(("------------------------------------------\n"));
}
#endif //UDF_FORMAT_MEDIA
} _SEH2_END;
UDFPrint(("UDFGetDiskInfo: %x\n", RC));
return(RC);
} // end UDFGetDiskInfo()
//#ifdef _BROWSE_UDF_
OSSTATUS
UDFPrepareForReadOperation(
IN PVCB Vcb,
IN uint32 Lba,
IN uint32 BCount
)
{
if( (Vcb->FsDeviceType != FILE_DEVICE_CD_ROM_FILE_SYSTEM) ) {
Vcb->VCBFlags &= ~UDF_VCB_LAST_WRITE;
return STATUS_SUCCESS;
}
uint32 i = Vcb->LastReadTrack;
BOOLEAN speed_changed = FALSE;
#ifdef _BROWSE_UDF_
PUCHAR tmp;
OSSTATUS RC;
SIZE_T ReadBytes;
#endif //_BROWSE_UDF_
#ifdef _UDF_STRUCTURES_H_
if(Vcb->BSBM_Bitmap) {
ULONG i;
for(i=0; i<BCount; i++) {
if(UDFGetBit((uint32*)(Vcb->BSBM_Bitmap), Lba+i)) {
UDFPrint(("R: Known BB @ %#x\n", Lba));
//return STATUS_FT_WRITE_RECOVERY; // this shall not be treated as error and
// we shall get IO request to BAD block
return STATUS_DEVICE_DATA_ERROR;
}
}
}
#endif //_UDF_STRUCTURES_H_
if(!UDFIsDvdMedia(Vcb) &&
(Vcb->VCBFlags & UDF_VCB_LAST_WRITE) &&
!(Vcb->VCBFlags & UDF_VCB_FLAGS_NO_SYNC_CACHE) &&
!(Vcb->CompatFlags & UDF_VCB_IC_NO_SYNCCACHE_AFTER_WRITE)){
// OSSTATUS RC;
RC = UDFSyncCache(Vcb);
}
if( (Vcb->VCBFlags & UDF_VCB_LAST_WRITE) &&
!(Vcb->VCBFlags & UDF_VCB_FLAGS_NO_SYNC_CACHE) &&
#ifndef UDF_FORMAT_MEDIA
(Vcb->CompatFlags & UDF_VCB_IC_SYNCCACHE_BEFORE_READ) &&
#endif //UDF_FORMAT_MEDIA
TRUE)
{
// OSSTATUS RC;
UDFSyncCache(Vcb);
}
#ifdef _BROWSE_UDF_
if(!UDFIsDvdMedia(Vcb)) {
// limit read speed after write operation
// to avoid performance degrade durring speed-up/down
// on read/write mode switching
if(Vcb->VCBFlags & UDF_VCB_FLAGS_USE_CAV) {
// limit both read & write speed to last write speed for CAV mode
// some drives damage data when speed is adjusted during recording process
// even in packet mode
if(Vcb->CurSpeed != Vcb->MaxWriteSpeed ||
Vcb->CurSpeed != Vcb->MaxReadSpeed) {
Vcb->CurSpeed = Vcb->MaxWriteSpeed;
speed_changed = TRUE;
}
} else
if(Vcb->VCBFlags & UDF_VCB_LAST_WRITE) {
// limit read speed to last write speed
if(Vcb->CurSpeed > Vcb->MaxWriteSpeed) {
Vcb->CurSpeed = Vcb->MaxWriteSpeed;
speed_changed = TRUE;
}
} else
if(Vcb->CurSpeed < Vcb->MaxReadSpeed ) {
// increment read speed (+1X)
Vcb->CurSpeed += 176/1;
speed_changed = TRUE;
}
if(Vcb->CurSpeed > Vcb->MaxReadSpeed) {
Vcb->CurSpeed = Vcb->MaxReadSpeed;
}
// send speed limits to drive
if(speed_changed) {
RtlZeroMemory(&(Vcb->SpeedBuf), sizeof(SET_CD_SPEED_EX_USER_IN));
Vcb->SpeedBuf.ReadSpeed = Vcb->CurSpeed;
Vcb->SpeedBuf.WriteSpeed = Vcb->MaxWriteSpeed;
if(Vcb->VCBFlags & UDF_VCB_FLAGS_USE_CAV) {
Vcb->SpeedBuf.RotCtrl = CdSpeed_RotCtrl_CAV;
}
UDFPrint((" UDFPrepareForReadOperation: set speed to %s %dX/%dX\n",
(Vcb->VCBFlags & UDF_VCB_FLAGS_USE_CAV) ? "CAV" : "CLV",
Vcb->SpeedBuf.ReadSpeed,
Vcb->SpeedBuf.WriteSpeed));
UDFPhSendIOCTL(IOCTL_CDRW_SET_SPEED,
Vcb->TargetDeviceObject,
&(Vcb->SpeedBuf),sizeof(SET_CD_SPEED_EX_USER_IN),
NULL,0,TRUE,NULL);
}
}
if(UDFIsDvdMedia(Vcb))
return STATUS_SUCCESS;
if(Vcb->LastReadTrack &&
((Vcb->TrackMap[i].FirstLba <= Lba) || (Vcb->TrackMap[i].FirstLba & 0x80000000)) &&
(Vcb->TrackMap[i].LastLba >= Lba)) {
check_for_data_track:
// check track mode (Mode1/XA)
switch((Vcb->TrackMap[i].DataParam & TrkInfo_Dat_Mask)) {
case TrkInfo_Dat_Mode1: // Mode1
case TrkInfo_Dat_XA: // XA Mode2
case TrkInfo_Dat_Unknown: // for some stupid irons
UDFSetMRWMode(Vcb);
break;
default:
Vcb->IncrementalSeekState = INCREMENTAL_SEEK_NONE;
return STATUS_INVALID_PARAMETER;
}
} else {
for(i=Vcb->FirstTrackNum; i<=Vcb->LastTrackNum; i++) {
if(((Vcb->TrackMap[i].FirstLba > Lba) && !(Vcb->TrackMap[i].FirstLba & 0x80000000)) ||
(Vcb->TrackMap[i].LastLba < Lba))
continue;
Vcb->LastReadTrack = i;
goto check_for_data_track;
}
Vcb->LastReadTrack = 0;
}
if(Vcb->IncrementalSeekState != INCREMENTAL_SEEK_WORKAROUND) {
Vcb->IncrementalSeekState = INCREMENTAL_SEEK_NONE;
return STATUS_SUCCESS;
}
UDFPrint((" UDFPrepareForReadOperation: seek workaround...\n"));
Vcb->IncrementalSeekState = INCREMENTAL_SEEK_DONE;
tmp = (PUCHAR)DbgAllocatePoolWithTag(NonPagedPool, Vcb->BlockSize, 'bNWD');
if(!tmp) {
Vcb->IncrementalSeekState = INCREMENTAL_SEEK_NONE;
return STATUS_INSUFFICIENT_RESOURCES;
}
for(i=0x1000; i<=Lba; i+=0x1000) {
RC = UDFPhReadSynchronous(Vcb->TargetDeviceObject, tmp, Vcb->BlockSize,
((uint64)UDFFixFPAddress(Vcb,i)) << Vcb->BlockSizeBits, &ReadBytes, 0);
UDFPrint((" seek workaround, LBA %x, status %x\n", i, RC));
}
DbgFreePool(tmp);
#endif //_BROWSE_UDF_
return STATUS_SUCCESS;
} // end UDFPrepareForReadOperation()
//#endif //_BROWSE_UDF_
void
UDFUpdateNWA(
PVCB Vcb,
uint32 LBA, // physical
uint32 BCount,
OSSTATUS RC
)
{
#ifndef UDF_READ_ONLY_BUILD
#ifdef _BROWSE_UDF_
if(!OS_SUCCESS(RC)) {
return;
}
if(!Vcb->CDR_Mode) {
if((Vcb->MediaClassEx == CdMediaClass_DVDRW ||
Vcb->MediaClassEx == CdMediaClass_DVDpRW ||
Vcb->MediaClassEx == CdMediaClass_DVDRAM ||
Vcb->MRWStatus == DiscInfo_BGF_Interrupted ||
Vcb->MRWStatus == DiscInfo_BGF_InProgress)
&& (LBA+BCount-1) > Vcb->LastLBA) {
ASSERT(Vcb->NWA > Vcb->LastLBA);
Vcb->NWA = LBA+BCount;
Vcb->LastLBA = Vcb->NWA-1;
}
if(Vcb->VCBFlags & UDF_VCB_FLAGS_FORCE_SYNC_CACHE)
goto sync_cache;
/* if(Vcb->CdrwBufferSize) {
Vcb->CdrwBufferSizeCounter += BCount * 2048;
if(Vcb->CdrwBufferSizeCounter >= Vcb->CdrwBufferSize + 2*2048) {
UDFPrint((" UDFUpdateNWA: buffer is full, sync...\n"));
Vcb->CdrwBufferSizeCounter = 0;
goto sync_cache;
}
}*/
if(Vcb->SyncCacheState == SYNC_CACHE_RECOVERY_RETRY) {
Vcb->VCBFlags |= UDF_VCB_FLAGS_FORCE_SYNC_CACHE;
}
Vcb->SyncCacheState = SYNC_CACHE_RECOVERY_NONE;
return;
}
if(Vcb->LastLBA < (LBA+BCount))
Vcb->LastLBA = LBA+BCount;
if(Vcb->NWA)
Vcb->NWA+=BCount+7;
sync_cache:
if(!(Vcb->CompatFlags & UDF_VCB_IC_NO_SYNCCACHE_AFTER_WRITE)) {
UDFPrint((" UDFUpdateNWA: syncing...\n"));
RC = UDFSyncCache(Vcb);
}
#endif //_BROWSE_UDF_
#endif //UDF_READ_ONLY_BUILD
} // end UDFUpdateNWA()
/*
This routine reads physical sectors
*/
/*OSSTATUS
UDFReadSectors(
IN PVCB Vcb,
IN BOOLEAN Translate, // Translate Logical to Physical
IN uint32 Lba,
IN uint32 BCount,
OUT int8* Buffer,
OUT PSIZE_T ReadBytes
)
{
if(Vcb->FastCache.ReadProc && (KeGetCurrentIrql() < DISPATCH_LEVEL)) {
return WCacheReadBlocks__(&(Vcb->FastCache), Vcb, Buffer, Lba, BCount, ReadBytes);
}
return UDFTRead(Vcb, Buffer, BCount*Vcb->BlockSize, Lba, ReadBytes);
} // end UDFReadSectors()*/
#ifdef _BROWSE_UDF_
/*
This routine reads physical sectors
*/
OSSTATUS
UDFReadInSector(
IN PVCB Vcb,
IN BOOLEAN Translate, // Translate Logical to Physical
IN uint32 Lba,
IN uint32 i, // offset in sector
IN uint32 l, // transfer length
IN BOOLEAN Direct, // Disable access to non-cached data
OUT int8* Buffer,
OUT PSIZE_T ReadBytes
)
{
int8* tmp_buff;
OSSTATUS status;
SIZE_T _ReadBytes;
(*ReadBytes) = 0;
if(WCacheIsInitialized__(&(Vcb->FastCache)) && (KeGetCurrentIrql() < DISPATCH_LEVEL)) {
status = WCacheDirect__(&(Vcb->FastCache), Vcb, Lba, FALSE, &tmp_buff, Direct);
if(OS_SUCCESS(status)) {
(*ReadBytes) += l;
RtlCopyMemory(Buffer, tmp_buff+i, l);
}
if(!Direct) WCacheEODirect__(&(Vcb->FastCache), Vcb);
} else {
if(Direct) {
return STATUS_INVALID_PARAMETER;
}
tmp_buff = (int8*)MyAllocatePool__(NonPagedPool, Vcb->BlockSize);
if(!tmp_buff) return STATUS_INSUFFICIENT_RESOURCES;
status = UDFReadSectors(Vcb, Translate, Lba, 1, FALSE, tmp_buff, &_ReadBytes);
if(OS_SUCCESS(status)) {
(*ReadBytes) += l;
RtlCopyMemory(Buffer, tmp_buff+i, l);
}
MyFreePool__(tmp_buff);
}
return status;
} // end UDFReadInSector()
/*
This routine reads data of unaligned offset & length
*/
OSSTATUS
UDFReadData(
IN PVCB Vcb,
IN BOOLEAN Translate, // Translate Logical to Physical
IN int64 Offset,
IN uint32 Length,
IN BOOLEAN Direct, // Disable access to non-cached data
OUT int8* Buffer,
OUT PSIZE_T ReadBytes
)
{
uint32 i, l, Lba, BS=Vcb->BlockSize;
uint32 BSh=Vcb->BlockSizeBits;
OSSTATUS status;
SIZE_T _ReadBytes = 0;
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
uint32 to_read;
(*ReadBytes) = 0;
if(!Length) return STATUS_SUCCESS;
if(Vcb->VCBFlags & UDF_VCB_FLAGS_DEAD)
return STATUS_NO_SUCH_DEVICE;
// read tail of the 1st sector if Offset is not sector_size-aligned
Lba = (uint32)(Offset >> BSh);
if((i = (uint32)(Offset & (BS-1)))) {
l = (BS - i) < Length ?
(BS - i) : Length;
// here we use 'ReadBytes' 'cause now it's set to zero
status = UDFReadInSector(Vcb, Translate, Lba, i, l, Direct, Buffer, ReadBytes);
if(!OS_SUCCESS(status)) return status;
if(!(Length = Length - l)) return STATUS_SUCCESS;
Lba ++;
Buffer += l;
}
// read sector_size-aligned part
i = Length >> BSh;
while(i) {
to_read = min(i, 64);
status = UDFReadSectors(Vcb, Translate, Lba, to_read, Direct, Buffer, &_ReadBytes);
(*ReadBytes) += _ReadBytes;
if(!OS_SUCCESS(status)) {
return status;
}
Buffer += to_read<<BSh;
Length -= to_read<<BSh;
Lba += to_read;
i -= to_read;
}
// read head of the last sector
if(!Length) return STATUS_SUCCESS;
status = UDFReadInSector(Vcb, Translate, Lba, 0, Length, Direct, Buffer, &_ReadBytes);
(*ReadBytes) += _ReadBytes;
return status;
} // end UDFReadData()
#endif //_BROWSE_UDF_
#ifndef UDF_READ_ONLY_BUILD
/*
This routine writes physical sectors. This routine supposes Lba & Length
alignment on WriteBlock (packet) size.
*/
OSSTATUS
UDFWriteSectors(
IN PVCB Vcb,
IN BOOLEAN Translate, // Translate Logical to Physical
IN uint32 Lba,
IN uint32 BCount,
IN BOOLEAN Direct, // Disable access to non-cached data
IN int8* Buffer,
OUT PSIZE_T WrittenBytes
)
{
OSSTATUS status;
#ifdef _BROWSE_UDF_
if(!Vcb->Modified || (Vcb->IntegrityType == INTEGRITY_TYPE_CLOSE)) {
UDFSetModified(Vcb);
if(Vcb->LVid && !Direct) {
status = UDFUpdateLogicalVolInt(Vcb,FALSE);
}
}
if(Vcb->CDR_Mode) {
if(Vcb->LastLBA < Lba+BCount-1)
Vcb->LastLBA = Lba+BCount-1;
}
#endif //_BROWSE_UDF_
if(Vcb->FastCache.WriteProc && (KeGetCurrentIrql() < DISPATCH_LEVEL)) {
status = WCacheWriteBlocks__(&(Vcb->FastCache), Vcb, Buffer, Lba, BCount, WrittenBytes, Direct);
ASSERT(OS_SUCCESS(status));
#ifdef _BROWSE_UDF_
UDFClrZeroBits(Vcb->ZSBM_Bitmap, Lba, BCount);
#endif //_BROWSE_UDF_
return status;
}
/* void* buffer;
OSSTATUS status;
SIZE_T _ReadBytes;
(*WrittenBytes) = 0;
buffer = DbgAllocatePool(NonPagedPool, Vcb->WriteBlockSize);
if(!buffer) return STATUS_INSUFFICIENT_RESOURCES;
status = UDFTRead(Vcb, Buffer, BCount<<Vcb->BlockSizeBits, (Lba&~(Vcb->WriteBlockSize-1), _WrittenBytes);*/
#ifdef UDF_DBG
status = UDFTWrite(Vcb, Buffer, BCount<<Vcb->BlockSizeBits, Lba, WrittenBytes);
ASSERT(OS_SUCCESS(status));
return status;
#else // UDF_DBG
return UDFTWrite(Vcb, Buffer, BCount<<Vcb->BlockSizeBits, Lba, WrittenBytes);
#endif // UDF_DBG
} // end UDFWriteSectors()
OSSTATUS
UDFWriteInSector(
IN PVCB Vcb,
IN BOOLEAN Translate, // Translate Logical to Physical
IN uint32 Lba,
IN uint32 i, // offset in sector
IN uint32 l, // transfer length
IN BOOLEAN Direct, // Disable access to non-cached data
OUT int8* Buffer,
OUT PSIZE_T WrittenBytes
)
{
int8* tmp_buff;
OSSTATUS status;
#ifdef _BROWSE_UDF_
SIZE_T _WrittenBytes;
SIZE_T ReadBytes;
if(!Vcb->Modified) {
UDFSetModified(Vcb);
if(Vcb->LVid)
status = UDFUpdateLogicalVolInt(Vcb,FALSE);
}
if(Vcb->CDR_Mode) {
if(Vcb->LastLBA < Lba)
Vcb->LastLBA = Lba;
}
#endif //_BROWSE_UDF_
(*WrittenBytes) = 0;
#ifdef _BROWSE_UDF_
if(WCacheIsInitialized__(&(Vcb->FastCache)) && (KeGetCurrentIrql() < DISPATCH_LEVEL)) {
#endif //_BROWSE_UDF_
status = WCacheDirect__(&(Vcb->FastCache), Vcb, Lba, TRUE, &tmp_buff, Direct);
if(OS_SUCCESS(status)) {
#ifdef _BROWSE_UDF_
UDFClrZeroBit(Vcb->ZSBM_Bitmap, Lba);
#endif //_BROWSE_UDF_
(*WrittenBytes) += l;
RtlCopyMemory(tmp_buff+i, Buffer, l);
}
if(!Direct) WCacheEODirect__(&(Vcb->FastCache), Vcb);
#ifdef _BROWSE_UDF_
} else {
// If Direct = TRUE we should never get here, but...
if(Direct) {
BrutePoint();
return STATUS_INVALID_PARAMETER;
}
tmp_buff = (int8*)MyAllocatePool__(NonPagedPool, Vcb->BlockSize);
if(!tmp_buff) {
BrutePoint();
return STATUS_INSUFFICIENT_RESOURCES;
}
// read packet
status = UDFReadSectors(Vcb, Translate, Lba, 1, FALSE, tmp_buff, &ReadBytes);
if(!OS_SUCCESS(status)) goto EO_WrSctD;
// modify packet
RtlCopyMemory(tmp_buff+i, Buffer, l);
// write modified packet
status = UDFWriteSectors(Vcb, Translate, Lba, 1, FALSE, tmp_buff, &_WrittenBytes);
if(OS_SUCCESS(status))
(*WrittenBytes) += l;
EO_WrSctD:
MyFreePool__(tmp_buff);
}
ASSERT(OS_SUCCESS(status));
if(!OS_SUCCESS(status)) {
UDFPrint(("UDFWriteInSector() for LBA %x failed\n", Lba));
}
#endif //_BROWSE_UDF_
return status;
} // end UDFWriteInSector()
/*
This routine writes data at unaligned offset & length
*/
OSSTATUS
UDFWriteData(
IN PVCB Vcb,
IN BOOLEAN Translate, // Translate Logical to Physical
IN int64 Offset,
IN SIZE_T Length,
IN BOOLEAN Direct, // setting this flag delays flushing of given
// data to indefinite term
IN int8* Buffer,
OUT PSIZE_T WrittenBytes
)
{
uint32 i, l, Lba, BS=Vcb->BlockSize;
uint32 BSh=Vcb->BlockSizeBits;
OSSTATUS status;
SIZE_T _WrittenBytes;
Vcb->VCBFlags |= UDF_VCB_SKIP_EJECT_CHECK;
(*WrittenBytes) = 0;
if(!Length) return STATUS_SUCCESS;
if(Vcb->VCBFlags & UDF_VCB_FLAGS_DEAD)
return STATUS_NO_SUCH_DEVICE;
// write tail of the 1st sector if Offset is not sector_size-aligned
Lba = (uint32)(Offset >> BSh);
if((i = ((uint32)Offset & (BS-1)))) {
l = (BS - i) < Length ?
(BS - i) : Length;
status = UDFWriteInSector(Vcb, Translate, Lba, i, l, Direct, Buffer, WrittenBytes);
if(!OS_SUCCESS(status)) return status;
if(!(Length = Length - l)) return STATUS_SUCCESS;
Lba ++;
Buffer += l;
}
// write sector_size-aligned part
i = Length >> BSh;
if(i) {
status = UDFWriteSectors(Vcb, Translate, Lba, i, Direct, Buffer, &_WrittenBytes);
(*WrittenBytes) += _WrittenBytes;
if(!OS_SUCCESS(status)) return status;
l = i<<BSh;
#ifdef _BROWSE_UDF_
UDFClrZeroBits(Vcb->ZSBM_Bitmap, Lba, i);
#endif //_BROWSE_UDF_
if(!(Length = Length - l)) return STATUS_SUCCESS;
Lba += i;
Buffer += l;
}
status = UDFWriteInSector(Vcb, Translate, Lba, 0, Length, Direct, Buffer, &_WrittenBytes);
(*WrittenBytes) += _WrittenBytes;
#ifdef _BROWSE_UDF_
UDFClrZeroBit(Vcb->ZSBM_Bitmap, Lba);
#endif //_BROWSE_UDF_
return status;
} // end UDFWriteData()
#endif //UDF_READ_ONLY_BUILD
OSSTATUS
UDFResetDeviceDriver(
IN PVCB Vcb,
IN PDEVICE_OBJECT TargetDeviceObject,
IN BOOLEAN Unlock
)
{
PCDRW_RESET_DRIVER_USER_IN tmp = (PCDRW_RESET_DRIVER_USER_IN)
MyAllocatePool__(NonPagedPool, sizeof(CDRW_RESET_DRIVER_USER_IN));
OSSTATUS RC;
if(!tmp)
return STATUS_INSUFFICIENT_RESOURCES;
RtlZeroMemory(tmp, sizeof(CDRW_RESET_DRIVER_USER_IN));
tmp->UnlockTray = (Unlock ? 1 : 0);
tmp->MagicWord = 0x3a6 | (Unlock ? 1 : 0);
RC = UDFPhSendIOCTL(IOCTL_CDRW_RESET_DRIVER_EX, TargetDeviceObject,
tmp, sizeof(CDRW_RESET_DRIVER_USER_IN), NULL, 0, TRUE,NULL);
if(Vcb) {
Vcb->LastReadTrack = 0;
Vcb->LastModifiedTrack = 0;
Vcb->OPCDone = FALSE;
if((Vcb->VCBFlags & UDF_VCB_FLAGS_USE_CAV) &&
Vcb->TargetDeviceObject) {
// limit both read & write speed to last write speed for CAV mode
// some drives damage data when speed is adjusted during recording process
// even in packet mode
UDFSetSpeeds(Vcb);
}
UDFSetCaching(Vcb);
}
MyFreePool__(tmp);
return RC;
} // end UDFResetDeviceDriver()
OSSTATUS
UDFSetSpeeds(
IN PVCB Vcb
)
{
OSSTATUS RC;
RtlZeroMemory(&(Vcb->SpeedBuf), sizeof(SET_CD_SPEED_EX_USER_IN));
if(Vcb->VCBFlags & UDF_VCB_FLAGS_USE_CAV) {
Vcb->SpeedBuf.RotCtrl = CdSpeed_RotCtrl_CAV;
Vcb->CurSpeed =
Vcb->SpeedBuf.ReadSpeed =
Vcb->SpeedBuf.WriteSpeed = Vcb->MaxWriteSpeed;
} else {
Vcb->SpeedBuf.ReadSpeed = Vcb->CurSpeed;
Vcb->SpeedBuf.WriteSpeed = Vcb->MaxWriteSpeed;
}
UDFPrint((" UDFSetSpeeds: set speed to %s %dX/%dX\n",
(Vcb->VCBFlags & UDF_VCB_FLAGS_USE_CAV) ? "CAV" : "CLV",
Vcb->SpeedBuf.ReadSpeed / 176,
Vcb->SpeedBuf.WriteSpeed / 176));
RC = UDFPhSendIOCTL(IOCTL_CDRW_SET_SPEED,
Vcb->TargetDeviceObject,
&(Vcb->SpeedBuf),sizeof(SET_CD_SPEED_EX_USER_IN),
NULL,0,TRUE,NULL);
UDFPrint(("UDFSetSpeeds: %x\n", RC));
return RC;
} // end UDFSetSpeeds()
NTSTATUS
UDFSetCaching(
IN PVCB Vcb
)
{
#pragma pack(push,1)
struct {
MODE_PARAMETER_HEADER Header;
MODE_CACHING_PAGE Data;
CHAR Padding [16];
} CachingPage;
struct {
MODE_PARAMETER_HEADER Header;
MODE_READ_WRITE_RECOVERY_PAGE Data;
CHAR Padding [16];
} RecoveryPage;
#ifdef _MSC_VER
#pragma pack(pop,1)
#else
#pragma pack(pop)
#endif
MODE_SENSE_USER_IN ModeSenseCtl;
OSSTATUS RC;
UDFPrint(("UDFSetCaching:\n"));
ModeSenseCtl.PageCode.Byte = MODE_PAGE_ERROR_RECOVERY;
RC = UDFPhSendIOCTL(IOCTL_CDRW_MODE_SENSE, Vcb->TargetDeviceObject,
&ModeSenseCtl,sizeof(ModeSenseCtl),
(PVOID)&RecoveryPage,sizeof(RecoveryPage),
FALSE, NULL);
if(OS_SUCCESS(RC)) {
UDFPrint((" Error recovery page:\n"
"PageCode %d\n"
"PageLength %d\n"
"DCRBit %d\n"
"DTEBit %d\n"
"PERBit %d\n"
"EERBit %d\n"
"RCBit %d\n"
"TBBit %d\n"
"ARRE %d\n"
"AWRE %d\n"
"ReadRetryCount %d\n"
"CorrectionSpan %d\n"
"HeadOffsetCount %d\n"
"DataStrobOffsetCount %d\n"
"ErrorRecoveryParam2.Fields.EMCDR %d\n"
"WriteRetryCount %d\n",
RecoveryPage.Data.PageCode,
RecoveryPage.Data.PageLength,
RecoveryPage.Data.ErrorRecoveryParam.Fields.DCRBit,
RecoveryPage.Data.ErrorRecoveryParam.Fields.DTEBit,
RecoveryPage.Data.ErrorRecoveryParam.Fields.PERBit,
RecoveryPage.Data.ErrorRecoveryParam.Fields.EERBit,
RecoveryPage.Data.ErrorRecoveryParam.Fields.RCBit,
RecoveryPage.Data.ErrorRecoveryParam.Fields.TBBit,
RecoveryPage.Data.ErrorRecoveryParam.Fields.ARRE,
RecoveryPage.Data.ErrorRecoveryParam.Fields.AWRE,
RecoveryPage.Data.ReadRetryCount,
RecoveryPage.Data.CorrectionSpan,
RecoveryPage.Data.HeadOffsetCount,
RecoveryPage.Data.DataStrobOffsetCount,
RecoveryPage.Data.ErrorRecoveryParam2.Fields.EMCDR,
RecoveryPage.Data.WriteRetryCount
));
}
ModeSenseCtl.PageCode.Byte = MODE_PAGE_CACHING;
RC = UDFPhSendIOCTL(IOCTL_CDRW_MODE_SENSE, Vcb->TargetDeviceObject,
&ModeSenseCtl,sizeof(ModeSenseCtl),
(PVOID)&CachingPage,sizeof(CachingPage),
FALSE, NULL);
if(!OS_SUCCESS(RC)) {
return RC;
}
UDFPrint((" Caching page:\n"
"PageCode %d\n"
"PageLength %d\n"
"ReadDisableCache %d\n"
"MultiplicationFactor %d\n"
"WriteCacheEnable %d\n"
"WriteRetensionPriority %d\n"
"ReadRetensionPriority %d\n",
CachingPage.Data.PageCode,
CachingPage.Data.PageLength,
CachingPage.Data.ReadDisableCache,
CachingPage.Data.MultiplicationFactor,
CachingPage.Data.WriteCacheEnable,
CachingPage.Data.WriteRetensionPriority,
CachingPage.Data.ReadRetensionPriority
));
RtlZeroMemory(&CachingPage.Header, sizeof(CachingPage.Header));
CachingPage.Data.PageCode = MODE_PAGE_CACHING;
CachingPage.Data.PageSavable = 0;
if( CachingPage.Data.ReadDisableCache ||
!CachingPage.Data.WriteCacheEnable) {
CachingPage.Data.ReadDisableCache = 0;
CachingPage.Data.WriteCacheEnable = 1;
RC = UDFPhSendIOCTL(IOCTL_CDRW_MODE_SELECT, Vcb->TargetDeviceObject,
(PVOID)&CachingPage,sizeof(CachingPage.Header) + 2 + CachingPage.Data.PageLength,
NULL,0,
FALSE, NULL);
} else {
RC = STATUS_SUCCESS;
}
UDFPrint(("UDFSetCaching: %x\n", RC));
return RC;
} // end UDFSetCaching()