dirctl.c

来自「一个类似windows」· C语言 代码 · 共 792 行 · 第 1/2 页

C
792
字号
/*
 *  ReactOS kernel
 *  Copyright (C) 2002, 2004 ReactOS Team
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2 of the License, or
 *  (at your option) any later version.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program; if not, write to the Free Software
 *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 */
/* $Id: dirctl.c 21710 2006-04-22 16:36:21Z tretiakov $
 *
 * COPYRIGHT:        See COPYING in the top level directory
 * PROJECT:          ReactOS kernel
 * FILE:             services/fs/cdfs/dirctl.c
 * PURPOSE:          CDROM (ISO 9660) filesystem driver
 * PROGRAMMER:       Art Yerkes
 *                   Eric Kohl
 * UPDATE HISTORY:
 */

/* INCLUDES *****************************************************************/

#include "cdfs.h"

#define NDEBUG
#include <debug.h>

/* FUNCTIONS ****************************************************************/

#define ROUND_DOWN(N, S) (((N) / (S)) * (S))
#define ROUND_UP(N, S) ((((N) + (S) - 1) / (S)) * (S))

/* FUNCTIONS ****************************************************************/

/*
 * FUNCTION: Retrieves the file name, be it in short or long file name format
 */
static NTSTATUS
CdfsGetEntryName(PDEVICE_EXTENSION DeviceExt,
		 PVOID *Context,
		 PVOID *Block,
		 PLARGE_INTEGER StreamOffset,
		 ULONG DirLength,
		 PVOID *Ptr,
		 PWSTR Name,
		 PULONG pIndex,
		 PULONG CurrentOffset)
{
  PDIR_RECORD Record = *Ptr;
  ULONG Index;

  if (*CurrentOffset >= DirLength)
     return(STATUS_NO_MORE_ENTRIES);

  if (*CurrentOffset == 0)
  {
     Index = 0;
     Record = (PDIR_RECORD)*Block;
     while (Index < *pIndex)
     {
       (*Ptr) = (PVOID)((ULONG_PTR)(*Ptr) + Record->RecordLength);
       (*CurrentOffset) += Record->RecordLength;
       Record = *Ptr;
       if ((ULONG_PTR)(*Ptr) - (ULONG_PTR)(*Block) >= BLOCKSIZE || Record->RecordLength == 0)
       {
	  DPRINT("Map next sector\n");
	  CcUnpinData(*Context);
	  StreamOffset->QuadPart += BLOCKSIZE;
	  *CurrentOffset = ROUND_UP(*CurrentOffset, BLOCKSIZE);
	  if (!CcMapData(DeviceExt->StreamFileObject,
		         StreamOffset,
		         BLOCKSIZE, TRUE,
		         Context, Block))
	  {
	     DPRINT("CcMapData() failed\n");
	     return(STATUS_UNSUCCESSFUL);
	  }
	  *Ptr = *Block;
	  Record = (PDIR_RECORD)*Ptr;
       }
       if (*CurrentOffset >= DirLength)
	 return(STATUS_NO_MORE_ENTRIES);

       Index++;
     }
  }

  if ((ULONG_PTR)(*Ptr) - (ULONG_PTR)(*Block) >= BLOCKSIZE || Record->RecordLength == 0)
  {
     DPRINT("Map next sector\n");
     CcUnpinData(*Context);
     StreamOffset->QuadPart += BLOCKSIZE;
     *CurrentOffset = ROUND_UP(*CurrentOffset, BLOCKSIZE);
     if (!CcMapData(DeviceExt->StreamFileObject,
		       StreamOffset,
		       BLOCKSIZE, TRUE,
		       Context, Block))
     {
       DPRINT("CcMapData() failed\n");
       return(STATUS_UNSUCCESSFUL);
     }
     *Ptr = *Block;
     Record = (PDIR_RECORD)*Ptr;
  }

  if (*CurrentOffset >= DirLength)
    return STATUS_NO_MORE_ENTRIES;

  DPRINT("Index %lu  RecordLength %lu  Offset %lu\n",
	 *pIndex, Record->RecordLength, *CurrentOffset);

  if (Record->FileIdLength == 1 && Record->FileId[0] == 0)
    {
      wcscpy(Name, L".");
    }
  else if (Record->FileIdLength == 1 && Record->FileId[0] == 1)
    {
      wcscpy(Name, L"..");
    }
  else
    {
      if (DeviceExt->CdInfo.JolietLevel == 0)
	{
	  ULONG i;

	  for (i = 0; i < Record->FileIdLength && Record->FileId[i] != ';'; i++)
	    Name[i] = (WCHAR)Record->FileId[i];
	  Name[i] = 0;
	}
      else
	{
	  CdfsSwapString(Name, Record->FileId, Record->FileIdLength);
	}
    }

  DPRINT("Name '%S'\n", Name);

  *Ptr = Record;

  return(STATUS_SUCCESS);
}


