mirror of
https://github.com/reactos/reactos.git
synced 2025-01-13 01:22:03 +00:00
9ea495ba33
svn path=/branches/header-work/; revision=45691
267 lines
5.1 KiB
C
267 lines
5.1 KiB
C
/*
|
|
* PROJECT: .inf file parser
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
* PROGRAMMER: Royce Mitchell III
|
|
* Eric Kohl
|
|
* Ge van Geldorp <gvg@reactos.org>
|
|
*/
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
#include "inflib.h"
|
|
#include "infros.h"
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
/* PRIVATE FUNCTIONS ********************************************************/
|
|
|
|
static int InfpHeapRefCount;
|
|
|
|
static VOID
|
|
CheckHeap()
|
|
{
|
|
if (NULL == InfpHeap)
|
|
{
|
|
InfpHeap = RtlCreateHeap(HEAP_GROWABLE, NULL, 0, 0, NULL, NULL);
|
|
}
|
|
if (0 <= InfpHeapRefCount)
|
|
{
|
|
InfpHeapRefCount++;
|
|
}
|
|
}
|
|
|
|
|
|
/* PUBLIC FUNCTIONS *********************************************************/
|
|
|
|
PVOID InfpHeap;
|
|
|
|
VOID
|
|
InfSetHeap(PVOID Heap)
|
|
{
|
|
if (NULL == InfpHeap)
|
|
{
|
|
InfpHeap = Heap;
|
|
InfpHeapRefCount = -1;
|
|
}
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
InfOpenBufferedFile(PHINF InfHandle,
|
|
PVOID Buffer,
|
|
ULONG BufferSize,
|
|
PULONG ErrorLine)
|
|
{
|
|
INFSTATUS Status;
|
|
PINFCACHE Cache;
|
|
PCHAR FileBuffer;
|
|
|
|
CheckHeap();
|
|
|
|
*InfHandle = NULL;
|
|
*ErrorLine = (ULONG)-1;
|
|
|
|
/* Allocate file buffer */
|
|
FileBuffer = MALLOC(BufferSize + 1);
|
|
if (FileBuffer == NULL)
|
|
{
|
|
DPRINT1("MALLOC() failed\n");
|
|
return(INF_STATUS_INSUFFICIENT_RESOURCES);
|
|
}
|
|
|
|
MEMCPY(FileBuffer, Buffer, BufferSize);
|
|
|
|
/* Append string terminator */
|
|
FileBuffer[BufferSize] = 0;
|
|
|
|
/* Allocate infcache header */
|
|
Cache = (PINFCACHE)MALLOC(sizeof(INFCACHE));
|
|
if (Cache == NULL)
|
|
{
|
|
DPRINT("MALLOC() failed\n");
|
|
FREE(FileBuffer);
|
|
return(INF_STATUS_INSUFFICIENT_RESOURCES);
|
|
}
|
|
|
|
/* Initialize inicache header */
|
|
ZEROMEMORY(Cache,
|
|
sizeof(INFCACHE));
|
|
|
|
/* Parse the inf buffer */
|
|
Status = InfpParseBuffer (Cache,
|
|
FileBuffer,
|
|
FileBuffer + BufferSize,
|
|
ErrorLine);
|
|
if (!INF_SUCCESS(Status))
|
|
{
|
|
FREE(Cache);
|
|
Cache = NULL;
|
|
}
|
|
|
|
/* Free file buffer */
|
|
FREE(FileBuffer);
|
|
|
|
*InfHandle = (HINF)Cache;
|
|
|
|
return(Status);
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
InfOpenFile(PHINF InfHandle,
|
|
PUNICODE_STRING FileName,
|
|
PULONG ErrorLine)
|
|
{
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
FILE_STANDARD_INFORMATION FileInfo;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
HANDLE FileHandle;
|
|
NTSTATUS Status;
|
|
PCHAR FileBuffer;
|
|
ULONG FileLength;
|
|
LARGE_INTEGER FileOffset;
|
|
PINFCACHE Cache;
|
|
|
|
CheckHeap();
|
|
|
|
*InfHandle = NULL;
|
|
*ErrorLine = (ULONG)-1;
|
|
|
|
/* Open the inf file */
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
FileName,
|
|
0,
|
|
NULL,
|
|
NULL);
|
|
|
|
Status = NtOpenFile(&FileHandle,
|
|
GENERIC_READ | SYNCHRONIZE,
|
|
&ObjectAttributes,
|
|
&IoStatusBlock,
|
|
FILE_SHARE_READ,
|
|
FILE_SYNCHRONOUS_IO_NONALERT | FILE_NON_DIRECTORY_FILE);
|
|
if (!INF_SUCCESS(Status))
|
|
{
|
|
DPRINT("NtOpenFile() failed (Status %lx)\n", Status);
|
|
return(Status);
|
|
}
|
|
|
|
DPRINT("NtOpenFile() successful\n");
|
|
|
|
/* Query file size */
|
|
Status = NtQueryInformationFile(FileHandle,
|
|
&IoStatusBlock,
|
|
&FileInfo,
|
|
sizeof(FILE_STANDARD_INFORMATION),
|
|
FileStandardInformation);
|
|
if (!INF_SUCCESS(Status))
|
|
{
|
|
DPRINT("NtQueryInformationFile() failed (Status %lx)\n", Status);
|
|
NtClose(FileHandle);
|
|
return(Status);
|
|
}
|
|
|
|
FileLength = FileInfo.EndOfFile.u.LowPart;
|
|
|
|
DPRINT("File size: %lu\n", FileLength);
|
|
|
|
/* Allocate file buffer */
|
|
FileBuffer = MALLOC(FileLength + 1);
|
|
if (FileBuffer == NULL)
|
|
{
|
|
DPRINT1("MALLOC() failed\n");
|
|
NtClose(FileHandle);
|
|
return(INF_STATUS_INSUFFICIENT_RESOURCES);
|
|
}
|
|
|
|
/* Read file */
|
|
FileOffset.QuadPart = 0ULL;
|
|
Status = NtReadFile(FileHandle,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FileBuffer,
|
|
FileLength,
|
|
&FileOffset,
|
|
NULL);
|
|
|
|
/* Append string terminator */
|
|
FileBuffer[FileLength] = 0;
|
|
|
|
NtClose(FileHandle);
|
|
|
|
if (!INF_SUCCESS(Status))
|
|
{
|
|
DPRINT("NtReadFile() failed (Status %lx)\n", Status);
|
|
FREE(FileBuffer);
|
|
return(Status);
|
|
}
|
|
|
|
/* Allocate infcache header */
|
|
Cache = (PINFCACHE)MALLOC(sizeof(INFCACHE));
|
|
if (Cache == NULL)
|
|
{
|
|
DPRINT("MALLOC() failed\n");
|
|
FREE(FileBuffer);
|
|
return(INF_STATUS_INSUFFICIENT_RESOURCES);
|
|
}
|
|
|
|
/* Initialize inicache header */
|
|
ZEROMEMORY(Cache,
|
|
sizeof(INFCACHE));
|
|
|
|
/* Parse the inf buffer */
|
|
Status = InfpParseBuffer (Cache,
|
|
FileBuffer,
|
|
FileBuffer + FileLength,
|
|
ErrorLine);
|
|
if (!INF_SUCCESS(Status))
|
|
{
|
|
FREE(Cache);
|
|
Cache = NULL;
|
|
}
|
|
|
|
/* Free file buffer */
|
|
FREE(FileBuffer);
|
|
|
|
*InfHandle = (HINF)Cache;
|
|
|
|
return(Status);
|
|
}
|
|
|
|
|
|
VOID
|
|
InfCloseFile(HINF InfHandle)
|
|
{
|
|
PINFCACHE Cache;
|
|
|
|
Cache = (PINFCACHE)InfHandle;
|
|
|
|
if (Cache == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
while (Cache->FirstSection != NULL)
|
|
{
|
|
Cache->FirstSection = InfpFreeSection(Cache->FirstSection);
|
|
}
|
|
Cache->LastSection = NULL;
|
|
|
|
FREE(Cache);
|
|
|
|
if (0 < InfpHeapRefCount)
|
|
{
|
|
InfpHeapRefCount--;
|
|
if (0 == InfpHeapRefCount)
|
|
{
|
|
RtlDestroyHeap(InfpHeap);
|
|
InfpHeap = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* EOF */
|