⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 objdir.c

📁 ReactOS是一些高手根据Windows XP的内核编写出的类XP。内核实现机理和API函数调用几乎相同。甚至可以兼容XP的程序。喜欢研究系统内核的人可以看一看。
💻 C
字号:
/* $Id: objdir.c 24501 2006-10-13 10:56:52Z janderwald $
 *
 * DESCRIPTION: Object Manager Simple Explorer
 * PROGRAMMER:  David Welch
 * REVISIONS
 * 	2000-04-30 (ea)
 * 		Added directory enumeration.
 * 		(tested under nt4sp4/x86)
 * 	2000-08-11 (ea)
 * 		Added symbolic link expansion.
 * 		(tested under nt4sp4/x86)
 * 	2001-05-01 (ea)
 * 		Fixed entries counter. Added more
 * 		error codes check. Removed wprintf,
 * 		because it does not work in .17.
 * 	2001-05-02 (ea)
 * 		Added -r option.
 */

#define WIN32_NO_STATUS
#include <windows.h>
#include <stdlib.h>
#include <ntndk.h>
#include <stdio.h>

#define MAX_DIR_ENTRY 256


static
PCHAR
STDCALL
RawUszAsz (
	PWCHAR	szU,
	PCHAR	szA
	)
{
	register PCHAR a = szA;

	while (*szU) {*szA++ = (CHAR) (0x00ff & * szU++);}
	*szA = '\0';
	return a;
}


static
PWCHAR
STDCALL
RawAszUsz (
	PCHAR	szA,
	PWCHAR	szW
	)
{
	register PWCHAR w = szW;

	while (*szA) {*szW++ = (WCHAR) *szA++;}
	*szW = L'\0';
	return w;
}


static
const char *
STDCALL
StatusToName (NTSTATUS Status)
{
	static char RawValue [16];

	switch (Status)
	{
		case STATUS_BUFFER_TOO_SMALL:
			return "STATUS_BUFFER_TOO_SMALL";
		case STATUS_INVALID_PARAMETER:
			return "STATUS_INVALID_PARAMETER";
		case STATUS_OBJECT_NAME_INVALID:
			return "STATUS_OBJECT_NAME_INVALID";
		case STATUS_OBJECT_NAME_NOT_FOUND:
			return "STATUS_OBJECT_NAME_NOT_FOUND";
		case STATUS_OBJECT_PATH_SYNTAX_BAD:
			return "STATUS_PATH_SYNTAX_BAD";
		case STATUS_NO_MORE_ENTRIES:
			return "STATUS_NO_MORE_ENTRIES";
		case STATUS_MORE_ENTRIES:
			return "STATUS_MORE_ENTRIES";
		case STATUS_ACCESS_DENIED:
			return "STATUS_ACCESS_DENIED";
		case STATUS_UNSUCCESSFUL:
			return "STATUS_UNSUCCESSFUL";
		case STATUS_INVALID_HANDLE:
			return "STATUS_INVALID_HANDLE";
	}
	sprintf (RawValue, "0x%08lx", Status);
	return (const char *) RawValue;
}


