830 lines
23 KiB
C
830 lines
23 KiB
C
|
/*** acpins.c - ACPI Name Space functions
|
||
|
*
|
||
|
* Copyright (c) 1996,1997 Microsoft Corporation
|
||
|
* Author: Michael Tsang (MikeTs)
|
||
|
* Created 09/09/96
|
||
|
*
|
||
|
* MODIFICATION HISTORY
|
||
|
*/
|
||
|
|
||
|
#include "pch.h"
|
||
|
|
||
|
#ifdef LOCKABLE_PRAGMA
|
||
|
#pragma ACPI_LOCKABLE_DATA
|
||
|
#pragma ACPI_LOCKABLE_CODE
|
||
|
#endif
|
||
|
|
||
|
/***LP GetNameSpaceObject - Find a name space object
|
||
|
*
|
||
|
* ENTRY
|
||
|
* pszObjPath -> object path string
|
||
|
* pnsScope - object scope to start the search (NULL means root)
|
||
|
* ppnsObj -> to hold the object found
|
||
|
* dwfNS - flags
|
||
|
*
|
||
|
* EXIT-SUCCESS
|
||
|
* returns STATUS_SUCCESS
|
||
|
* EXIT-FAILURE
|
||
|
* returns AMLIERR_ code
|
||
|
*/
|
||
|
|
||
|
NTSTATUS LOCAL GetNameSpaceObject(PSZ pszObjPath, PNSOBJ pnsScope, PPNSOBJ ppns,
|
||
|
ULONG dwfNS)
|
||
|
{
|
||
|
TRACENAME("GETNAMESPACEOBJECT")
|
||
|
NTSTATUS rc = STATUS_SUCCESS;
|
||
|
PSZ psz;
|
||
|
|
||
|
ENTER(3, ("GetNameSpaceObject(ObjPath=%s,Scope=%s,ppns=%x,Flags=%x)\n",
|
||
|
pszObjPath, GetObjectPath(pnsScope), ppns, dwfNS));
|
||
|
|
||
|
if (pnsScope == NULL)
|
||
|
pnsScope = gpnsNameSpaceRoot;
|
||
|
|
||
|
if (*pszObjPath == '\\')
|
||
|
{
|
||
|
psz = &pszObjPath[1];
|
||
|
pnsScope = gpnsNameSpaceRoot;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
psz = pszObjPath;
|
||
|
|
||
|
while ((*psz == '^') && (pnsScope != NULL))
|
||
|
{
|
||
|
psz++;
|
||
|
pnsScope = pnsScope->pnsParent;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
*ppns = pnsScope;
|
||
|
|
||
|
if (pnsScope == NULL)
|
||
|
rc = AMLIERR_OBJ_NOT_FOUND;
|
||
|
else if (*psz != '\0')
|
||
|
{
|
||
|
BOOLEAN fSearchUp;
|
||
|
PNSOBJ pns;
|
||
|
|
||
|
fSearchUp = (BOOLEAN)(!(dwfNS & NSF_LOCAL_SCOPE) &&
|
||
|
(pszObjPath[0] != '\\') &&
|
||
|
(pszObjPath[0] != '^') &&
|
||
|
(STRLEN(pszObjPath) <= sizeof(NAMESEG)));
|
||
|
for (;;)
|
||
|
{
|
||
|
do
|
||
|
{
|
||
|
if ((pns = pnsScope->pnsFirstChild) == NULL)
|
||
|
rc = AMLIERR_OBJ_NOT_FOUND;
|
||
|
else
|
||
|
{
|
||
|
BOOLEAN fFound;
|
||
|
PSZ pszEnd;
|
||
|
ULONG dwLen;
|
||
|
NAMESEG dwName;
|
||
|
|
||
|
if ((pszEnd = STRCHR(psz, '.')) != NULL)
|
||
|
dwLen = (ULONG)(pszEnd - psz);
|
||
|
else
|
||
|
dwLen = STRLEN(psz);
|
||
|
|
||
|
if (dwLen > sizeof(NAMESEG))
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_INVALID_NAME,
|
||
|
("GetNameSpaceObject: invalid name - %s",
|
||
|
pszObjPath));
|
||
|
|
||
|
// Satisfy the compiler...
|
||
|
fFound = FALSE;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
dwName = NAMESEG_BLANK;
|
||
|
MEMCPY(&dwName, psz, dwLen);
|
||
|
//
|
||
|
// Search all siblings for a matching NameSeg.
|
||
|
//
|
||
|
fFound = FALSE;
|
||
|
do
|
||
|
{
|
||
|
if (pns->dwNameSeg == dwName)
|
||
|
{
|
||
|
pnsScope = pns;
|
||
|
fFound = TRUE;
|
||
|
break;
|
||
|
}
|
||
|
pns = (PNSOBJ)pns->list.plistNext;
|
||
|
} while (pns != pns->pnsParent->pnsFirstChild);
|
||
|
}
|
||
|
|
||
|
if (rc == STATUS_SUCCESS)
|
||
|
{
|
||
|
if (!fFound)
|
||
|
rc = AMLIERR_OBJ_NOT_FOUND;
|
||
|
else
|
||
|
{
|
||
|
psz += dwLen;
|
||
|
if (*psz == '.')
|
||
|
{
|
||
|
psz++;
|
||
|
}
|
||
|
else if (*psz == '\0')
|
||
|
{
|
||
|
*ppns = pnsScope;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} while (rc == STATUS_SUCCESS);
|
||
|
|
||
|
if ((rc == AMLIERR_OBJ_NOT_FOUND) && fSearchUp &&
|
||
|
(pnsScope != NULL) && (pnsScope->pnsParent != NULL))
|
||
|
{
|
||
|
pnsScope = pnsScope->pnsParent;
|
||
|
rc = STATUS_SUCCESS;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ((dwfNS & NSF_WARN_NOTFOUND) && (rc == AMLIERR_OBJ_NOT_FOUND))
|
||
|
{
|
||
|
rc = AMLI_LOGERR(rc, ("GetNameSpaceObject: object %s not found",
|
||
|
pszObjPath));
|
||
|
}
|
||
|
|
||
|
if (rc != STATUS_SUCCESS)
|
||
|
{
|
||
|
*ppns = NULL;
|
||
|
}
|
||
|
|
||
|
EXIT(3, ("GetNameSpaceObject=%x (pns=%x)\n", rc, *ppns));
|
||
|
return rc;
|
||
|
} //GetNameSpaceObject
|
||
|
|
||
|
/***LP CreateNameSpaceObject - Create a name space object under current scope
|
||
|
*
|
||
|
* ENTRY
|
||
|
* pheap -> HEAP
|
||
|
* pszName -> name string of the object (NULL if creating noname object)
|
||
|
* pnsScope - scope to create object under (NULL means root)
|
||
|
* powner -> object owner
|
||
|
* ppns -> to hold the pointer to the new object (can be NULL)
|
||
|
* dwfNS - flags
|
||
|
*
|
||
|
* EXIT-SUCCESS
|
||
|
* returns STATUS_SUCCESS
|
||
|
* EXIT-FAILURE
|
||
|
* returns AMLIERR_ code
|
||
|
*/
|
||
|
|
||
|
NTSTATUS LOCAL CreateNameSpaceObject(PHEAP pheap, PSZ pszName, PNSOBJ pnsScope,
|
||
|
POBJOWNER powner, PPNSOBJ ppns,
|
||
|
ULONG dwfNS)
|
||
|
{
|
||
|
TRACENAME("CREATENAMESPACEOBJECT")
|
||
|
NTSTATUS rc = STATUS_SUCCESS;
|
||
|
PNSOBJ pns = NULL;
|
||
|
|
||
|
ENTER(3, ("CreateNameSpaceObject(pheap=%x,Name=%s,pnsScope=%x,powner=%x,ppns=%x,Flags=%x)\n",
|
||
|
pheap, pszName? pszName: "<null>", pnsScope, powner, ppns, dwfNS));
|
||
|
|
||
|
if (pnsScope == NULL)
|
||
|
pnsScope = gpnsNameSpaceRoot;
|
||
|
|
||
|
if (pszName == NULL)
|
||
|
{
|
||
|
if ((pns = NEWNSOBJ(pheap, sizeof(NSOBJ))) == NULL)
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_OUT_OF_MEM,
|
||
|
("CreateNameSpaceObject: fail to allocate name space object"));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ASSERT(gpnsNameSpaceRoot != NULL);
|
||
|
MEMZERO(pns, sizeof(NSOBJ));
|
||
|
pns->pnsParent = pnsScope;
|
||
|
InsertOwnerObjList(powner, pns);
|
||
|
ListInsertTail(&pns->list,
|
||
|
(PPLIST)&pnsScope->pnsFirstChild);
|
||
|
}
|
||
|
}
|
||
|
else if ((*pszName != '\0') &&
|
||
|
((rc = GetNameSpaceObject(pszName, pnsScope, &pns,
|
||
|
NSF_LOCAL_SCOPE)) == STATUS_SUCCESS))
|
||
|
{
|
||
|
if (!(dwfNS & NSF_EXIST_OK))
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_OBJ_ALREADY_EXIST,
|
||
|
("CreateNameSpaceObject: object already exist - %s",
|
||
|
pszName));
|
||
|
}
|
||
|
}
|
||
|
else if ((*pszName == '\0') || (rc == AMLIERR_OBJ_NOT_FOUND))
|
||
|
{
|
||
|
rc = STATUS_SUCCESS;
|
||
|
//
|
||
|
// Are we creating root?
|
||
|
//
|
||
|
if (STRCMP(pszName, "\\") == 0)
|
||
|
{
|
||
|
ASSERT(gpnsNameSpaceRoot == NULL);
|
||
|
ASSERT(powner == NULL);
|
||
|
if ((pns = NEWNSOBJ(pheap, sizeof(NSOBJ))) == NULL)
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_OUT_OF_MEM,
|
||
|
("CreateNameSpaceObject: fail to allocate name space object"));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
MEMZERO(pns, sizeof(NSOBJ));
|
||
|
pns->dwNameSeg = NAMESEG_ROOT;
|
||
|
gpnsNameSpaceRoot = pns;
|
||
|
InsertOwnerObjList(powner, pns);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
PSZ psz;
|
||
|
PNSOBJ pnsParent;
|
||
|
|
||
|
if ((psz = STRRCHR(pszName, '.')) != NULL)
|
||
|
{
|
||
|
*psz = '\0';
|
||
|
psz++;
|
||
|
rc = GetNameSpaceObject(pszName, pnsScope, &pnsParent,
|
||
|
NSF_LOCAL_SCOPE | NSF_WARN_NOTFOUND);
|
||
|
}
|
||
|
else if (*pszName == '\\')
|
||
|
{
|
||
|
psz = &pszName[1];
|
||
|
//
|
||
|
// By this time, we'd better created root already.
|
||
|
//
|
||
|
ASSERT(gpnsNameSpaceRoot != NULL);
|
||
|
pnsParent = gpnsNameSpaceRoot;
|
||
|
}
|
||
|
else if (*pszName == '^')
|
||
|
{
|
||
|
psz = pszName;
|
||
|
pnsParent = pnsScope;
|
||
|
while ((*psz == '^') && (pnsParent != NULL))
|
||
|
{
|
||
|
pnsParent = pnsParent->pnsParent;
|
||
|
psz++;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ASSERT(pnsScope != NULL);
|
||
|
psz = pszName;
|
||
|
pnsParent = pnsScope;
|
||
|
}
|
||
|
|
||
|
if (rc == STATUS_SUCCESS)
|
||
|
{
|
||
|
int iLen = STRLEN(psz);
|
||
|
|
||
|
if ((*psz != '\0') && (iLen > sizeof(NAMESEG)))
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_INVALID_NAME,
|
||
|
("CreateNameSpaceObject: invalid name - %s",
|
||
|
psz));
|
||
|
}
|
||
|
else if ((pns = NEWNSOBJ(pheap, sizeof(NSOBJ)))
|
||
|
== NULL)
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_OUT_OF_MEM,
|
||
|
("CreateNameSpaceObject: fail to allocate name space object"));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
MEMZERO(pns, sizeof(NSOBJ));
|
||
|
|
||
|
if (*pszName == '\0')
|
||
|
pns->dwNameSeg = NAMESEG_NONE;
|
||
|
else
|
||
|
{
|
||
|
pns->dwNameSeg = NAMESEG_BLANK;
|
||
|
MEMCPY(&pns->dwNameSeg, psz, iLen);
|
||
|
}
|
||
|
|
||
|
pns->pnsParent = pnsParent;
|
||
|
InsertOwnerObjList(powner, pns);
|
||
|
ListInsertTail(&pns->list,
|
||
|
(PPLIST)&pnsParent->pnsFirstChild);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ((rc == STATUS_SUCCESS) && (ppns != NULL))
|
||
|
*ppns = pns;
|
||
|
|
||
|
EXIT(3, ("CreateNameSpaceObject=%x (pns=%x)\n", rc, pns));
|
||
|
return rc;
|
||
|
} //CreateNameSpaceObject
|
||
|
|
||
|
/***LP FreeNameSpaceObjects - Free Name Space object and its children
|
||
|
*
|
||
|
* ENTRY
|
||
|
* pnsObj -> name space object
|
||
|
*
|
||
|
* EXIT
|
||
|
* None
|
||
|
*/
|
||
|
|
||
|
VOID LOCAL FreeNameSpaceObjects(PNSOBJ pnsObj)
|
||
|
{
|
||
|
TRACENAME("FREENAMESPACEOBJECTS")
|
||
|
PNSOBJ pns, pnsSibling, pnsParent;
|
||
|
#ifdef DEBUGGER
|
||
|
POBJSYM pos;
|
||
|
#endif
|
||
|
|
||
|
ENTER(3, ("FreeNameSpaceObjects(Obj=%s)\n", GetObjectPath(pnsObj)));
|
||
|
|
||
|
ASSERT(pnsObj != NULL);
|
||
|
|
||
|
for (pns = pnsObj; pns != NULL;)
|
||
|
{
|
||
|
while (pns->pnsFirstChild != NULL)
|
||
|
{
|
||
|
pns = pns->pnsFirstChild;
|
||
|
}
|
||
|
|
||
|
pnsSibling = NSGETNEXTSIBLING(pns);
|
||
|
pnsParent = NSGETPARENT(pns);
|
||
|
|
||
|
ENTER(4, ("FreeNSObj(Obj=%s)\n", GetObjectPath(pns)));
|
||
|
#ifdef DEBUGGER
|
||
|
//
|
||
|
// If I am in the symbol list, get rid of it before I die.
|
||
|
//
|
||
|
for (pos = gDebugger.posSymbolList; pos != NULL; pos = pos->posNext)
|
||
|
{
|
||
|
if (pns == pos->pnsObj)
|
||
|
{
|
||
|
if (pos->posPrev != NULL)
|
||
|
pos->posPrev->posNext = pos->posNext;
|
||
|
|
||
|
if (pos->posNext != NULL)
|
||
|
pos->posNext->posPrev = pos->posPrev;
|
||
|
|
||
|
if (pos == gDebugger.posSymbolList)
|
||
|
gDebugger.posSymbolList = pos->posNext;
|
||
|
|
||
|
FREESYOBJ(pos);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
//
|
||
|
// All my children are gone, I must die now.
|
||
|
//
|
||
|
ASSERT(pns->pnsFirstChild == NULL);
|
||
|
|
||
|
if ((pns->ObjData.dwDataType == OBJTYPE_OPREGION) &&
|
||
|
(((POPREGIONOBJ)pns->ObjData.pbDataBuff)->bRegionSpace ==
|
||
|
REGSPACE_MEM))
|
||
|
{
|
||
|
ASSERT(KeGetCurrentIrql() == PASSIVE_LEVEL);
|
||
|
MmUnmapIoSpace((PVOID)
|
||
|
((POPREGIONOBJ)pns->ObjData.pbDataBuff)->uipOffset,
|
||
|
((POPREGIONOBJ)pns->ObjData.pbDataBuff)->dwLen);
|
||
|
}
|
||
|
|
||
|
if (pns->pnsParent == NULL)
|
||
|
{
|
||
|
//
|
||
|
// I am root!
|
||
|
//
|
||
|
ASSERT(pns == gpnsNameSpaceRoot);
|
||
|
gpnsNameSpaceRoot = NULL;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ListRemoveEntry(&pns->list,
|
||
|
(PPLIST)&pns->pnsParent->pnsFirstChild);
|
||
|
}
|
||
|
//
|
||
|
// Free any attached data buffer if any
|
||
|
//
|
||
|
FreeDataBuffs(&pns->ObjData, 1);
|
||
|
//
|
||
|
// Free myself
|
||
|
//
|
||
|
if (pns->dwRefCount == 0)
|
||
|
{
|
||
|
FREENSOBJ(pns);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pns->ObjData.dwfData |= DATAF_NSOBJ_DEFUNC;
|
||
|
ListInsertTail(&pns->list, &gplistDefuncNSObjs);
|
||
|
}
|
||
|
EXIT(4, ("FreeNSObj!\n"));
|
||
|
|
||
|
if (pns == pnsObj)
|
||
|
{
|
||
|
//
|
||
|
// I was the last one, done!
|
||
|
//
|
||
|
pns = NULL;
|
||
|
}
|
||
|
else if (pnsSibling != NULL)
|
||
|
{
|
||
|
//
|
||
|
// I have siblings, go kill them.
|
||
|
//
|
||
|
pns = pnsSibling;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ASSERT(pnsParent != NULL);
|
||
|
pns = pnsParent;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
EXIT(3, ("FreeNameSpaceObjects!\n"));
|
||
|
} //FreeNameSpaceObjects
|
||
|
|
||
|
/***LP LoadDDB - Load and parse Differentiated Definition Block
|
||
|
*
|
||
|
* ENTRY
|
||
|
* pctxt -> CTXT
|
||
|
* pdsdt -> DSDT block
|
||
|
* pnsScope -> current scope
|
||
|
* ppowner -> to hold new object owner
|
||
|
*
|
||
|
* EXIT-SUCCESS
|
||
|
* returns STATUS_SUCCESS
|
||
|
* EXIT-FAILURE
|
||
|
* returns AMLIERR_ code
|
||
|
*/
|
||
|
|
||
|
NTSTATUS
|
||
|
LOCAL
|
||
|
LoadDDB(
|
||
|
PCTXT pctxt,
|
||
|
PDSDT pdsdt,
|
||
|
PNSOBJ pnsScope,
|
||
|
POBJOWNER *ppowner
|
||
|
)
|
||
|
{
|
||
|
BOOLEAN freeTable = FALSE;
|
||
|
NTSTATUS rc = STATUS_SUCCESS;
|
||
|
|
||
|
if (!ValidateTable(pdsdt)) {
|
||
|
|
||
|
rc = AMLI_LOGERR(
|
||
|
AMLIERR_INVALID_TABLE,
|
||
|
("LoadDDB: invalid table %s at 0x%08x",
|
||
|
NameSegString(pdsdt->Header.Signature), pdsdt)
|
||
|
);
|
||
|
freeTable = TRUE;
|
||
|
|
||
|
} else {
|
||
|
|
||
|
rc = NewObjOwner( gpheapGlobal, ppowner);
|
||
|
if (rc == STATUS_SUCCESS) {
|
||
|
|
||
|
if (pctxt->pcall == NULL) {
|
||
|
|
||
|
rc = PushCall(pctxt, NULL, &pctxt->Result);
|
||
|
|
||
|
}
|
||
|
if (rc == STATUS_SUCCESS) {
|
||
|
|
||
|
#ifdef DEBUGGER
|
||
|
gDebugger.pbBlkBegin = pdsdt->DiffDefBlock;
|
||
|
gDebugger.pbBlkEnd = (PUCHAR)pdsdt + pdsdt->Header.Length;
|
||
|
#endif
|
||
|
|
||
|
rc = PushScope(
|
||
|
pctxt,
|
||
|
pdsdt->DiffDefBlock,
|
||
|
(PUCHAR)pdsdt + pdsdt->Header.Length, pctxt->pbOp,
|
||
|
pnsScope, *ppowner, gpheapGlobal, &pctxt->Result
|
||
|
);
|
||
|
|
||
|
}
|
||
|
|
||
|
} else {
|
||
|
|
||
|
freeTable = TRUE;
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
if (freeTable) {
|
||
|
|
||
|
pctxt->powner = NULL;
|
||
|
FreeContext(pctxt);
|
||
|
|
||
|
}
|
||
|
return rc;
|
||
|
} //LoadDDB
|
||
|
|
||
|
/***LP LoadMemDDB - Load DDB from physical memory
|
||
|
*
|
||
|
* ENTRY
|
||
|
* pctxt -> CTXT
|
||
|
* pDDB -> beginning of DDB
|
||
|
* ppowner -> to hold owner handle
|
||
|
*
|
||
|
* EXIT-SUCCESS
|
||
|
* returns STATUS_SUCCESS
|
||
|
* EXIT-FAILURE
|
||
|
* returns AMLIERR_ code
|
||
|
*/
|
||
|
|
||
|
NTSTATUS LOCAL LoadMemDDB(PCTXT pctxt, PDSDT pDDB, POBJOWNER *ppowner)
|
||
|
{
|
||
|
TRACENAME("LOADMEMDDB")
|
||
|
NTSTATUS rc = STATUS_SUCCESS;
|
||
|
|
||
|
ENTER(3, ("LoadMemDDB(pctxt=%x,Addr=%x,ppowner=%x)\n",
|
||
|
pctxt, pDDB, ppowner));
|
||
|
|
||
|
if ((ghValidateTable.pfnHandler != NULL) &&
|
||
|
((rc = ((PFNVT)ghValidateTable.pfnHandler)(pDDB,
|
||
|
ghValidateTable.uipParam)) !=
|
||
|
STATUS_SUCCESS))
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_INVALID_TABLE,
|
||
|
("LoadMemDDB: table validation failed (rc=%x)",
|
||
|
rc));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
rc = LoadDDB(pctxt, pDDB, pctxt->pnsScope, ppowner);
|
||
|
}
|
||
|
|
||
|
EXIT(3, ("LoadMemDDB=%x (powner=%x)\n", rc, *ppowner));
|
||
|
return rc;
|
||
|
} //LoadMemDDB
|
||
|
|
||
|
/***LP LoadFieldUnitDDB - Load DDB from a FieldUnit object
|
||
|
*
|
||
|
* ENTRY
|
||
|
* pctxt -> CTXT
|
||
|
* pdataObj -> FieldUnit object
|
||
|
* ppowner -> to hold owner handle
|
||
|
*
|
||
|
* EXIT-SUCCESS
|
||
|
* returns STATUS_SUCCESS
|
||
|
* EXIT-FAILURE
|
||
|
* returns AMLIERR_ code
|
||
|
*/
|
||
|
|
||
|
NTSTATUS LOCAL LoadFieldUnitDDB(PCTXT pctxt, POBJDATA pdataObj,
|
||
|
POBJOWNER *ppowner)
|
||
|
{
|
||
|
TRACENAME("LOADFIELDUNITDDB")
|
||
|
NTSTATUS rc = STATUS_SUCCESS;
|
||
|
POBJDATA pdataTmp;
|
||
|
DESCRIPTION_HEADER *pdh;
|
||
|
|
||
|
ENTER(3, ("LoadFieldUnitDDB(pctxt=%x,pdataObj=%x,ppowner=%x)\n",
|
||
|
pctxt, pdataObj, ppowner));
|
||
|
|
||
|
if ((pdataTmp = NEWODOBJ(pctxt->pheapCurrent, sizeof(OBJDATA))) == NULL)
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_OUT_OF_MEM,
|
||
|
("LoadFieldUnitDDB: failed to allocate temp. object data"));
|
||
|
}
|
||
|
else if ((pdh = NEWBDOBJ(gpheapGlobal, sizeof(DESCRIPTION_HEADER))) == NULL)
|
||
|
{
|
||
|
FREEODOBJ(pdataTmp);
|
||
|
rc = AMLI_LOGERR(AMLIERR_OUT_OF_MEM,
|
||
|
("LoadFieldUnitDDB: failed to allocate description header"));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
PUCHAR pbTable;
|
||
|
|
||
|
MEMZERO(pdataTmp, sizeof(OBJDATA));
|
||
|
pdataTmp->dwDataType = OBJTYPE_BUFFDATA;
|
||
|
pdataTmp->dwDataLen = sizeof(DESCRIPTION_HEADER);
|
||
|
pdataTmp->pbDataBuff = (PUCHAR)pdh;
|
||
|
|
||
|
if ((rc = ReadObject(pctxt, pdataObj, pdataTmp)) == STATUS_SUCCESS)
|
||
|
{
|
||
|
if ((pbTable = NEWBDOBJ(gpheapGlobal, pdh->Length)) == NULL)
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_OUT_OF_MEM,
|
||
|
("LoadFieldUnitDDB: failed to allocate buffer"));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
MEMCPY(pbTable, pdh, sizeof(DESCRIPTION_HEADER));
|
||
|
pdataTmp->dwDataLen = pdh->Length - sizeof(DESCRIPTION_HEADER);
|
||
|
pdataTmp->pbDataBuff = pbTable + sizeof(DESCRIPTION_HEADER);
|
||
|
|
||
|
if ((rc = ReadObject(pctxt, pdataObj, pdataTmp)) ==
|
||
|
STATUS_SUCCESS)
|
||
|
{
|
||
|
if ((ghValidateTable.pfnHandler != NULL) &&
|
||
|
((rc = ((PFNVT)ghValidateTable.pfnHandler)(
|
||
|
(PDSDT)pbTable, ghValidateTable.uipParam)) !=
|
||
|
STATUS_SUCCESS))
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_INVALID_TABLE,
|
||
|
("LoadFieldUnitDDB: table validation failed (rc=%x)",
|
||
|
rc));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
rc = LoadDDB(pctxt, (PDSDT)pbTable, pctxt->pnsScope,
|
||
|
ppowner);
|
||
|
}
|
||
|
}
|
||
|
else if (rc == AMLISTA_PENDING)
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_FATAL,
|
||
|
("LoadFieldUnitDDB: definition block loading cannot block"));
|
||
|
}
|
||
|
|
||
|
FREEBDOBJ(pbTable);
|
||
|
}
|
||
|
}
|
||
|
else if (rc == AMLISTA_PENDING)
|
||
|
{
|
||
|
rc = AMLI_LOGERR(AMLIERR_FATAL,
|
||
|
("LoadFieldUnitDDB: definition block loading cannot block"));
|
||
|
}
|
||
|
|
||
|
FREEBDOBJ(pdh);
|
||
|
FREEODOBJ(pdataTmp);
|
||
|
}
|
||
|
|
||
|
EXIT(3, ("LoadFieldUnitDDB=%x (powner=%x)\n", rc, *ppowner));
|
||
|
return rc;
|
||
|
} //LoadFieldUnitDDB
|
||
|
|
||
|
/***LP UnloadDDB - Unload Differentiated Definition Block
|
||
|
*
|
||
|
* ENTRY
|
||
|
* powner -> OBJOWNER
|
||
|
*
|
||
|
* EXIT-SUCCESS
|
||
|
* returns STATUS_SUCCESS
|
||
|
* EXIT-FAILURE
|
||
|
* returns AMLIERR_ code
|
||
|
*/
|
||
|
|
||
|
VOID LOCAL UnloadDDB(POBJOWNER powner)
|
||
|
{
|
||
|
TRACENAME("UNLOADDDB")
|
||
|
|
||
|
ENTER(3, ("UnloadDDB(powner=%x)\n", powner));
|
||
|
//
|
||
|
// Walk name space and remove all objects belongs to this DDB.
|
||
|
//
|
||
|
FreeObjOwner(powner, TRUE);
|
||
|
#ifdef DEBUG
|
||
|
{
|
||
|
KIRQL oldIrql;
|
||
|
|
||
|
KeAcquireSpinLock( &gdwGHeapSpinLock, &oldIrql );
|
||
|
gdwGHeapSnapshot = gdwGlobalHeapSize;
|
||
|
KeReleaseSpinLock( &gdwGHeapSpinLock, oldIrql );
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
EXIT(3, ("UnloadDDB!\n"));
|
||
|
} //UnloadDDB
|
||
|
|
||
|
/***LP EvalPackageElement - Evaluate a package element
|
||
|
*
|
||
|
* ENTRY
|
||
|
* ppkg -> package object
|
||
|
* iPkgIndex - package index (0-based)
|
||
|
* pdataResult -> result object
|
||
|
*
|
||
|
* EXIT-SUCCESS
|
||
|
* returns STATUS_SUCCESS
|
||
|
* EXIT-FAILURE
|
||
|
* returns AMLIERR_ code
|
||
|
*/
|
||
|
|
||
|
NTSTATUS LOCAL EvalPackageElement(PPACKAGEOBJ ppkg, int iPkgIndex,
|
||
|
POBJDATA pdataResult)
|
||
|
{
|
||
|
TRACENAME("EVALPACKAGEELEMENT")
|
||
|
NTSTATUS rc = STATUS_SUCCESS;
|
||
|
|
||
|
ENTER(3, ("EvalPackageElement(ppkg=%x,Index=%d,pdataResult=%x)\n",
|
||
|
ppkg, iPkgIndex, pdataResult));
|
||
|
|
||
|
ASSERT(pdataResult != NULL);
|
||
|
if (iPkgIndex >= (int)ppkg->dwcElements)
|
||
|
{
|
||
|
rc = AMLIERR_INDEX_TOO_BIG;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
rc = DupObjData(gpheapGlobal, pdataResult, &ppkg->adata[iPkgIndex]);
|
||
|
}
|
||
|
|
||
|
EXIT(3, ("EvalPackageElement=%x (Type=%s,Value=%x,Len=%d,Buff=%x)\n",
|
||
|
rc, GetObjectTypeName(pdataResult->dwDataType),
|
||
|
pdataResult->uipDataValue, pdataResult->dwDataLen,
|
||
|
pdataResult->pbDataBuff));
|
||
|
return rc;
|
||
|
} //EvalPackageElement
|
||
|
|
||
|
#ifdef DEBUGGER
|
||
|
/***LP DumpNameSpaceObject - Dump name space object
|
||
|
*
|
||
|
* ENTRY
|
||
|
* pszPath -> name space path string
|
||
|
* fRecursive - TRUE if also dump the subtree recursively
|
||
|
*
|
||
|
* EXIT-SUCCESS
|
||
|
* returns STATUS_SUCCESS
|
||
|
* EXIT-FAILURE
|
||
|
* returns AMLIERR_ code
|
||
|
*/
|
||
|
|
||
|
LONG LOCAL DumpNameSpaceObject(PSZ pszPath, BOOLEAN fRecursive)
|
||
|
{
|
||
|
TRACENAME("DUMPNAMESPACEOBJECT")
|
||
|
NTSTATUS rc = STATUS_SUCCESS;
|
||
|
PNSOBJ pns;
|
||
|
char szName[sizeof(NAMESEG) + 1];
|
||
|
|
||
|
ENTER(3, ("DumpNameSpaceObject(Path=%s,fRecursive=%x)\n",
|
||
|
pszPath, fRecursive));
|
||
|
|
||
|
if ((rc = GetNameSpaceObject(pszPath, NULL, &pns,
|
||
|
NSF_LOCAL_SCOPE)) == STATUS_SUCCESS)
|
||
|
{
|
||
|
PRINTF("\nACPI Name Space: %s (%x)\n", pszPath, pns);
|
||
|
if (!fRecursive)
|
||
|
{
|
||
|
STRCPYN(szName, (PSZ)&pns->dwNameSeg, sizeof(NAMESEG));
|
||
|
DumpObject(&pns->ObjData, szName, 0);
|
||
|
}
|
||
|
else
|
||
|
DumpNameSpaceTree(pns, 0);
|
||
|
}
|
||
|
else if (rc == AMLIERR_OBJ_NOT_FOUND)
|
||
|
{
|
||
|
PRINTF(MODNAME "_ERROR: object %s not found\n", pszPath);
|
||
|
}
|
||
|
|
||
|
EXIT(3, ("DumpNameSpaceObject=%x\n", rc));
|
||
|
return rc;
|
||
|
} //DumpNameSpaceObject
|
||
|
|
||
|
/***LP DumpNameSpaceTree - Dump all the name space objects in the subtree
|
||
|
*
|
||
|
* ENTRY
|
||
|
* pnsObj -> name space subtree root
|
||
|
* dwLevel - indent level
|
||
|
*
|
||
|
* EXIT
|
||
|
* None
|
||
|
*/
|
||
|
|
||
|
VOID LOCAL DumpNameSpaceTree(PNSOBJ pnsObj, ULONG dwLevel)
|
||
|
{
|
||
|
TRACENAME("DUMPNAMESPACETREE")
|
||
|
PNSOBJ pns, pnsNext;
|
||
|
char szName[sizeof(NAMESEG) + 1];
|
||
|
|
||
|
ENTER(3, ("DumpNameSpaceTree(pns=%x,level=%d)\n", pnsObj, dwLevel));
|
||
|
//
|
||
|
// First, dump myself
|
||
|
//
|
||
|
STRCPYN(szName, (PSZ)&pnsObj->dwNameSeg, sizeof(NAMESEG));
|
||
|
DumpObject(&pnsObj->ObjData, szName, dwLevel);
|
||
|
//
|
||
|
// Then, recursively dump each of my children
|
||
|
//
|
||
|
for (pns = pnsObj->pnsFirstChild; pns != NULL; pns = pnsNext)
|
||
|
{
|
||
|
//
|
||
|
// If this is the last child, we have no more.
|
||
|
//
|
||
|
if ((pnsNext = (PNSOBJ)pns->list.plistNext) == pnsObj->pnsFirstChild)
|
||
|
pnsNext = NULL;
|
||
|
//
|
||
|
// Dump a child
|
||
|
//
|
||
|
DumpNameSpaceTree(pns, dwLevel + 1);
|
||
|
}
|
||
|
|
||
|
EXIT(3, ("DumpNameSpaceTree!\n"));
|
||
|
} //DumpNameSpaceTree
|
||
|
|
||
|
#endif //ifdef DEBUGGER
|