/*
 * FUNCTION: Find a file
 */
static NTSTATUS
CdfsFindFile(PDEVICE_EXTENSION DeviceExt,
	     PFCB Fcb,
	     PFCB Parent,
	     PUNICODE_STRING FileToFind,
	     PULONG pDirIndex,
	     PULONG pOffset)
{
  WCHAR name[256];
  WCHAR ShortNameBuffer[13];
  UNICODE_STRING TempString;
  UNICODE_STRING ShortName;
  UNICODE_STRING LongName;
  PVOID Block;
  NTSTATUS Status;
  ULONG len;
  ULONG DirIndex;
  ULONG Offset = 0;
  BOOLEAN IsRoot;
  PVOID Context = NULL;
  ULONG DirSize;
  PDIR_RECORD Record;
  LARGE_INTEGER StreamOffset;
  BOOLEAN HasSpaces;
  GENERATE_NAME_CONTEXT NameContext;

  DPRINT("FindFile(Parent %x, FileToFind '%wZ', DirIndex: %d)\n",
	 Parent, FileToFind, pDirIndex ? *pDirIndex : 0);
  DPRINT("FindFile: old Pathname %x, old Objectname %x)\n",
	 Fcb->PathName, Fcb->ObjectName);

  IsRoot = FALSE;
  DirIndex = 0;

  if (FileToFind == NULL || FileToFind->Length == 0)
    {
      CHECKPOINT;
      RtlInitUnicodeString(&TempString, L".");
      FileToFind = &TempString;
    }

  if (Parent)
    {
      if (Parent->Entry.ExtentLocationL == DeviceExt->CdInfo.RootStart)
	{
	  IsRoot = TRUE;
	}
    }
  else
    {
      IsRoot = TRUE;
    }

  if (IsRoot == TRUE)
    {
      StreamOffset.QuadPart = (LONGLONG)DeviceExt->CdInfo.RootStart * (LONGLONG)BLOCKSIZE;
      DirSize = DeviceExt->CdInfo.RootSize;


      if (FileToFind->Buffer[0] == 0 ||
	  (FileToFind->Buffer[0] == '\\' && FileToFind->Buffer[1] == 0) ||
	  (FileToFind->Buffer[0] == '.' && FileToFind->Buffer[1] == 0))
	{
	  /* it's root : complete essentials fields then return ok */
	  RtlZeroMemory(Fcb, sizeof(FCB));

	  Fcb->PathName[0] = '\\';
	  Fcb->ObjectName = &Fcb->PathName[1];
	  Fcb->Entry.ExtentLocationL = DeviceExt->CdInfo.RootStart;
	  Fcb->Entry.DataLengthL = DeviceExt->CdInfo.RootSize;
	  Fcb->Entry.FileFlags = 0x02; //FILE_ATTRIBUTE_DIRECTORY;

	  if (pDirIndex)
	    *pDirIndex = 0;
	  if (pOffset)
	    *pOffset = 0;
	  DPRINT("CdfsFindFile: new Pathname %S, new Objectname %S)\n",Fcb->PathName, Fcb->ObjectName);
	  return STATUS_SUCCESS;
	}
    }
  else
    {
      StreamOffset.QuadPart = (LONGLONG)Parent->Entry.ExtentLocationL * (LONGLONG)BLOCKSIZE;
      DirSize = Parent->Entry.DataLengthL;
    }

  DPRINT("StreamOffset %I64u  DirSize %lu\n", StreamOffset.QuadPart, DirSize);

  if (pDirIndex && (*pDirIndex))
    DirIndex = *pDirIndex;

  if (pOffset && (*pOffset))
    {
      Offset = *pOffset;
      StreamOffset.QuadPart += ROUND_DOWN(Offset, BLOCKSIZE);
    }

  if (!CcMapData(DeviceExt->StreamFileObject, &StreamOffset,
		BLOCKSIZE, TRUE, &Context, &Block))
    {
      DPRINT("CcMapData() failed\n");
      return STATUS_UNSUCCESSFUL;
    }

  Record = (PDIR_RECORD) ((ULONG_PTR)Block + Offset % BLOCKSIZE);
  if (Offset)
    {
      Offset += Record->RecordLength;
      Record = (PDIR_RECORD)((ULONG_PTR)Record + Record->RecordLength);
    }

  while(TRUE)
    {
      DPRINT("RecordLength %u  ExtAttrRecordLength %u  NameLength %u\n",
	     Record->RecordLength, Record->ExtAttrRecordLength, Record->FileIdLength);

      Status = CdfsGetEntryName(DeviceExt, &Context, &Block, &StreamOffset,
	                        DirSize, (PVOID*)&Record, name, &DirIndex, &Offset);

      if (Status == STATUS_NO_MORE_ENTRIES)
	{
	  break;
	}
      else if (Status == STATUS_UNSUCCESSFUL)
	{
	  /* Note: the directory cache has already been unpinned */
	  return Status;
	}

      DPRINT("Name '%S'\n", name);

      RtlInitUnicodeString(&LongName, name);
      ShortName.Length = 0;
      ShortName.MaximumLength = 26;
      ShortName.Buffer = ShortNameBuffer;

      if ((RtlIsNameLegalDOS8Dot3(&LongName, NULL, &HasSpaces) == FALSE) ||
	  (HasSpaces == TRUE))
	{
	  /* Build short name */
	  RtlGenerate8dot3Name(&LongName,
			       FALSE,
			       &NameContext,
			       &ShortName);
	}
      else
	{
	  /* copy short name */
	  RtlUpcaseUnicodeString(&ShortName,
				 &LongName,
				 FALSE);
	}

      DPRINT("ShortName '%wZ'\n", &ShortName);

      if (FsRtlIsNameInExpression(FileToFind, &LongName, TRUE, NULL) ||
	  FsRtlIsNameInExpression(FileToFind, &ShortName, TRUE, NULL))
	{
	  if (Parent && Parent->PathName)
	    {
	      len = wcslen(Parent->PathName);
	      memcpy(Fcb->PathName, Parent->PathName, len*sizeof(WCHAR));
	      Fcb->ObjectName=&Fcb->PathName[len];
	      if (len != 1 || Fcb->PathName[0] != '\\')
		{
		  Fcb->ObjectName[0] = '\\';
		  Fcb->ObjectName = &Fcb->ObjectName[1];
		}
	    }
	  else
	    {
	      Fcb->ObjectName=Fcb->PathName;
	      Fcb->ObjectName[0]='\\';
	      Fcb->ObjectName=&Fcb->ObjectName[1];
	    }

	  DPRINT("PathName '%S'  ObjectName '%S'\n", Fcb->PathName, Fcb->ObjectName);

	  memcpy(&Fcb->Entry, Record, sizeof(DIR_RECORD));
	  wcsncpy(Fcb->ObjectName, name, MAX_PATH);

	  /* Copy short name */
	  Fcb->ShortNameU.Length = ShortName.Length;
	  Fcb->ShortNameU.MaximumLength = ShortName.Length;
	  Fcb->ShortNameU.Buffer = Fcb->ShortNameBuffer;
	  memcpy(Fcb->ShortNameBuffer, ShortName.Buffer, ShortName.Length);

	  if (pDirIndex)
	    *pDirIndex = DirIndex;
	  if (pOffset)
	    *pOffset = Offset;

	  DPRINT("FindFile: new Pathname %S, new Objectname %S, DirIndex %d\n",
		 Fcb->PathName, Fcb->ObjectName, DirIndex);

	  CcUnpinData(Context);

	  return STATUS_SUCCESS;
	}

      Offset += Record->RecordLength;
      Record = (PDIR_RECORD)((ULONG_PTR)Record + Record->RecordLength);
      DirIndex++;
    }

  CcUnpinData(Context);

  if (pDirIndex)
    *pDirIndex = DirIndex;

  if (pOffset)
    *pOffset = Offset;

  return STATUS_UNSUCCESSFUL;
}


static NTSTATUS
CdfsGetNameInformation(PFCB Fcb,
		       PDEVICE_EXTENSION DeviceExt,
		       PFILE_NAMES_INFORMATION Info,
		       ULONG BufferLength)
{
  ULONG Length;

  DPRINT("CdfsGetNameInformation() called\n");

  Length = wcslen(Fcb->ObjectName) * sizeof(WCHAR);
  if ((sizeof (FILE_BOTH_DIR_INFORMATION) + Length) > BufferLength)
    return(STATUS_BUFFER_OVERFLOW);

  Info->FileNameLength = Length;
  Info->NextEntryOffset =
    ROUND_UP(sizeof(FILE_BOTH_DIR_INFORMATION) + Length, 4);
  memcpy(Info->FileName, Fcb->ObjectName, Length);

  return(STATUS_SUCCESS);
}


static NTSTATUS

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?