BOOL
STDCALL
ExpandSymbolicLink (
	IN	PUNICODE_STRING	DirectoryName,
	IN	PUNICODE_STRING	SymbolicLinkName,
	IN OUT	PUNICODE_STRING	TargetObjectName
	)
{
	NTSTATUS		Status;
	HANDLE			hSymbolicLink;
	OBJECT_ATTRIBUTES	oa;
	UNICODE_STRING		Path;
	WCHAR			PathBuffer [MAX_PATH];
	ULONG			DataWritten = 0;


	Path.Buffer = PathBuffer;
	Path.Length = 0;
	Path.MaximumLength = sizeof PathBuffer;

	RtlCopyUnicodeString (& Path, DirectoryName);
	if (L'\\' != Path.Buffer [(Path.Length / sizeof Path.Buffer[0]) - 1])
	{
		RtlAppendUnicodeToString (& Path, L"\\");
	}
	RtlAppendUnicodeStringToString (& Path, SymbolicLinkName);

	oa.Length			= sizeof (OBJECT_ATTRIBUTES);
	oa.ObjectName			= & Path;
	oa.Attributes			= 0; /* OBJ_CASE_INSENSITIVE; */
	oa.RootDirectory		= NULL;
	oa.SecurityDescriptor		= NULL;
	oa.SecurityQualityOfService	= NULL;

	Status = NtOpenSymbolicLinkObject(
			& hSymbolicLink,
			SYMBOLIC_LINK_QUERY,	/* 0x20001 */
			& oa
			);

	if (!NT_SUCCESS(Status))
	{
		printf (
			"Failed to open SymbolicLink object (Status: %s)\n",
			StatusToName (Status)
			);
		return FALSE;
	}
	TargetObjectName->Length = TargetObjectName->MaximumLength;
	memset (
		TargetObjectName->Buffer,
		0,
		TargetObjectName->MaximumLength
		);
	Status = NtQuerySymbolicLinkObject(
			hSymbolicLink,
			TargetObjectName,
			& DataWritten
			);
	if (!NT_SUCCESS(Status))
	{
		printf (
			"Failed to query SymbolicLink object (Status: %s)\n",
			StatusToName (Status)
			);
		NtClose (hSymbolicLink);
		return FALSE;
	}
	NtClose (hSymbolicLink);
	return TRUE;
}


BOOL
STDCALL
ListDirectory (
	IN	PUNICODE_STRING	DirectoryNameW,
	IN	BOOL		Recurse
	)
{
	CHAR			DirectoryNameA [MAX_PATH];
	OBJECT_ATTRIBUTES	ObjectAttributes;
	NTSTATUS		Status;
	HANDLE			DirectoryHandle;
	BYTE			DirectoryEntry [512];
	POBJECT_DIRECTORY_INFORMATION pDirectoryEntry = (POBJECT_DIRECTORY_INFORMATION) DirectoryEntry;
	POBJECT_DIRECTORY_INFORMATION pDirectoryEntries = (POBJECT_DIRECTORY_INFORMATION) DirectoryEntry;
	ULONG			Context = 0;
	ULONG			ReturnLength = 0;
	ULONG			EntryCount = 0;

	/* For expanding symbolic links */
	WCHAR			TargetName [2 * MAX_PATH];
	UNICODE_STRING		TargetObjectName = {
					sizeof TargetName,
					sizeof TargetName,
					TargetName
				};

	/* Convert to ANSI the directory's name */
	RawUszAsz (DirectoryNameW->Buffer, DirectoryNameA);
	/*
	 * Prepare parameters for next call.
	 */
	InitializeObjectAttributes (
		& ObjectAttributes,
		DirectoryNameW,
		0,
		NULL,
		NULL
		);
	/*
	 * Try opening the directory.
	 */
	Status = NtOpenDirectoryObject (
			& DirectoryHandle,
			DIRECTORY_QUERY,
			& ObjectAttributes
			);
	if (!NT_SUCCESS(Status))
	{
		printf (
			"Failed to open directory object \"%s\" (Status: %s)\n",
			DirectoryNameA,
			StatusToName (Status)
			);
		return (FALSE);
	}
	printf ("\n Directory of %s\n\n", DirectoryNameA);

        for(;;)
        {
	/*
	 * Enumerate each item in the directory.
	 */
	Status = NtQueryDirectoryObject (
			DirectoryHandle,
			pDirectoryEntries,
			sizeof DirectoryEntry,
			FALSE,/* ReturnSingleEntry */
			FALSE, /* RestartScan */
			& Context,
			& ReturnLength
			);
	if (!NT_SUCCESS(Status) && Status != STATUS_NO_MORE_ENTRIES)
	{
		printf (
			"Failed to query directory object (Status: %s)\n",
			StatusToName (Status)
			);
		NtClose (DirectoryHandle);
		return (FALSE);
	}
	if (Status == STATUS_NO_MORE_ENTRIES)
	{
          break;
        }
	pDirectoryEntry = pDirectoryEntries;
	while (EntryCount < Context)
	{
		CHAR ObjectNameA [MAX_PATH];
		CHAR TypeNameA [MAX_PATH];
		CHAR TargetNameA [MAX_PATH];

		if (0 == wcscmp (L"SymbolicLink", pDirectoryEntry->TypeName.Buffer))
		{
			if (TRUE == ExpandSymbolicLink (
					DirectoryNameW,
					& pDirectoryEntry->Name,
					& TargetObjectName
					)
				)
			{

				printf (
					"%-16s %s -> %s\n",
					RawUszAsz (pDirectoryEntry->TypeName.Buffer, TypeNameA),
					RawUszAsz (pDirectoryEntry->Name.Buffer, ObjectNameA),
					RawUszAsz (TargetObjectName.Buffer, TargetNameA)
					);
			}
			else
			{
				printf (
					"%-16s %s -> (error!)\n",
					RawUszAsz (pDirectoryEntry->TypeName.Buffer, TypeNameA),
					RawUszAsz (pDirectoryEntry->Name.Buffer, ObjectNameA)
					);
			}
		}
		else
		{
			printf (
				"%-16s %s\n",
				RawUszAsz (pDirectoryEntry->TypeName.Buffer, TypeNameA),
				RawUszAsz (pDirectoryEntry->Name.Buffer, ObjectNameA)
				);
		}
		++ pDirectoryEntry;
		++ EntryCount;
	}
	};
	printf ("\n\t%lu object(s)\n", EntryCount);
	/*
	 * Free any resource.
	 */
	NtClose (DirectoryHandle);
	/*
	 * Recurse into, if required so.
	 */
	if (FALSE != Recurse)
	{
		pDirectoryEntry = (POBJECT_DIRECTORY_INFORMATION) DirectoryEntry;
		while (0 != pDirectoryEntry->TypeName.Length)
		{
			if (0 == wcscmp (L"Directory", pDirectoryEntry->TypeName.Buffer))
			{
				WCHAR		CurrentName [MAX_PATH];
				UNICODE_STRING	CurrentDirectory;

				CurrentName [0] = L'\0';
				wcscpy (CurrentName, DirectoryNameW->Buffer);
				if (wcslen (CurrentName) > 1)
				{
					wcscat (CurrentName, L"\\");
				}
				wcscat (CurrentName, pDirectoryEntry->Name.Buffer);
				RtlInitUnicodeString (& CurrentDirectory, CurrentName);
				ListDirectory (& CurrentDirectory, Recurse);
			}
			++ pDirectoryEntry;
		}
	}
	return (TRUE);
}


int main(int argc, char* argv[])
{
	WCHAR		DirectoryNameW [MAX_PATH];
	UNICODE_STRING	DirectoryName;
	BOOL		Recurse = FALSE;

	/*
	 * Check user arguments.
	 */
	switch (argc)
	{
	case 2:
		RawAszUsz (argv[1], DirectoryNameW);
		break;
	case 3:
		if (strcmp (argv[1], "-r"))
		{
			fprintf (
				stderr,
				"%s: unknown option '%s'.\n",
				argv [0], argv[1]
				);
			return EXIT_FAILURE;
		}
		RawAszUsz (argv[2], DirectoryNameW);
		Recurse = TRUE;
		break;
	default:
		fprintf (
			stderr,
			"\nUsage: %s [-r] directory\n\n"
			"  -r          recurse\n"
			"  directory   a directory name in the system namespace\n\n",
			argv [0]
			);
		return EXIT_FAILURE;
	}
	/*
	 * List the directory.
	 */
	RtlInitUnicodeString (& DirectoryName, DirectoryNameW);
	return (FALSE == ListDirectory (& DirectoryName, Recurse))
		? EXIT_FAILURE
		: EXIT_SUCCESS;
}


/* EOF */

⌨️ 快捷键说明

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