/*++ Copyright (C) 1996-1999 Microsoft Corporation Module Name: EVALTREE.CPP Abstract: WBEM Evaluation Tree History: --*/ #include "precomp.h" #include #pragma warning(disable:4786) #include #include #include #include "evaltree.h" #include "evaltree.inl" #include "TwoPropNode.h" #include "dumbnode.h" // #define DUMP_EVAL_TREES 1 HRESULT CoreGetNumParents(_IWmiObject* pClass, ULONG *plNumParents) { /* *plNumParents = ((CWbemObject*)pClass)->GetNumParents(); */ DWORD dwSize; HRESULT hres = pClass->GetDerivation(0, 0, plNumParents, &dwSize, NULL); if(hres != WBEM_E_BUFFER_TOO_SMALL && hres != S_OK) return WBEM_E_FAILED; return S_OK; } RELEASE_ME _IWmiObject* CoreGetEmbeddedObj(_IWmiObject* pObj, long lHandle) { _IWmiObject* pEmbedded = NULL; HRESULT hres = pObj->GetPropAddrByHandle(lHandle, 0, NULL, (void**)&pEmbedded); if(FAILED(hres)) return NULL; return pEmbedded; } INTERNAL CCompressedString* CoreGetPropertyString(_IWmiObject* pObj, long lHandle) { CCompressedString* pcs = NULL; HRESULT hres = pObj->GetPropAddrByHandle(lHandle, WMIOBJECT_FLAG_ENCODING_V1, NULL, (void**)&pcs); if(FAILED(hres)) return NULL; return pcs; } INTERNAL CCompressedString* CoreGetClassInternal(_IWmiObject* pObj) { CCompressedString* pcs = NULL; HRESULT hres = pObj->GetPropAddrByHandle(FASTOBJ_CLASSNAME_PROP_HANDLE, WMIOBJECT_FLAG_ENCODING_V1, NULL, (void**)&pcs); if(FAILED(hres)) return NULL; return pcs; } INTERNAL CCompressedString* CoreGetParentAtIndex(_IWmiObject* pObj, long lIndex) { return ((CWbemObject*)pObj)->GetParentAtIndex(lIndex); } bool CoreIsDerived(_IWmiObject* pThis, _IWmiObject* pFrom) { CCompressedString* pcs = CoreGetClassInternal(pFrom); if(pcs == NULL) return false; try { return (pThis->InheritsFrom(pcs->CreateWStringCopy()) == S_OK); } catch (CX_MemoryException) { return false; } } //****************************************************************************** //****************************************************************************** // TOKEN VALUE //****************************************************************************** //****************************************************************************** CTokenValue::CTokenValue() { VariantInit(&m_v); } CTokenValue::CTokenValue(CTokenValue& Other) { VariantInit(&m_v); *this = Other; } CTokenValue::~CTokenValue() { VariantClear(&m_v); } bool CTokenValue::SetVariant(VARIANT& v) { if(FAILED(VariantCopy(&m_v, &v))) return false; // Convert to a better type // ======================== if(V_VT(&v) == VT_I2 || V_VT(&v) == VT_UI1) { if(FAILED(VariantChangeType(&m_v, &m_v, 0, VT_I4))) return false; } else if(V_VT(&v) == VT_R4) { if(FAILED(VariantChangeType(&m_v, &m_v, 0, VT_R8))) return false; } return true; } void CTokenValue::operator=(CTokenValue& Other) { if(!SetVariant(Other.m_v)) throw CX_MemoryException(); } CTokenValue::operator unsigned __int64() const { if(V_VT(&m_v) == VT_I4) { return V_I4(&m_v); } else if(V_VT(&m_v) == VT_BSTR) { unsigned __int64 ui64; if(ReadUI64(V_BSTR(&m_v), ui64)) return ui64; else return 0; // TBD: errors } else return 0; } CTokenValue::operator unsigned long() const { if(V_VT(&m_v) == VT_I4) { return V_I4(&m_v); } else if(V_VT(&m_v) == VT_BSTR) { unsigned __int64 ui64; if(ReadUI64(V_BSTR(&m_v), ui64)) return (unsigned long)ui64; else return 0; // TBD: errors } else return 0; } CTokenValue::operator __int64() const { if(V_VT(&m_v) == VT_I4) { return V_I4(&m_v); } else if(V_VT(&m_v) == VT_BSTR) { __int64 i64; if(ReadI64(V_BSTR(&m_v), i64)) return i64; else return 0; // TBD: errors } else return 0; } CTokenValue::operator short() const { if(V_VT(&m_v) == VT_I4) return V_I4(&m_v); else if(V_VT(&m_v) == VT_BOOL) return V_BOOL(&m_v); else return 0; } CTokenValue::operator float() const { if(V_VT(&m_v) == VT_I4) return V_I4(&m_v); else if(V_VT(&m_v) == VT_R8) return V_R8(&m_v); else return 0; } CTokenValue::operator double() const { if(V_VT(&m_v) == VT_I4) return V_I4(&m_v); else if(V_VT(&m_v) == VT_R8) return V_R8(&m_v); else return 0; } CTokenValue::operator WString() const { if(V_VT(&m_v) == VT_BSTR) return WString(V_BSTR(&m_v)); else return WString(L""); } int CTokenValue::Compare(const CTokenValue& Other) const { if ( V_VT(&m_v) != V_VT(&Other.m_v) ) return V_VT(&m_v) - V_VT(&Other.m_v); switch(V_VT(&m_v)) { case VT_I4: return V_I4(&m_v) - V_I4(&Other.m_v); case VT_R8: return (V_R8(&m_v) - V_R8(&Other.m_v)<0 ? -1 : 1); case VT_BSTR: return wbem_wcsicmp(V_BSTR(&m_v), V_BSTR(&Other.m_v)); case VT_BOOL: return V_BOOL(&m_v) - V_BOOL(&Other.m_v); } return 0; } //****************************************************************************** //****************************************************************************** // OBJECT INFO //****************************************************************************** //****************************************************************************** bool CObjectInfo::SetLength(long lLength) { if(lLength > m_lLength) { delete [] m_apObj; m_apObj = new _IWmiObject*[lLength]; if (m_apObj == NULL) return false; memset(m_apObj, 0, lLength * sizeof(_IWmiObject*)); } m_lLength = lLength; return true; } void CObjectInfo::Clear() { for(long i = 1; i < m_lLength; i++) { if(m_apObj[i]) m_apObj[i]->Release(); m_apObj[i] = NULL; } m_apObj[0] = NULL; // do not release } void CObjectInfo::SetObjectAt(long lIndex, READ_ONLY _IWmiObject* pObj) { if(m_apObj[lIndex]) m_apObj[lIndex]->Release(); m_apObj[lIndex] = pObj; } CObjectInfo::~CObjectInfo() { for(long i = 0; i < m_lLength; i++) { if(m_apObj[i]) m_apObj[i]->Release(); } delete [] m_apObj; } //****************************************************************************** //****************************************************************************** // IMPLICATION LIST //****************************************************************************** //****************************************************************************** CImplicationList::CRecord::CRecord(const CImplicationList::CRecord& Other) : m_PropName(Other.m_PropName), m_pClass(Other.m_pClass), m_lObjIndex(Other.m_lObjIndex), m_nNull(Other.m_nNull) { if(m_pClass) m_pClass->AddRef(); for(int i = 0; i < Other.m_awsNotClasses.Size(); i++) { if(m_awsNotClasses.Add(Other.m_awsNotClasses[i]) != CWStringArray::no_error) { throw CX_MemoryException(); } } } CImplicationList::CRecord::~CRecord() { if(m_pClass) m_pClass->Release(); } HRESULT CImplicationList::CRecord::ImproveKnown(_IWmiObject* pClass) { if ( pClass == NULL ) { // // not much we can improve on, but NULL is still a valid param. // return WBEM_S_NO_ERROR; } if(m_nNull == EVAL_VALUE_TRUE) { // Contradiction // ============= return WBEM_E_TYPE_MISMATCH; } m_nNull = EVAL_VALUE_FALSE; ULONG lNumParents, lRecordNumParents; HRESULT hres = CoreGetNumParents(pClass, &lNumParents); if(FAILED(hres)) return hres; if(m_pClass) { hres = CoreGetNumParents(m_pClass, &lRecordNumParents); if(FAILED(hres)) return hres; } if(m_pClass == NULL || lNumParents > lRecordNumParents) { // Better than before. Check for inconsistencies // ============================================= if(m_pClass) { if(!CoreIsDerived(pClass, m_pClass)) return WBEM_E_TYPE_MISMATCH; } for(int i = 0; i < m_awsNotClasses.Size(); i++) { if(pClass->InheritsFrom(m_awsNotClasses[i]) == S_OK) { // Contradiction // ============= return WBEM_E_TYPE_MISMATCH; } } // Replace // ======= pClass->AddRef(); if(m_pClass) m_pClass->Release(); m_pClass = pClass; } else { // Verify that we are a parent of the selected and do not replace // ============================================================== if(!CoreIsDerived(m_pClass, pClass)) return WBEM_E_TYPE_MISMATCH; } return WBEM_S_NO_ERROR; } HRESULT CImplicationList::CRecord::ImproveKnownNot(LPCWSTR wszClassName) { if(m_nNull == EVAL_VALUE_TRUE) { // Contradiction // ============= return WBEM_E_TYPE_MISMATCH; } // Check for inconsistencies // ========================= if(m_nNull == EVAL_VALUE_FALSE && m_pClass && m_pClass->InheritsFrom(wszClassName) == S_OK) { // Contradiction // ============= return WBEM_E_TYPE_MISMATCH; } try { if(m_awsNotClasses.Add(wszClassName) < 0) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } m_nNull = EVAL_VALUE_FALSE; return WBEM_S_NO_ERROR; } HRESULT CImplicationList::CRecord::ImproveKnownNull() { if(m_nNull == EVAL_VALUE_FALSE) { return WBEM_E_TYPE_MISMATCH; } m_nNull = EVAL_VALUE_TRUE; return WBEM_S_NO_ERROR; } void CImplicationList::CRecord::Dump(FILE* f, int nOffset) { LPWSTR wszProp = m_PropName.GetText(); CEvalNode::PrintOffset(f, nOffset); fprintf(f, "Learn about %S:\n", wszProp); if(m_pClass) { VARIANT v; m_pClass->Get(L"__CLASS", 0, &v, NULL, NULL); CEvalNode::PrintOffset(f, nOffset+1); fprintf(f, "Is of class %S\n", V_BSTR(&v)); } for(int i = 0; i < m_awsNotClasses.Size(); i++) { CEvalNode::PrintOffset(f, nOffset+1); fprintf(f, "Not of class %S\n", m_awsNotClasses[i]); } if(m_nNull == EVAL_VALUE_TRUE) { CEvalNode::PrintOffset(f, nOffset+1); fprintf(f, "Is NULL\n"); } } CImplicationList::CImplicationList(long lFlags) : m_lNextIndex(1), m_lRequiredDepth(1), m_pParent(NULL), m_lFlags(lFlags) { CPropertyName Empty; CRecord* pRecord = new CRecord(Empty, 0); if(pRecord == NULL) throw CX_MemoryException(); if(m_apRecords.Add(pRecord) < 0) throw CX_MemoryException(); } CImplicationList::CImplicationList(CImplicationList& Other, bool bLink) : m_lNextIndex(Other.m_lNextIndex), m_lRequiredDepth(Other.m_lRequiredDepth), m_pParent(NULL), m_lFlags(Other.m_lFlags) { if(bLink) m_pParent = &Other; for(int i = 0; i < Other.m_apRecords.GetSize(); i++) { CRecord* pOtherRecord = Other.m_apRecords[i]; CRecord* pNewRecord = new CRecord(*pOtherRecord); if(pNewRecord == NULL) throw CX_MemoryException(); if(m_apRecords.Add(pNewRecord) < 0) throw CX_MemoryException(); } } CImplicationList::~CImplicationList() { m_lNextIndex = 0; } void CImplicationList::FindBestComputedContainer(CPropertyName* pPropName, long* plRecord, long* plMatched) { // Look for the largest match // ========================== long lMax = -1; long lMaxRecord = -1; for(long lRecord = 0; lRecord < m_apRecords.GetSize(); lRecord++) { CRecord* pRecord = m_apRecords[lRecord]; if ( pRecord->m_lObjIndex == -1 ) { // // we only consider computed records // continue; } for(int i = 0; i < pPropName->GetNumElements() && i < pRecord->m_PropName.GetNumElements(); i++) { if(wbem_wcsicmp(pPropName->GetStringAt(i), pRecord->m_PropName.GetStringAt(i))) break; } if(i > lMax) { lMax = i; lMaxRecord = lRecord; } } if(plRecord) *plRecord = lMaxRecord; if(plMatched) *plMatched = lMax; } HRESULT CImplicationList::FindRecordForProp(CPropertyName* pPropName, long lNumElements, long* plRecord) { if(lNumElements == -1) { if(pPropName) lNumElements = pPropName->GetNumElements(); else lNumElements = 0; } // // Look for the exact match // for(long lRecord = 0; lRecord < m_apRecords.GetSize(); lRecord++) { CRecord* pRecord = m_apRecords[lRecord]; if(pRecord->m_PropName.GetNumElements() != lNumElements) continue; for(int i = 0; i < lNumElements; i++) { if(wbem_wcsicmp(pPropName->GetStringAt(i), pRecord->m_PropName.GetStringAt(i))) break; } if(i == lNumElements) { break; } } if(lRecord < m_apRecords.GetSize()) { // Found it! *plRecord = lRecord; return S_OK; } else return WBEM_E_NOT_FOUND; } HRESULT CImplicationList::FindBestComputedContainer(CPropertyName* pPropName, long* plFirstUnknownProp, long* plObjIndex, RELEASE_ME _IWmiObject** ppContainerClass) { if (!pPropName) return WBEM_E_FAILED; long lMax, lMaxRecord; FindBestComputedContainer(pPropName, &lMaxRecord, &lMax); if(lMaxRecord < 0) return WBEM_E_FAILED; if(plFirstUnknownProp) *plFirstUnknownProp = lMax; CRecord* pRecord = m_apRecords[lMaxRecord]; if(plObjIndex) *plObjIndex = pRecord->m_lObjIndex; if(ppContainerClass) { *ppContainerClass = pRecord->m_pClass; if(pRecord->m_pClass) pRecord->m_pClass->AddRef(); } return WBEM_S_NO_ERROR; } HRESULT CImplicationList::FindClassForProp(CPropertyName* pPropName, long lNumElements, RELEASE_ME _IWmiObject** ppClass) { // don't have a property name? djinn one up for use... CPropertyName* pPropNameLocal; CPropertyName blank; if (pPropName) pPropNameLocal = pPropName; else pPropNameLocal = ␣ long lRecord; CRecord* pRecord; if(SUCCEEDED(FindRecordForProp(pPropNameLocal, -1, &lRecord))) { // // A record is there --- return its class // *ppClass = m_apRecords[lRecord]->m_pClass; if(*ppClass) { (*ppClass)->AddRef(); return WBEM_S_NO_ERROR; } else return WBEM_E_NOT_FOUND; } else return WBEM_E_NOT_FOUND; } HRESULT CImplicationList::FindOrCreateRecordForProp(CPropertyName* pPropName, CImplicationList::CRecord** ppRecord) { // don't have a property name? djinn one up for use... CPropertyName* pPropNameLocal; CPropertyName blank; if (pPropName) pPropNameLocal = pPropName; else pPropNameLocal = ␣ long lRecord; CRecord* pRecord; if(SUCCEEDED(FindRecordForProp(pPropNameLocal, -1, &lRecord))) { // // A record is there --- improve it // pRecord = m_apRecords[lRecord]; } else { try { pRecord = new CRecord(*pPropNameLocal, -1); if(pRecord == NULL) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } if(m_apRecords.Add(pRecord) < 0) return WBEM_E_OUT_OF_MEMORY; } *ppRecord = pRecord; return WBEM_S_NO_ERROR; } HRESULT CImplicationList::ImproveKnown(CPropertyName* pPropName, _IWmiObject* pClass) { CRecord* pRecord; HRESULT hres = FindOrCreateRecordForProp(pPropName, &pRecord); if(FAILED(hres)) return hres; return pRecord->ImproveKnown(pClass); } HRESULT CImplicationList::ImproveKnownNot(CPropertyName* pPropName, LPCWSTR wszClassName) { CRecord* pRecord; HRESULT hres = FindOrCreateRecordForProp(pPropName, &pRecord); if(FAILED(hres)) return hres; return pRecord->ImproveKnownNot(wszClassName); } HRESULT CImplicationList::ImproveKnownNull(CPropertyName* pPropName) { CRecord* pRecord; HRESULT hres = FindOrCreateRecordForProp(pPropName, &pRecord); if(FAILED(hres)) return hres; return pRecord->ImproveKnownNull(); } HRESULT CImplicationList::AddComputation(CPropertyName& PropName, _IWmiObject* pClass, long* plObjIndex) { CRecord* pRecord; HRESULT hres = FindOrCreateRecordForProp(&PropName, &pRecord); if(FAILED(hres)) return hres; if(pClass) { hres = pRecord->ImproveKnown(pClass); if(FAILED(hres)) return hres; } pRecord->m_lObjIndex = m_lNextIndex; *plObjIndex = m_lNextIndex++; RequireDepth(m_lNextIndex); return WBEM_S_NO_ERROR; } long CImplicationList::GetRequiredDepth() { return m_lRequiredDepth; } void CImplicationList::RequireDepth(long lDepth) { if(lDepth > m_lRequiredDepth) { m_lRequiredDepth = lDepth; if(m_pParent) m_pParent->RequireDepth(lDepth); } } HRESULT CImplicationList::MergeIn(CImplicationList* pList) { // // Add everything we learn from the second list into our own // HRESULT hres; for(int i = 0; i < pList->m_apRecords.GetSize(); i++) { CRecord* pRecord = pList->m_apRecords[i]; hres = MergeIn(pRecord); if(FAILED(hres)) return hres; } return S_OK; } HRESULT CImplicationList::MergeIn(CImplicationList::CRecord* pRecord) { HRESULT hres; // // Add everything we learn from the record into our own list // if(pRecord->m_pClass) { hres = ImproveKnown(&pRecord->m_PropName, pRecord->m_pClass); if(FAILED(hres)) return hres; } for(int i = 0; i < pRecord->m_awsNotClasses.Size(); i++) { hres = ImproveKnownNot(&pRecord->m_PropName, pRecord->m_awsNotClasses[i]); if(FAILED(hres)) return hres; } if(pRecord->m_nNull == EVAL_VALUE_TRUE) { hres = ImproveKnownNull(&pRecord->m_PropName); if(FAILED(hres)) return hres; } return WBEM_S_NO_ERROR; } void CImplicationList::Dump(FILE* f, int nOffset) { for(int i = 0; i < m_apRecords.GetSize(); i++) m_apRecords[i]->Dump(f, nOffset); } //****************************************************************************** //****************************************************************************** // EVAL NODE //****************************************************************************** //****************************************************************************** CEvalNode::CEvalNode() { #ifdef CHECK_TREES g_treeChecker.AddNode(this); #endif } CEvalNode::CEvalNode(const CEvalNode& other) { #ifdef CHECK_TREES g_treeChecker.AddNode(this); #endif } CEvalNode::~CEvalNode() { #ifdef CHECK_TREES g_treeChecker.RemoveNode(this); #endif } void CEvalNode::PrintOffset(FILE* f, int nOffset) { for(int i = 0; i < nOffset; i++) { fprintf(f, " "); } } void CEvalNode::DumpNode(FILE* f, int nOffset, CEvalNode* pNode) { if(pNode == NULL) { PrintOffset(f, nOffset); fprintf(f, "FALSE\n"); } else pNode->Dump(f, nOffset); } CEvalNode* CEvalNode::CloneNode(const CEvalNode* pNode) { if(pNode == NULL) return NULL; CEvalNode* pNew = pNode->Clone(); if ( pNew == NULL ) throw CX_MemoryException(); return pNew; } bool CEvalNode::IsNoop(CEvalNode* pNode, int nOp) { if(pNode) return pNode->IsNoop(nOp); else return CValueNode::IsNoop(NULL, nOp); } #ifdef CHECK_TREES void CEvalNode::CheckNode(CTreeChecker *pCheck) { pCheck->CheckoffNode(this); } #endif //****************************************************************************** //****************************************************************************** // SORTED ARRAY //****************************************************************************** //****************************************************************************** // construct from a 'normal' array CSortedArray::CSortedArray(unsigned nElements, QueryID* pArray) : CFlexArray(nElements) { memcpy(GetArrayPtr(), pArray, nElements * sizeof(void*)); SetSize(nElements); } int CSortedArray::CopyDataFrom(const QueryID* pArray, unsigned nElements) { EnsureExtent(nElements); SetSize(nElements); memcpy(GetArrayPtr(), pArray, nElements * sizeof(void*)); return nElements; } unsigned CSortedArray::Find(QueryID n) { unsigned ret = InvalidID; // bailout if empty if(Size() == 0) return InvalidID; unsigned lBound = 0; unsigned uBound = Size() -1; // bailout checks - if it's on the boundary, don't search // if it's outside the boundaries, we ain't got it if (n == GetAt(uBound)) ret = uBound; else if (n == GetAt(lBound)) ret = lBound; else if ((n > GetAt(lBound)) && (n < GetAt(uBound))) { // binary search // warning: break in middle of loop do { unsigned testBound = (lBound + uBound) / 2; if (n < GetAt(testBound)) uBound = testBound; else if (n > GetAt(testBound)) lBound = testBound; else { ret = testBound; break; } } while (lBound < uBound -1); } return ret; } // inserts n in proper position // no dups allowed void CSortedArray::Insert(QueryID n) { // looks a lot like 'find' unsigned lBound = 0; unsigned uBound = Size() == 0 ? 0 : Size() -1; unsigned testBound = InvalidID; // check boundaries, empty array, out of bounds conditions... if ((Size() == 0) || (n < GetAt(lBound))) CFlexArray::InsertAt(0, (void *)n); else if (n > GetAt(uBound)) Add(n); else if ((n != GetAt(uBound)) && (n != GetAt(lBound))) { // binary search // warning: break in middle of loop do { testBound = (lBound + uBound) / 2; if (n < GetAt(testBound)) uBound = testBound; else if (n > GetAt(testBound)) lBound = testBound; else break; } while (lBound < uBound -1); // at this point, three cases: // 1) we found the item at testBound // 2) we didn't find it, uBound = lBound +1 // 3) we didn't find it, uBound == lBound if (n != GetAt(testBound)) { if (n < GetAt(lBound)) InsertAt(lBound, (void *)n); else InsertAt(uBound, (void *)n); } } } // removes element with value of n // NOT the nth element bool CSortedArray::Remove(QueryID n) { unsigned index; index = Find(n); if (index != InvalidID) { CFlexArray::RemoveAt(index); return true; } else return false; } //returns zero if arrays are equivalent // same number of USED elements w/ same values // LEVN: changed to return < 0 if less, > 0 if more int CSortedArray::Compare(CSortedArray& otherArray) { int nCompare = Size() - otherArray.Size(); if(nCompare) return nCompare; nCompare = memcmp(GetArrayPtr(), otherArray.GetArrayPtr(), Size() * sizeof(void*)); return nCompare; } // changes all QueryID's to begin at newBase // e.g. if the array is {0,1,5} // Rebase(6) will change to {6,7,11} void CSortedArray::Rebase(QueryID newBase) { for (int i = 0; i < Size(); i++) SetAt(i, (void *)(GetAt(i) + newBase)); } // adds the values from the other array into this one int CSortedArray::AddDataFrom(const CSortedArray& otherArray) { // Check for emptiness if(otherArray.Size() == 0) return no_error; // Ensure there is enough room in our array for the union // ====================================================== if(EnsureExtent(m_nSize + otherArray.m_nSize)) return out_of_memory; // Start merging from the end // ========================== int nThisSourceIndex = m_nSize - 1; int nThisDestIndex = m_nSize + otherArray.m_nSize - 1; int nOtherIndex = otherArray.m_nSize - 1; while(nThisSourceIndex >= 0 && nOtherIndex >= 0) { int nCompare = (QueryID)m_pArray[nThisSourceIndex] - (QueryID)otherArray[nOtherIndex]; if(nCompare < 0) { m_pArray[nThisDestIndex--] = otherArray[nOtherIndex--]; } else if(nCompare > 0) { m_pArray[nThisDestIndex--] = m_pArray[nThisSourceIndex--]; } else { m_pArray[nThisDestIndex--] = otherArray[nOtherIndex--]; nThisSourceIndex--; } } // Add remainders // ============== while(nThisSourceIndex >= 0) m_pArray[nThisDestIndex--] = m_pArray[nThisSourceIndex--]; while(nOtherIndex >= 0) m_pArray[nThisDestIndex--] = otherArray[nOtherIndex--]; // Move the array forward if needed // ================================ if(nThisDestIndex >= 0) { for(int i = nThisDestIndex+1; i < m_nSize + otherArray.m_nSize; i++) { m_pArray[i-nThisDestIndex-1] = m_pArray[i]; } } m_nSize = m_nSize + otherArray.m_nSize - (nThisDestIndex+1); return no_error; } // adds the values from the other array into this one int CSortedArray::AddDataFrom(const QueryID* pOtherArray, unsigned nValues) { // Check for emptiness if(nValues == 0) return no_error; // Ensure there is enough room in our array for the union // ====================================================== if(EnsureExtent(m_nSize + nValues)) return out_of_memory; // Start merging from the end // ========================== int nThisSourceIndex = m_nSize - 1; int nThisDestIndex = m_nSize + nValues - 1; int nOtherIndex = nValues - 1; while(nThisSourceIndex >= 0 && nOtherIndex >= 0) { int nCompare = (QueryID)m_pArray[nThisSourceIndex] - (QueryID)pOtherArray[nOtherIndex]; if(nCompare < 0) { m_pArray[nThisDestIndex--] = (void*)pOtherArray[nOtherIndex--]; } else if(nCompare > 0) { m_pArray[nThisDestIndex--] = m_pArray[nThisSourceIndex--]; } else { m_pArray[nThisDestIndex--] = (void*)pOtherArray[nOtherIndex--]; nThisSourceIndex--; } } // Add remainders // ============== while(nThisSourceIndex >= 0) m_pArray[nThisDestIndex--] = m_pArray[nThisSourceIndex--]; while(nOtherIndex >= 0) m_pArray[nThisDestIndex--] = (void*)pOtherArray[nOtherIndex--]; // Move the array forward if needed // ================================ if(nThisDestIndex >= 0) { for(int i = nThisDestIndex+1; i < m_nSize + nValues; i++) { m_pArray[i-nThisDestIndex-1] = m_pArray[i]; } } m_nSize = m_nSize + nValues - (nThisDestIndex+1); return no_error; } // copies this array to destination // only copies size number of elements // returns number of elements copied unsigned CSortedArray::CopyTo(QueryID* pDest, unsigned size) { unsigned mySize = Size(); unsigned nElementsToCopy = min(size, mySize); if (nElementsToCopy) memcpy(pDest, GetArrayPtr(), nElementsToCopy * sizeof(void*)); return nElementsToCopy; } //****************************************************************************** //****************************************************************************** // VALUE NODE //****************************************************************************** //****************************************************************************** CValueNode::CValueNode(int nNumValues) { } CValueNode::~CValueNode() { // this page intentionally left blank } /* virtual */ int CValueNode::GetType() { return EVAL_NODE_TYPE_VALUE; } // changes all QueryID's in all arrays to begin at newBase // e.g. if the array is {0,1,5} // Rebase(6) will change to {6,7,11} void CValueNode::Rebase(QueryID newBase) { for (int i = 0; i < m_nValues; i++) m_trueIDs[i] += newBase; } // returns array index of n // or InvalidID if not found unsigned CValueNode::FindQueryID(QueryID n) { unsigned ret = InvalidID; if ( m_nValues == 0 ) { return ret; } unsigned lBound = 0; unsigned uBound = m_nValues - 1; // bailout checks - if it's on the boundary, don't search // if it's outside the boundaries, we ain't got it if (n == m_trueIDs[uBound]) ret = uBound; else if (n == m_trueIDs[lBound]) ret = lBound; else if ((n > m_trueIDs[lBound]) && (n < m_trueIDs[uBound])) { // binary search // warning: break in middle of loop do { unsigned testBound = (lBound + uBound) / 2; if (n < m_trueIDs[testBound]) uBound = testBound; else if (n > m_trueIDs[testBound]) lBound = testBound; else { ret = testBound; break; } } while (lBound < uBound -1); } return ret; } bool CValueNode::RemoveQueryID(QueryID nQuery) { unsigned n; if ((n = FindQueryID(nQuery)) != InvalidID) { if ((m_nValues > 1) && (n != m_nValues -1)) memcpy(&(m_trueIDs[n]), &(m_trueIDs[n+1]), (m_nValues -n -1) * sizeof(QueryID)); if (m_nValues > 0) m_trueIDs[--m_nValues] = InvalidID; } return (n != InvalidID); } // pointers point to arrays that are at the corresponding sizes // caller is responsible for ensuring that the output array is large enough // return value #elements inserted into new array; unsigned CValueNode::ORarrays(QueryID* pArray1, unsigned size1, QueryID* pArray2, unsigned size2, QueryID* pOutput) { unsigned nElements = 0; // // really shouldn't happen - one side has should have come from this // _DBG_ASSERT( !(pArray1 == NULL && pArray2 == NULL) ); if (pArray2 == NULL) { nElements = size1; memcpy(pOutput, pArray1, sizeof(QueryID) * size1); } else if (pArray1 == NULL) { nElements = size2; memcpy(pOutput, pArray2, sizeof(QueryID) * size2); } else { QueryID* pQID1 = pArray1; QueryID* pQID2 = pArray2; QueryID* pOut = pOutput; // note that the 'ends' are really one past the end, careful now... QueryID* pEnd1 = pQID1 + size1; QueryID* pEnd2 = pQID2 + size2; // walk through both arrays, always adding smallest value to new array while ((pQID1 < pEnd1) && (pQID2 < pEnd2)) { if (*pQID1 == *pQID2) { // found match, add to array & bump up both cursors *pOut++ = *pQID1++; pQID2++; nElements++; } else if (*pQID2 < *pQID1) { // add it *pOut++ = *pQID2++; nElements++; } else { // other side must be smaller, add IT. *pOut++ = *pQID1++; nElements++; } } // run out whichever array we didn't finish // only one should ever hit while (pQID1 < pEnd1) { *pOut++ = *pQID1++; nElements++; } while (pQID2 < pEnd2) { *pOut++ = *pQID2++; nElements++; } } return nElements; } unsigned CValueNode::ANDarrays(QueryID* pArray1, unsigned size1, QueryID* pArray2, unsigned size2, QueryID* pOutput) { unsigned nElements = 0; if ((pArray1 != NULL) && (pArray2 != NULL)) { QueryID* pQID1 = pArray1; QueryID* pQID2 = pArray2; QueryID* pOut = pOutput; // note that the 'ends' are really one past the end, careful now... QueryID* pEnd1 = pQID1 + size1; QueryID* pEnd2 = pQID2 + size2; // walk through both arrays, adding any values that appear in both while ((pQID1 < pEnd1) && (pQID2 < pEnd2)) { if (*pQID1 == *pQID2) { // found match, add to array & bump up both cursors *pOut++ = *pQID1++; pQID2++; nElements++; } else if (*pQID2 < *pQID1) pQID2++; else pQID1++; } } return nElements; } unsigned CValueNode::CombineArrays(QueryID* pArray1, unsigned size1, QueryID* pArray2, unsigned size2, QueryID* pOutput) { unsigned nElements = 0; // // really shouldn't happen - one side has should have come from this // _DBG_ASSERT( !(pArray1 == NULL && pArray2 == NULL) ); if (pArray2 == NULL) { nElements = size1; memcpy(pOutput, pArray1, sizeof(QueryID) * size1); } else if (pArray1 == NULL) { nElements = size2; memcpy(pOutput, pArray2, sizeof(QueryID) * size2); } else { QueryID* pQID1 = pArray1; QueryID* pQID2 = pArray2; QueryID* pOut = pOutput; // note that the 'ends' are really one past the end, careful now... QueryID* pEnd1 = pQID1 + size1; QueryID* pEnd2 = pQID2 + size2; while ((pQID1 < pEnd1) && (pQID2 < pEnd2)) { if (*pQID1 == *pQID2) { // found match, add to array & bump up both cursors *pOut++ = *pQID1++; pQID2++; nElements++; } else if (*pQID2 < *pQID1) { // add it *pOut++ = *pQID2++; nElements++; } else { // other side must be smaller, add IT. *pOut++ = *pQID1++; nElements++; } } // run out whichever array we didn't finish // only one should ever hit while (pQID1 < pEnd1) { *pOut++ = *pQID1++; nElements++; } while (pQID2 < pEnd2) { *pOut++ = *pQID2++; nElements++; } } return nElements; } // size of new arrays is predicated on the assumption that // there will usually be more TRUE nodes than INVALID ones HRESULT CValueNode::CombineWith(CEvalNode* pRawArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& Implications, bool bDeleteThis, bool bDeleteArg2, CEvalNode** ppRes) { HRESULT hRes = WBEM_S_NO_ERROR; CValueNode* pArg2 = (CValueNode*)pRawArg2; // Check for immediate solutions // ============================= if(nOp != EVAL_OP_AND) { if(IsAllFalse(pArg2) && bDeleteThis) { // Just return this! // ================= *ppRes = this; if(bDeleteArg2) delete pArg2; return WBEM_S_NO_ERROR; } else if(IsAllFalse(this) && bDeleteArg2) { // Just return arg2! // ================= *ppRes = pRawArg2; if(bDeleteThis) delete this; return WBEM_S_NO_ERROR; } } // if we got enough room in a stack array, we'll use that, // elsewise we'll allocate one from the heap const unsigned NewArraySize = 128; QueryID newArray[NewArraySize]; QueryID* pNewArray = newArray; CDeleteMe deleteMe; CValueNode* pNew; unsigned nElements = 0; unsigned arg2Values; QueryID* arg2Array; if (pArg2) { arg2Values = pArg2->m_nValues; arg2Array = pArg2->m_trueIDs; } else { arg2Values = 0; arg2Array = NULL; } if (nOp == EVAL_OP_AND) { if (max(m_nValues, arg2Values) > NewArraySize) { pNewArray = new QueryID[max(m_nValues, arg2Values)]; if(pNewArray == NULL) return WBEM_E_OUT_OF_MEMORY; deleteMe = pNewArray; } nElements = ANDarrays(m_trueIDs, m_nValues, arg2Array, arg2Values, pNewArray); } // HMH: it sure looks to me like OR and COMBINE are the same for this case // I'm too afraid to risk changing it, tho else if (nOp == EVAL_OP_OR) { if ((m_nValues + arg2Values) > NewArraySize) { pNewArray = new QueryID[m_nValues+arg2Values]; if(pNewArray == NULL) return WBEM_E_OUT_OF_MEMORY; deleteMe = pNewArray; } nElements = ORarrays(m_trueIDs, m_nValues, arg2Array, arg2Values, pNewArray); } else if ((nOp == EVAL_OP_COMBINE) || (nOp == EVAL_OP_INVERSE_COMBINE)) { if ((m_nValues + arg2Values) > NewArraySize) { pNewArray = new QueryID[m_nValues + arg2Values]; if(pNewArray == NULL) return WBEM_E_OUT_OF_MEMORY; deleteMe = pNewArray; } nElements = CombineArrays(m_trueIDs, m_nValues, arg2Array, arg2Values, pNewArray); } // check to see if we can reuse a node, note this could result in the array 'shrinking' if (nElements == 0) *ppRes = NULL; else if (bDeleteThis && (nElements <= m_nValues)) { *ppRes = this; memcpy(m_trueIDs, pNewArray, nElements * sizeof(QueryID)); m_nValues = nElements; bDeleteThis = false; } else if (bDeleteArg2 && pArg2 && (nElements <= pArg2->m_nValues)) { *ppRes = pArg2; memcpy(pArg2->m_trueIDs, pNewArray, nElements * sizeof(QueryID)); pArg2->m_nValues = nElements; bDeleteArg2 = false; } else if (pNew = CreateNode(nElements)) { // can't reuse one, start a new one *ppRes = pNew; memcpy(pNew->m_trueIDs, pNewArray, nElements * sizeof(QueryID)); } else hRes = WBEM_E_OUT_OF_MEMORY; // Delete what needed deletion // =========================== // deleteMe will take care of the array pointer if allocated if(bDeleteThis) delete this; if(bDeleteArg2) delete pArg2; return WBEM_S_NO_ERROR; } //static bool CValueNode::IsNoop(CValueNode* pNode, int nOp) { if(nOp == EVAL_OP_OR) return IsAllFalse(pNode); else if(nOp == EVAL_OP_AND) return false; // BUGBUG: would be nice to have IsAllTrue, but can't else if(nOp == EVAL_OP_COMBINE || nOp == EVAL_OP_INVERSE_COMBINE) return IsAllFalse(pNode); else { //? return false; } } HRESULT CValueNode::TryShortCircuit(CEvalNode* pArg2, int nOp, bool bDeleteThis, bool bDeleteArg2, CEvalNode** ppRes) { if(IsAllFalse(this)) { if(nOp == EVAL_OP_AND) { // FALSE & X is FALSE if(bDeleteThis) *ppRes = this; else { *ppRes = Clone(); if(*ppRes == NULL) return WBEM_E_OUT_OF_MEMORY; } if(bDeleteArg2) delete pArg2; return WBEM_S_NO_ERROR; } else // OR and COMBINE are identical in this case { // FALSE | X is X // // Well, this is true, but the problem is with optimizations. // Some branches in X might not be valid under the implications in // this branch of the tree, and so need to be removed. For now, I // will simply turn off this short-circuiting path. It may turn out // that there are some critical performance gains to be had by // keeping it, in which case we would need to put this back and // make an efficient pass through it, checking branches. // return WBEM_S_FALSE; /* if(bDeleteArg2) *ppRes = pArg2; else if (pArg2) { *ppRes = pArg2->Clone(); if(*ppRes == NULL) return WBEM_E_OUT_OF_MEMORY; } else *ppRes = NULL; if(bDeleteThis) delete this; return WBEM_S_NO_ERROR; */ } } return WBEM_S_FALSE; } HRESULT CValueNode::Project(CContextMetaData* pMeta, CImplicationList& Implications, CProjectionFilter* pFilter, EProjectionType eType, bool bDeleteThis, CEvalNode** ppNewNode) { // // Projection of a constant is, again, a constant // if(bDeleteThis) { *ppNewNode = this; } else { *ppNewNode = Clone(); if(*ppNewNode == NULL) return WBEM_E_OUT_OF_MEMORY; } return S_OK; } CEvalNode* CValueNode::Clone() const { CValueNode* pNew; if (pNew = CreateNode(m_nValues)) memcpy(pNew->m_trueIDs, m_trueIDs, m_nValues * sizeof(QueryID)); return pNew; } int CValueNode::Compare(CEvalNode* pRawOther) { if (!pRawOther) return m_nValues; CValueNode* pOther = (CValueNode*)pRawOther; int nCompare = m_nValues - pOther->m_nValues; if ((nCompare == 0) && (m_nValues != 0)) nCompare = memcmp(m_trueIDs, pOther->m_trueIDs, m_nValues * sizeof(QueryID)); return nCompare; } CValueNode* CValueNode::GetStandardTrue() { CValueNode* pNew = CreateNode(1); if(pNew) pNew->m_trueIDs[0] = 0; return pNew; } CValueNode* CValueNode::GetStandardInvalid() { return new CInvalidNode; } /*static*/ CValueNode* CValueNode::CreateNode(size_t nNumValues) { return new(nNumValues) CValueNode; } /*static*/ CValueNode* CValueNode::CreateNode(CSortedArray& values) { CValueNode* pNode; pNode = new(values.Size()) CValueNode; if (pNode) values.CopyTo(pNode->m_trueIDs, values.Size()); return pNode; } void* CValueNode::operator new( size_t stAllocateBlock, unsigned nEntries) { void *pvTemp; if (pvTemp = ::new byte[ stAllocateBlock + ((nEntries ==0)? 0 : ((nEntries -1)* sizeof(QueryID)))] ) ((CValueNode*)pvTemp)->m_nValues = nEntries; return pvTemp; } // VC 5 only allows one delete operator per class #if _MSC_VER >= 1200 void CValueNode::operator delete( void *p, unsigned nEntries ) { ::delete[] (byte*)p; } #endif void CValueNode::Dump(FILE* f, int nOffset) { PrintOffset(f, nOffset); fprintf(f, "TRUE: "); for (int i = 0; i < m_nValues; i++) { fprintf(f, "%u", m_trueIDs[i]); if(i < m_nValues-1) fprintf(f, ", %u", m_trueIDs[i]); } fprintf(f, "\n"); } void CInvalidNode::Dump(FILE* f, int nOffset) { PrintOffset(f, nOffset); fprintf(f, "Invalid node (0x%p)\n", this); } //****************************************************************************** //****************************************************************************** // EMBEDDING INFO //****************************************************************************** //****************************************************************************** CEmbeddingInfo::CEmbeddingInfo() : m_lNumJumps(0), m_aJumps(NULL), m_lStartingObjIndex(0) { } CEmbeddingInfo::CEmbeddingInfo(const CEmbeddingInfo& Other) : m_lNumJumps(0), m_aJumps(NULL), m_lStartingObjIndex(0) { *this = Other; } void CEmbeddingInfo::operator=(const CEmbeddingInfo& Other) { m_EmbeddedObjPropName = Other.m_EmbeddedObjPropName; m_lNumJumps = Other.m_lNumJumps; m_lStartingObjIndex = Other.m_lStartingObjIndex; delete [] m_aJumps; if(m_lNumJumps > 0) { m_aJumps = new JumpInfo[m_lNumJumps]; if (m_aJumps == NULL) throw CX_MemoryException(); memcpy(m_aJumps, Other.m_aJumps, m_lNumJumps * sizeof(JumpInfo)); } else m_aJumps = NULL; } CEmbeddingInfo::~CEmbeddingInfo() { delete [] m_aJumps; } bool CEmbeddingInfo::IsEmpty() const { return (m_EmbeddedObjPropName.GetNumElements() == 0); } bool CEmbeddingInfo::SetPropertyNameButLast(const CPropertyName& Name) { try { m_EmbeddedObjPropName.Empty(); for(int i = 0; i < Name.GetNumElements() - 1; i++) { m_EmbeddedObjPropName.AddElement(Name.GetStringAt(i)); } return true; } catch (CX_MemoryException) { return false; } } int CEmbeddingInfo::ComparePrecedence(const CEmbeddingInfo* pOther) { if (pOther) { int nCompare = m_EmbeddedObjPropName.GetNumElements() - pOther->m_EmbeddedObjPropName.GetNumElements(); if(nCompare) return nCompare; for(int i = 0; i < m_EmbeddedObjPropName.GetNumElements(); i++) { nCompare = wbem_wcsicmp(m_EmbeddedObjPropName.GetStringAt(i), pOther->m_EmbeddedObjPropName.GetStringAt(i)); if(nCompare) return nCompare; } } return 0; } BOOL CEmbeddingInfo::operator==(const CEmbeddingInfo& Other) { if(m_lStartingObjIndex != Other.m_lStartingObjIndex) return FALSE; if(m_lNumJumps != Other.m_lNumJumps) return FALSE; if(m_aJumps == NULL) { if(Other.m_aJumps == NULL) return TRUE; else return FALSE; } else { if(Other.m_aJumps == NULL) return FALSE; else return (memcmp(m_aJumps, Other.m_aJumps, m_lNumJumps * sizeof(JumpInfo)) == 0); } } HRESULT CEmbeddingInfo::Compile( CContextMetaData* pNamespace, CImplicationList& Implications, _IWmiObject** ppResultClass ) { HRESULT hres; long lFirstUnknownProp; _IWmiObject* pContainerClass; hres = Implications.FindBestComputedContainer( &m_EmbeddedObjPropName, &lFirstUnknownProp, &m_lStartingObjIndex, &pContainerClass ); if(FAILED(hres)) return hres; int nNumEmbeddedJumps = m_EmbeddedObjPropName.GetNumElements(); if(lFirstUnknownProp < nNumEmbeddedJumps) { // Not everything is already loaded // ================================ delete [] m_aJumps; m_lNumJumps = nNumEmbeddedJumps - lFirstUnknownProp; m_aJumps = new JumpInfo[m_lNumJumps]; if (!m_aJumps) return WBEM_E_OUT_OF_MEMORY; JumpInfo* pji = NULL; for(int i = lFirstUnknownProp; i < nNumEmbeddedJumps; i++) { if(pContainerClass == NULL) return WBEMESS_E_REGISTRATION_TOO_BROAD; // Get the handle of this property // =============================== CIMTYPE ct; pji = m_aJumps + i - lFirstUnknownProp; pji->lTarget = -1; hres = pContainerClass->GetPropertyHandleEx( (LPWSTR)m_EmbeddedObjPropName.GetStringAt(i), 0L, &ct, &pji->lJump ); if(FAILED(hres) || ct != CIM_OBJECT) { // Invalid. Return pContainerClass->Release(); return WBEM_E_INVALID_PROPERTY; } // // Check if the implications know anything about the class name // for this property // _IWmiObject* pNewContainerClass = NULL; hres = Implications.FindClassForProp(&m_EmbeddedObjPropName, i+1, &pNewContainerClass); if(FAILED(hres)) { // // Nothing implied --- have to go with the CIMTYPE qualifier // // // Get CIMTYPE qualifier // CVar vCimtype; DWORD dwSize; hres = pContainerClass->GetPropQual( (LPWSTR)m_EmbeddedObjPropName.GetStringAt(i), L"CIMTYPE", 0, 0, NULL, NULL, &dwSize, NULL); if(hres != WBEM_E_BUFFER_TOO_SMALL) return WBEM_E_INVALID_PROPERTY; LPWSTR wszCimType = (LPWSTR)new BYTE[dwSize]; if(wszCimType == NULL) return WBEM_E_OUT_OF_MEMORY; CVectorDeleteMe vdm((BYTE*)wszCimType); CIMTYPE ctQualType; hres = pContainerClass->GetPropQual( (LPWSTR)m_EmbeddedObjPropName.GetStringAt(i), L"CIMTYPE", 0, dwSize, &ctQualType, NULL, &dwSize, wszCimType); if(FAILED(hres) || ctQualType != CIM_STRING) return WBEM_E_INVALID_PROPERTY; // // Figure out what it references, if available // WCHAR* pwc = wcschr(wszCimType, L':'); if(pwc) { // Information about the class is available // ======================================== hres = pNamespace->GetClass(pwc+1, &pNewContainerClass); if(FAILED(hres)) { return WBEM_E_INVALID_CIM_TYPE; } } } pContainerClass->Release(); pContainerClass = pNewContainerClass; } // Get a location for the result and store in the last jump info elem // ================================================================== Implications.AddComputation( m_EmbeddedObjPropName, pContainerClass, &pji->lTarget ); } else { // Everything is covered // ===================== delete [] m_aJumps; m_aJumps = NULL; m_lNumJumps = 0; } if(ppResultClass) { *ppResultClass = pContainerClass; } else { if(pContainerClass) pContainerClass->Release(); } return WBEM_S_NO_ERROR; } HRESULT CEmbeddingInfo::GetContainerObject( CObjectInfo& ObjInfo, INTERNAL _IWmiObject** ppInst) { if( m_lNumJumps == 0 ) { *ppInst = ObjInfo.GetObjectAt(m_lStartingObjIndex); return WBEM_S_NO_ERROR; } _IWmiObject* pCurrent = ObjInfo.GetObjectAt(m_lStartingObjIndex); pCurrent->AddRef(); for(int i = 0; i < m_lNumJumps; i++) { _IWmiObject* pNew = NULL; HRESULT hres = pCurrent->GetPropAddrByHandle( m_aJumps[i].lJump, 0, NULL, ( void** )&pNew ); if( FAILED( hres ) ) { return hres; } pCurrent->Release(); pCurrent = pNew; if(pNew == NULL) { *ppInst = pCurrent; return WBEM_S_FALSE; } // // save the object if required. // if ( m_aJumps[i].lTarget != -1 ) { ObjInfo.SetObjectAt( m_aJumps[i].lTarget, pCurrent ); } } *ppInst = pCurrent; return WBEM_S_NO_ERROR; } bool CEmbeddingInfo::AreJumpsRelated(const CEmbeddingInfo* pInfo) { if ( !pInfo ) { return false; } // // look through the targets of the info and see if we depend on any. // for( int i=0; i < pInfo->m_lNumJumps; i++ ) { if ( pInfo->m_aJumps[i].lTarget == m_lStartingObjIndex ) { return true; } } return false; } bool CEmbeddingInfo::MixInJumps(const CEmbeddingInfo* pInfo) { // // Assumes AreJumpsRelated() has been called and returned TRUE. // m_lStartingObjIndex = pInfo->m_lStartingObjIndex; JumpInfo* aNewJumps = new JumpInfo[m_lNumJumps + pInfo->m_lNumJumps]; if(aNewJumps == NULL) return false; if( pInfo->m_lNumJumps > 0 ) { memcpy( aNewJumps, pInfo->m_aJumps, sizeof(JumpInfo)*(pInfo->m_lNumJumps) ); } if( m_lNumJumps > 0 ) { memcpy( aNewJumps + pInfo->m_lNumJumps, m_aJumps, sizeof(JumpInfo)*m_lNumJumps ); } m_lNumJumps += pInfo->m_lNumJumps; delete [] m_aJumps; m_aJumps = aNewJumps; return true; } void CEmbeddingInfo::Dump(FILE* f) { fprintf(f, "Name="); int i; for(i = 0; i < m_EmbeddedObjPropName.GetNumElements(); i++) { if(i != 0) fprintf(f, "."); fprintf(f, "%S", m_EmbeddedObjPropName.GetStringAt(i)); } fprintf(f, ", Alg=%d -> (", m_lStartingObjIndex); for(i = 0; i < m_lNumJumps; i++) { if(i != 0) fprintf(f, ", "); fprintf(f, "0x%x : %d", m_aJumps[i].lJump, m_aJumps[i].lTarget ); } fprintf(f, ")"); } //****************************************************************************** //****************************************************************************** // BRANCHING NODE //****************************************************************************** //****************************************************************************** CNodeWithImplications::CNodeWithImplications(const CNodeWithImplications& Other) : CEvalNode(Other), m_pExtraImplications(NULL) { if(Other.m_pExtraImplications) { m_pExtraImplications = new CImplicationList(*Other.m_pExtraImplications, false); // no link if(m_pExtraImplications == NULL) throw CX_MemoryException(); } } void CNodeWithImplications::Dump(FILE* f, int nOffset) { if(m_pExtraImplications) m_pExtraImplications->Dump(f, nOffset); } CBranchingNode::CBranchingNode() : CNodeWithImplications(), m_pNullBranch(NULL), m_pInfo(NULL) { m_pNullBranch = CValueNode::GetStandardFalse(); } CBranchingNode::CBranchingNode(const CBranchingNode& Other, BOOL bChildren) : CNodeWithImplications(Other), m_pInfo(NULL) { if (Other.m_pInfo) { m_pInfo = new CEmbeddingInfo(*(Other.m_pInfo)); if(m_pInfo == NULL) throw CX_MemoryException(); } int i; if(bChildren) { m_pNullBranch = (CBranchingNode*)CloneNode(Other.m_pNullBranch); if(m_pNullBranch == NULL && Other.m_pNullBranch != NULL) throw CX_MemoryException(); for(i = 0; i < Other.m_apBranches.GetSize(); i++) { CBranchingNode* pNewBranch = (CBranchingNode*)CloneNode(Other.m_apBranches[i]); if(pNewBranch == NULL && Other.m_apBranches[i] != NULL) throw CX_MemoryException(); if(m_apBranches.Add(pNewBranch) < 0) throw CX_MemoryException(); } } else { m_pNullBranch = CValueNode::GetStandardFalse(); } } /* virtual */ int CBranchingNode::GetType() { return EVAL_NODE_TYPE_BRANCH; } CBranchingNode::~CBranchingNode() { delete m_pNullBranch; delete m_pInfo; } bool CBranchingNode::MixInJumps( const CEmbeddingInfo* pJump ) { bool bRet; if ( !pJump ) { return true; } // // we want to find the first node in the tree that is related to the // ancestor embedding info. If this node is related, then we mix in the // jumps and return. If not, then we propagate the info down the tree. // if ( m_pInfo ) { if ( m_pInfo->AreJumpsRelated( pJump ) ) { return m_pInfo->MixInJumps( pJump ); } } for(int i = 0; i < m_apBranches.GetSize(); i++) { if ( CEvalNode::GetType(m_apBranches[i]) == EVAL_NODE_TYPE_BRANCH ) { if ( !((CBranchingNode*)m_apBranches[i])->MixInJumps( pJump ) ) { return false; } } } return true; } bool CBranchingNode::SetEmbeddedObjPropName(CPropertyName& Name) { try { if (!m_pInfo) { m_pInfo = new CEmbeddingInfo; if(m_pInfo == NULL) return false; } m_pInfo->SetEmbeddedObjPropName(Name); return true; } catch (CX_MemoryException) { return false; } } void CBranchingNode::SetNullBranch(CEvalNode* pBranch) { delete m_pNullBranch; m_pNullBranch = pBranch; } DELETE_ME CBranchIterator* CBranchingNode::GetBranchIterator() { return new CDefaultBranchIterator(this); } int CBranchingNode::ComparePrecedence(CBranchingNode* pArg1, CBranchingNode* pArg2) { int nCompare; nCompare = pArg1->GetSubType() - pArg2->GetSubType(); if(nCompare) return nCompare; // Compare embedding specs // ======================= if (pArg1->m_pInfo && pArg2->m_pInfo) { nCompare = pArg1->m_pInfo->ComparePrecedence(pArg2->m_pInfo); if(nCompare) return nCompare; } else if (pArg2->m_pInfo) return -1; else if (pArg1->m_pInfo) return 1; // Embedding are the same --- compare lower levels // =============================================== return pArg1->ComparePrecedence(pArg2); } DWORD CBranchingNode::ApplyPredicate(CLeafPredicate* pPred) { DWORD dwRes; for(int i = 0; i < m_apBranches.GetSize(); i++) { if(m_apBranches[i] == NULL) dwRes = (*pPred)(NULL); else dwRes = m_apBranches[i]->ApplyPredicate(pPred); if(dwRes & WBEM_DISPOSITION_FLAG_DELETE) { m_apBranches.SetAt(i, NULL); dwRes &= ~WBEM_DISPOSITION_FLAG_DELETE; } if(dwRes & WBEM_DISPOSITION_FLAG_INVALIDATE) { m_apBranches.SetAt(i, CValueNode::GetStandardInvalid()); dwRes &= ~WBEM_DISPOSITION_FLAG_INVALIDATE; } if(dwRes == WBEM_DISPOSITION_STOPLEVEL) return WBEM_DISPOSITION_NORMAL; if(dwRes == WBEM_DISPOSITION_STOPALL) return dwRes; } if(m_pNullBranch) dwRes = m_pNullBranch->ApplyPredicate(pPred); else dwRes = (*pPred)(NULL); if(dwRes & WBEM_DISPOSITION_FLAG_DELETE) { delete m_pNullBranch; m_pNullBranch = NULL; dwRes &= ~WBEM_DISPOSITION_FLAG_DELETE; } if(dwRes & WBEM_DISPOSITION_FLAG_INVALIDATE) { delete m_pNullBranch; m_pNullBranch = CValueNode::GetStandardInvalid(); dwRes &= ~WBEM_DISPOSITION_FLAG_INVALIDATE; } if(dwRes == WBEM_DISPOSITION_STOPALL) return dwRes; return WBEM_DISPOSITION_NORMAL; } HRESULT CBranchingNode::Project(CContextMetaData* pMeta, CImplicationList& Implications, CProjectionFilter* pFilter, EProjectionType eType, bool bDeleteThis, CEvalNode** ppNewNode) { HRESULT hres; try { // // Record what we learn by even getting here // CImplicationList TheseImplications(Implications); if(GetExtraImplications()) { hres = TheseImplications.MergeIn(GetExtraImplications()); if(FAILED(hres)) return hres; } // BUGBUG: we could be more efficient and not clone the node when // bDeleteThis is not specified. CBranchingNode* pNew; if(bDeleteThis) { pNew = this; } else { pNew = (CBranchingNode*)Clone(); if(pNew == NULL) return WBEM_E_OUT_OF_MEMORY; } // BUGBUG: it is not always necessary to project all children. Could be // more efficient, but can't find the perfect algorithm... // // Project all our children // CBranchIterator* pit = pNew->GetBranchIterator(); if(pit == NULL) return WBEM_E_OUT_OF_MEMORY; CDeleteMe dm1(pit); while(pit->IsValid()) { if(!CEvalNode::IsInvalid(pit->GetNode())) { CImplicationList BranchImplications(TheseImplications); pit->RecordBranch(pMeta, BranchImplications); CEvalNode* pNewBranch; hres = CEvalTree::Project(pMeta, BranchImplications, pit->GetNode(), pFilter, eType, true, &pNewBranch); if(FAILED(hres)) return hres; pit->SetNode(pNewBranch); // old one already deleted } pit->Advance(); } // // Determine if this is an "in" node for this filter // bool bIn = pFilter->IsInSet(this); if(bIn) { // // For nodes martching the filter that's it --- both necessary and // sufficient conditions are simply projections of what's below // *ppNewNode = pNew; } else { // // The node does not match the filter. Now is when the difference // between sufficient and necessary conditions applies // int nBranchOp; if(eType == e_Sufficient) { // // For a condition to be sufficient for the truth of the entire // node, it should appear in every single branch of the node. // Therefore, we must AND all the branches together. Except for // invalid ones --- they can't happen, so we can omit them from // the analysis // nBranchOp = EVAL_OP_AND; } else if(eType == e_Necessary) { // // For a condition to be necessary for this node, it has to // appear in at least one branch. Therefore, we must OR all // the branches together. Except for invalid ones. // nBranchOp = EVAL_OP_OR; } else return WBEM_E_INVALID_PARAMETER; // // Perform the needed operation on them all // CBranchIterator* pitInner = pNew->GetBranchIterator(); if(pitInner == NULL) return WBEM_E_OUT_OF_MEMORY; CDeleteMe dm2(pitInner); // // Merge all of the, in, one at a time, deleting them as we go. // It is safe to delete them always, since we cloned the entire node // in the beginning if deletion was not allowed. // CEvalNode* pCombination = NULL; bool bInited = false; while(pitInner->IsValid()) { if(!CEvalNode::IsInvalid(pitInner->GetNode())) { if(bInited) { hres = CEvalTree::Combine(pCombination, pitInner->GetNode(), nBranchOp, pMeta, Implications, true, true, &pCombination); if(FAILED(hres)) return hres; } else { pCombination = pitInner->GetNode(); bInited = true; } pitInner->SetNode(NULL); } pitInner->Advance(); } if(!bInited) { // // No valid nodes?? // pCombination = CValueNode::GetStandardInvalid(); } // // The combination is it. // *ppNewNode = pCombination; delete pNew; } } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } return S_OK; } int CBranchingNode::Compare(CEvalNode* pRawOther) { CBranchingNode* pOther = (CBranchingNode*)pRawOther; int nCompare; nCompare = GetSubType() - pOther->GetSubType(); if(nCompare) return nCompare; // First, compare embeddings // ========================= if(m_pInfo == NULL && pOther->m_pInfo != NULL) return -1; if(m_pInfo != NULL && pOther->m_pInfo == NULL) return 1; if(m_pInfo != NULL && pOther->m_pInfo != NULL) { nCompare = m_pInfo->ComparePrecedence(pOther->m_pInfo); if(nCompare) return nCompare; } if(m_apBranches.GetSize() != pOther->m_apBranches.GetSize()) return m_apBranches.GetSize() - pOther->m_apBranches.GetSize(); // Then, compare derived portions // ============================== nCompare = SubCompare(pOther); if(nCompare) return nCompare; // Finally, compare children // ========================= for(int i = 0; i < m_apBranches.GetSize(); i++) { nCompare = CEvalTree::Compare(m_apBranches[i], pOther->m_apBranches[i]); if(nCompare) return nCompare; } return 0; } HRESULT CBranchingNode::CombineWith(CEvalNode* pRawArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& Implications, bool bDeleteThis, bool bDeleteArg2, CEvalNode** ppRes) { CBranchingNode* pArg2 = (CBranchingNode*)pRawArg2; HRESULT hres; // Compare arguments for precedence // ================================ int nCompare = ComparePrecedence(this, pArg2); if(nCompare < 0) { // Put pArg1 first and continue // ============================ hres = CombineInOrderWith(pArg2, nOp, pNamespace, Implications, bDeleteThis, bDeleteArg2, ppRes); } else if(nCompare > 0) { // Put pArg2 first and continue (reverse delete indicators!!) // ========================================================== hres = pArg2->CombineInOrderWith(this, FlipEvalOp(nOp), pNamespace, Implications, bDeleteArg2, bDeleteThis, ppRes); } else { // They are about the same property. Combine lookup lists. // ======================================================= hres = CombineBranchesWith(pArg2, nOp, pNamespace, Implications, bDeleteThis, bDeleteArg2, ppRes); } return hres; } HRESULT CBranchingNode::CombineInOrderWith(CEvalNode* pArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& OrigImplications, bool bDeleteThis, bool bDeleteArg2, CEvalNode** ppRes) { HRESULT hres; CBranchingNode* pNew = NULL; if(bDeleteThis) { pNew = this; } else { // // I'd like to clone self here, but can't because there is no // such virtual method. Something to be improved, perhaps // pNew = (CBranchingNode*)Clone(); if(pNew == NULL) return WBEM_E_OUT_OF_MEMORY; } try { CImplicationList Implications(OrigImplications); hres = pNew->AdjustCompile(pNamespace, Implications); if(FAILED(hres)) { if ( !bDeleteThis ) delete pNew; return hres; } // // Maintain a counter telling us whether any invalid branches (that // cannot occur under these implications) were detected. If so, we // need to re-optimize this node // bool bInvalidBranchesDetected = false; for(int i = 0; i < m_apBranches.GetSize(); i++) { CEvalNode* pNewBranch = NULL; CImplicationList BranchImplications(Implications); hres = RecordBranch(pNamespace, BranchImplications, i); if(SUCCEEDED(hres)) { // Always delete the branch --- if bDeleteThis, we should, and // if not, we cloned it! hres = CEvalTree::Combine(pNew->m_apBranches[i], pArg2, nOp, pNamespace, BranchImplications, true, false, &pNewBranch); if(FAILED(hres)) { if ( !bDeleteThis ) delete pNew; return hres; } pNew->m_apBranches.Discard(i); } else { pNewBranch = CValueNode::GetStandardInvalid(); bInvalidBranchesDetected = true; } pNew->m_apBranches.SetAt(i, pNewBranch); } CEvalNode* pNewBranch = NULL; CImplicationList BranchImplications(Implications); hres = RecordBranch(pNamespace, BranchImplications, -1); if(SUCCEEDED(hres)) { // // Always delete the branch --- if bDeleteThis, we should, and // if not, we cloned it! // hres = CEvalTree::Combine(pNew->GetNullBranch(), pArg2, nOp, pNamespace, BranchImplications, true, false, &pNewBranch); if(FAILED(hres)) { if ( !bDeleteThis ) delete pNew; return hres; } pNew->m_pNullBranch = NULL; } else { pNewBranch = CValueNode::GetStandardInvalid(); bInvalidBranchesDetected = true; } pNew->SetNullBranch(pNewBranch); if(bDeleteArg2) delete pArg2; // // If invalid branches were cut, re-optimize // if(bInvalidBranchesDetected) { hres = pNew->Optimize(pNamespace, ppRes); if ( SUCCEEDED( hres ) && *ppRes != pNew) delete pNew; return hres; } else { *ppRes = pNew; return WBEM_S_NO_ERROR; } } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } } BOOL CBranchingNode::AreAllSame(CEvalNode** apNodes, int nSize, int* pnFoundIndex) { BOOL bFoundTwo = FALSE; *pnFoundIndex = -1; CEvalNode* pFound = NULL; for(int i = 0; i < nSize; i++) { // ignore invalid nodes --- they don't count if(CEvalNode::IsInvalid(apNodes[i])) continue; if(*pnFoundIndex == -1) { // // This is the first valid chils this node has. Record it --- it // might be the only one // *pnFoundIndex = i; pFound = apNodes[i]; } else if(CEvalTree::Compare(apNodes[i], pFound) != 0) { bFoundTwo = TRUE; break; } } return !bFoundTwo; } HRESULT CBranchingNode::StoreBranchImplications(CContextMetaData* pNamespace, int nBranchIndex, CEvalNode* pResult) { if(pResult) { CImplicationList* pBranchImplications = NULL; try { pBranchImplications = new CImplicationList; if(pBranchImplications == NULL) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } HRESULT hres = RecordBranch(pNamespace, *pBranchImplications, nBranchIndex); if(FAILED(hres)) { delete pBranchImplications; return hres; } if(pBranchImplications->IsEmpty()) { // Nothing useful to say! delete pBranchImplications; pBranchImplications = NULL; } pResult->SetExtraImplications(pBranchImplications); // acquires } return WBEM_S_NO_ERROR; } HRESULT CBranchingNode::Optimize(CContextMetaData* pNamespace, CEvalNode** ppNew) { int i; HRESULT hres; // Optimize all branches // ===================== for(i = 0; i < m_apBranches.GetSize(); i++) { if(m_apBranches[i]) { CEvalNode* pNew = NULL; m_apBranches[i]->Optimize(pNamespace, &pNew); if(pNew != m_apBranches[i]) { m_apBranches.SetAt(i, pNew); } } } if(CEvalNode::GetType(m_pNullBranch) == EVAL_NODE_TYPE_BRANCH) { CEvalNode* pNew; ((CBranchingNode*)m_pNullBranch)->Optimize(pNamespace, &pNew); if(pNew != m_pNullBranch) { SetNullBranch(pNew); } } // Self-optimize // ============= OptimizeSelf(); // Count the number of branches // ============================ int nFoundIndex = -1; BOOL bFoundTwo = !AreAllSame(m_apBranches.GetArrayPtr(), m_apBranches.GetSize(), &nFoundIndex); if(bFoundTwo) { *ppNew = this; return WBEM_S_NO_ERROR; } if(nFoundIndex == -1) { if(CEvalNode::IsInvalid(m_pNullBranch)) { // // Totally invalid, the whole lot // *ppNew = m_pNullBranch; m_pNullBranch = NULL; } else { // // There are no valid branches, except for the NullBranch. // We can replace ourselves with the NullBranch // *ppNew = m_pNullBranch; m_pNullBranch = NULL; // // Now, we need to copy // the branch implications into the "extras". This is because // the information about which branch was taken in this test is // critical to the compilation of the lower nodes --- it tells them // about the classes of some of our embedded objects. // hres = StoreBranchImplications(pNamespace, -1, *ppNew); if(FAILED(hres)) return hres; // // since this node is going away, mix in the embedding info it // has with the child node. // if ( CEvalNode::GetType(*ppNew) == EVAL_NODE_TYPE_BRANCH ) { CBranchingNode* pBranchNode = (CBranchingNode*)*ppNew; if(!pBranchNode->MixInJumps( m_pInfo )) return WBEM_E_OUT_OF_MEMORY; } } } else { // // There is one valid branch in the regular list. Two hopes: that the // NullBranch is invalid, or that it is the same as the only valid // branch in the regular list // if(CEvalNode::IsInvalid(m_pNullBranch) || CEvalTree::Compare(m_pNullBranch, m_apBranches[nFoundIndex]) == 0) { // // Hurray. We could replace ourselves with the remaining branch. // m_apBranches.SetAt(nFoundIndex, NULL, ppNew); // // Now, we need to copy // the branch implications into the "extras". This is because // the information about which branch was taken in this test is // critical to the compilation of the lower nodes --- it tells them // about the classes of some of our embedded objects. // hres = StoreBranchImplications(pNamespace, nFoundIndex, *ppNew); if(FAILED(hres)) return hres; // // since this node is going away, mix in the embedding info it // has with the child node. // if ( CEvalNode::GetType(*ppNew) == EVAL_NODE_TYPE_BRANCH ) { CBranchingNode* pBranchNode = (CBranchingNode*)*ppNew; if(!pBranchNode->MixInJumps( m_pInfo )) return WBEM_E_OUT_OF_MEMORY; } } else { *ppNew = this; } return WBEM_S_NO_ERROR; } return WBEM_S_NO_ERROR; } void CBranchingNode::Dump(FILE* f, int nOffset) { CNodeWithImplications::Dump(f, nOffset); if (m_pInfo) { PrintOffset(f, nOffset); fprintf(f, "Embedding: "); m_pInfo->Dump(f); fprintf(f, "\n"); } } //****************************************************************************** //****************************************************************************** // PROPERTY NODE //****************************************************************************** //****************************************************************************** bool CPropertyNode::SetPropertyInfo(LPCWSTR wszPropName, long lPropHandle) { m_lPropHandle = lPropHandle; try { m_wsPropName = wszPropName; } catch (CX_MemoryException) { return false; } return true; } int CPropertyNode::ComparePrecedence(CBranchingNode* pOther) { CPropertyNode* pOtherNode = (CPropertyNode*)pOther; return m_lPropHandle - pOtherNode->m_lPropHandle; } bool CPropertyNode::SetEmbeddingInfo(const CEmbeddingInfo* pInfo) { try { if ((pInfo == NULL) || (pInfo->IsEmpty())) { delete m_pInfo; m_pInfo = NULL; } else if (!m_pInfo) { m_pInfo = new CEmbeddingInfo(*pInfo); if(m_pInfo == NULL) return false; } else *m_pInfo = *pInfo; } catch (CX_MemoryException) { return false; } return true; } HRESULT CPropertyNode::SetNullTest(int nOperator) { if(nOperator == QL1_OPERATOR_EQUALS) { if(m_apBranches.Add(CValueNode::GetStandardFalse()) < 0) return WBEM_E_OUT_OF_MEMORY; CEvalNode* pTrue = CValueNode::GetStandardTrue(); if(pTrue == NULL) return WBEM_E_OUT_OF_MEMORY; SetNullBranch(pTrue); } else if(nOperator == QL1_OPERATOR_NOTEQUALS) { CEvalNode* pTrue = CValueNode::GetStandardTrue(); if(pTrue == NULL) return WBEM_E_OUT_OF_MEMORY; if(m_apBranches.Add(pTrue) < 0) { delete pTrue; return WBEM_E_OUT_OF_MEMORY; } SetNullBranch(CValueNode::GetStandardFalse()); } else return WBEM_E_INVALID_QUERY; return WBEM_S_NO_ERROR; } HRESULT CPropertyNode::SetOperator(int nOperator) { m_apBranches.RemoveAll(); #define GET_STD_TRUE CValueNode::GetStandardTrue() #define GET_STD_FALSE CValueNode::GetStandardFalse() #define ADD_STD_TRUE {CEvalNode* p = GET_STD_TRUE; \ if(p == NULL) return WBEM_E_OUT_OF_MEMORY; \ if(m_apBranches.Add(p) < 0) {delete p; return WBEM_E_OUT_OF_MEMORY;}} #define ADD_STD_FALSE {CEvalNode* p = GET_STD_FALSE; \ if(m_apBranches.Add(p) < 0) {delete p; return WBEM_E_OUT_OF_MEMORY;}} switch(nOperator) { case QL1_OPERATOR_EQUALS: ADD_STD_FALSE; ADD_STD_TRUE; ADD_STD_FALSE; break; case QL1_OPERATOR_NOTEQUALS: ADD_STD_TRUE; ADD_STD_FALSE; ADD_STD_TRUE; break; case QL1_OPERATOR_LESS: ADD_STD_TRUE; ADD_STD_FALSE; ADD_STD_FALSE; break; case QL1_OPERATOR_GREATER: ADD_STD_FALSE; ADD_STD_FALSE; ADD_STD_TRUE; break; case QL1_OPERATOR_LESSOREQUALS: ADD_STD_TRUE; ADD_STD_TRUE; ADD_STD_FALSE; break; case QL1_OPERATOR_GREATEROREQUALS: ADD_STD_FALSE; ADD_STD_TRUE; ADD_STD_TRUE; break; case QL1_OPERATOR_LIKE: ADD_STD_TRUE; ADD_STD_FALSE; break; case QL1_OPERATOR_UNLIKE: ADD_STD_FALSE; ADD_STD_TRUE; break; default: return WBEM_E_CRITICAL_ERROR; } return WBEM_S_NO_ERROR; } //****************************************************************************** //****************************************************************************** // STRING PROPERTY NODE //****************************************************************************** //****************************************************************************** CStringPropNode::CStringPropNode(const CStringPropNode& Other, BOOL bChildren) : CFullCompareNode(Other, bChildren) { } CStringPropNode::~CStringPropNode() { } HRESULT CStringPropNode::Evaluate(CObjectInfo& ObjInfo, INTERNAL CEvalNode** ppNext) { *ppNext = NULL; HRESULT hres; _IWmiObject* pObj; hres = GetContainerObject(ObjInfo, &pObj); if( S_OK != hres ) { return hres; } // Get the property from the object // ================================ CCompressedString* pcs = CoreGetPropertyString(pObj, m_lPropHandle); CInternalString is; if(pcs == NULL) { *ppNext = m_pNullBranch; return WBEM_S_NO_ERROR; } is.AcquireCompressedString(pcs); // Search for the value // ==================== TTestPointIterator it; bool bMatch = m_aTestPoints.Find(is, &it); if(bMatch) *ppNext = it->m_pAt; else if(it == m_aTestPoints.End()) *ppNext = m_pRightMost; else *ppNext = it->m_pLeftOf; is.Unbind(); return WBEM_S_NO_ERROR; } void CStringPropNode::Dump(FILE* f, int nOffset) { CBranchingNode::Dump(f, nOffset); PrintOffset(f, nOffset); fprintf(f, "LastPropName = (0x%x)\n", m_lPropHandle); for(TConstTestPointIterator it = m_aTestPoints.Begin(); it != m_aTestPoints.End(); it++) { PrintOffset(f, nOffset); if (it != m_aTestPoints.Begin()) { TConstTestPointIterator itPrev(it); itPrev--; fprintf(f, "%s < ", itPrev->m_Test.GetText()); } fprintf(f, "X < %s\n", it->m_Test.GetText()); DumpNode(f, nOffset +1, it->m_pLeftOf); PrintOffset(f, nOffset); fprintf(f, "X = %s\n", it->m_Test.GetText()); DumpNode(f, nOffset +1, it->m_pAt); } PrintOffset(f, nOffset); if (it != m_aTestPoints.Begin()) { TConstTestPointIterator itPrev(it); itPrev--; fprintf(f, "X > %s\n", itPrev->m_Test.GetText()); } else fprintf(f, "ANY\n"); DumpNode(f, nOffset+1, m_pRightMost); PrintOffset(f, nOffset); fprintf(f, "NULL->\n"); DumpNode(f, nOffset+1, m_pNullBranch); } /***************************************************************************** CLikeStringPropNode ******************************************************************************/ CLikeStringPropNode::CLikeStringPropNode( const CLikeStringPropNode& Other, BOOL bChildren ) : CPropertyNode( Other, bChildren ) { m_Like = Other.m_Like; } int CLikeStringPropNode::ComparePrecedence( CBranchingNode* pRawOther ) { int nCompare = CPropertyNode::ComparePrecedence( pRawOther ); if( nCompare ) { return nCompare; } CLikeStringPropNode* pOther = (CLikeStringPropNode*)pRawOther; return wbem_wcsicmp( m_Like.GetExpression(), pOther->m_Like.GetExpression() ); } int CLikeStringPropNode::SubCompare( CEvalNode* pRawOther ) { int nCompare; CLikeStringPropNode* pOther = (CLikeStringPropNode*)pRawOther; _DBG_ASSERT( m_apBranches.GetSize() == 2 ); _DBG_ASSERT( pOther->m_apBranches.GetSize() == 2 ); nCompare = CEvalTree::Compare( m_apBranches[0], pOther->m_apBranches[0] ); if ( nCompare ) { return nCompare; } nCompare = CEvalTree::Compare( m_apBranches[1], pOther->m_apBranches[1] ); if ( nCompare ) { return nCompare; } return CEvalTree::Compare( m_pNullBranch, pOther->m_pNullBranch ); } HRESULT CLikeStringPropNode::Evaluate( CObjectInfo& ObjInfo, CEvalNode** ppNext ) { *ppNext = NULL; HRESULT hr; // // get the string value. // _IWmiObject* pObj; hr = GetContainerObject( ObjInfo, &pObj ); if( S_OK != hr ) { return hr; } CCompressedString* pcs = CoreGetPropertyString( pObj, m_lPropHandle ); // // if null, then simply take null branch. // if( pcs == NULL ) { *ppNext = m_pNullBranch; return WBEM_S_NO_ERROR; } CInternalString is; is.AcquireCompressedString(pcs); WString ws = is; // // run through like filter. take branch accordingly. // if ( m_Like.Match( ws ) ) { *ppNext = m_apBranches[0]; } else { *ppNext = m_apBranches[1]; } is.Unbind(); return WBEM_S_NO_ERROR; } HRESULT CLikeStringPropNode::SetTest( VARIANT& v ) { if ( V_VT(&v) != VT_BSTR ) { return WBEM_E_TYPE_MISMATCH; } try { m_Like.SetExpression( V_BSTR(&v) ); } catch(CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } return WBEM_S_NO_ERROR; } HRESULT CLikeStringPropNode::CombineBranchesWith( CBranchingNode* pRawArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& Implications, bool bDeleteThis, bool bDeleteArg2, CEvalNode** ppRes ) { HRESULT hres; *ppRes = NULL; CLikeStringPropNode* pArg2 = (CLikeStringPropNode*)pRawArg2; if ( !bDeleteThis ) { CLikeStringPropNode* pClone = (CLikeStringPropNode*)Clone(); if ( pClone == NULL ) return WBEM_E_OUT_OF_MEMORY; return pClone->CombineBranchesWith( pRawArg2, nOp, pNamespace, Implications, true, // reuse clone! bDeleteArg2, ppRes ); } CEvalNode* pNew = NULL; // // merge the 'match' branches // hres = CEvalTree::Combine( m_apBranches[0], pArg2->m_apBranches[0], nOp, pNamespace, Implications, true, bDeleteArg2, &pNew ); if(FAILED(hres)) return hres; m_apBranches.Discard( 0 ); m_apBranches.SetAt( 0, pNew ); if( bDeleteArg2 ) { pArg2->m_apBranches.Discard( 0 ); } // // merge the 'nomatch' branches // hres = CEvalTree::Combine( m_apBranches[1], pArg2->m_apBranches[1], nOp, pNamespace, Implications, true, bDeleteArg2, &pNew ); if(FAILED(hres)) return hres; m_apBranches.Discard( 1 ); m_apBranches.SetAt( 1, pNew ); if( bDeleteArg2 ) { pArg2->m_apBranches.Discard( 1 ); } // // merge the 'null' branches // hres = CEvalTree::Combine( m_pNullBranch, pArg2->m_pNullBranch, nOp, pNamespace, Implications, true, bDeleteArg2, &pNew ); if(FAILED(hres)) return hres; m_pNullBranch = pNew; if( bDeleteArg2 ) pArg2->m_pNullBranch = NULL; // // Delete what needs deleting // if(bDeleteArg2) delete pArg2; *ppRes = this; return WBEM_S_NO_ERROR; } HRESULT CLikeStringPropNode::OptimizeSelf() { _DBG_ASSERT( m_apBranches.GetSize() == 2 ); return WBEM_S_NO_ERROR; } void CLikeStringPropNode::Dump(FILE* f, int nOffset) { CBranchingNode::Dump(f, nOffset); PrintOffset(f, nOffset); fprintf(f, "LastPropName = (0x%x)\n", m_lPropHandle); PrintOffset( f, nOffset ); fprintf(f, "Like Expression : %S\n", m_Like.GetExpression() ); PrintOffset( f, nOffset ); fprintf(f, "Match : \n" ); DumpNode(f, nOffset+1, m_apBranches[0] ); PrintOffset( f, nOffset ); fprintf(f, "NoMatch : \n" ); DumpNode(f, nOffset+1, m_apBranches[1] ); PrintOffset( f, nOffset ); fprintf(f, "NULL : \n" ); DumpNode(f, nOffset+1, m_pNullBranch ); } //****************************************************************************** //****************************************************************************** // INHERITANCE NODE //****************************************************************************** //****************************************************************************** CInheritanceNode::CInheritanceNode() : m_lDerivationIndex(-1), m_lNumPoints(0), m_apcsTestPoints(NULL) { // Add a none-of-the-above node // ============================ m_apBranches.Add(CValueNode::GetStandardFalse()); } CInheritanceNode::CInheritanceNode(const CInheritanceNode& Other, BOOL bChildren) : CBranchingNode(Other, bChildren), m_lDerivationIndex(Other.m_lDerivationIndex) { m_lNumPoints = Other.m_lNumPoints; m_apcsTestPoints = new CCompressedString*[m_lNumPoints]; if(m_apcsTestPoints == NULL) throw CX_MemoryException(); for(int i = 0; i < m_lNumPoints; i++) { m_apcsTestPoints[i] = (CCompressedString*) _new BYTE[Other.m_apcsTestPoints[i]->GetLength()]; if(m_apcsTestPoints[i] == NULL) throw CX_MemoryException(); memcpy((void*)m_apcsTestPoints[i], (void*)Other.m_apcsTestPoints[i], Other.m_apcsTestPoints[i]->GetLength()); } } /* virtual */ long CInheritanceNode::GetSubType() { return EVAL_NODE_TYPE_INHERITANCE; } CInheritanceNode::~CInheritanceNode() { RemoveAllTestPoints(); } void CInheritanceNode::RemoveAllTestPoints() { for(int i = 0; i < m_lNumPoints; i++) { delete [] (BYTE*)m_apcsTestPoints[i]; } delete [] m_apcsTestPoints; m_apcsTestPoints = NULL; } bool CInheritanceNode::SetPropertyInfo(CContextMetaData* pNamespace, CPropertyName& PropName) { return SetEmbeddedObjPropName(PropName); } HRESULT CInheritanceNode::AddClass(CContextMetaData* pNamespace, LPCWSTR wszClassName, CEvalNode* pDestination) { HRESULT hres; // Get the class from the namespace // ================================ _IWmiObject* pObj = NULL; hres = pNamespace->GetClass(wszClassName, &pObj); if(FAILED(hres)) return hres; hres = AddClass(pNamespace, wszClassName, pObj, pDestination); pObj->Release(); return hres; } HRESULT CInheritanceNode::AddClass(CContextMetaData* pNamespace, LPCWSTR wszClassName, _IWmiObject* pClass, CEvalNode* pDestination) { // Get the number of items in its derivation --- that's the index where we // need to look for its name in its children // ======================================================================= ULONG lDerivationIndex; HRESULT hRes = CoreGetNumParents(pClass, &lDerivationIndex); if (FAILED (hRes)) return hRes; if(m_lDerivationIndex == -1) { // We don't have a currently set derivation index --- this is the first // ==================================================================== m_lDerivationIndex = lDerivationIndex; } else if(m_lDerivationIndex != lDerivationIndex) { // Can't add this class --- derivation index mismatch // ================================================== return WBEM_E_FAILED; } // Allocate a compressed string // ============================ int nLength = CCompressedString::ComputeNecessarySpace(wszClassName); CCompressedString* pcs = (CCompressedString*)new BYTE[nLength]; if (pcs) pcs->SetFromUnicode(wszClassName); else return WBEM_E_OUT_OF_MEMORY; // Extend the lists by one // ======================= CCompressedString** apcsNewTestPoints = new CCompressedString*[m_lNumPoints+1]; if (!apcsNewTestPoints) return WBEM_E_OUT_OF_MEMORY; // Insert it into the list of tests and the list of branches // ========================================================= int i = 0; while(i < m_lNumPoints && pcs->CheapCompare(*m_apcsTestPoints[i]) > 0) { apcsNewTestPoints[i] = m_apcsTestPoints[i]; i++; } apcsNewTestPoints[i] = pcs; m_apBranches.InsertAt(i+1, pDestination); while(i < m_lNumPoints) { apcsNewTestPoints[i+1] = m_apcsTestPoints[i]; } // Set the new list // ================ delete [] m_apcsTestPoints; m_apcsTestPoints = apcsNewTestPoints; m_lNumPoints++; return WBEM_S_NO_ERROR; } HRESULT CInheritanceNode::RecordBranch(CContextMetaData* pNamespace, CImplicationList& Implications, long lBranchIndex) { HRESULT hres = WBEM_S_NO_ERROR; if(lBranchIndex == -1) { // Recording NULL branch // ===================== hres = Implications.ImproveKnownNull(GetEmbeddedObjPropName()); } else if(lBranchIndex == 0) { // Recording none of the above branch // ================================== for(int i = 0; i < m_lNumPoints; i++) { LPWSTR wszClassName = NULL; try { wszClassName = m_apcsTestPoints[i]->CreateWStringCopy().UnbindPtr(); } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } if(wszClassName == NULL) return WBEM_E_OUT_OF_MEMORY; CVectorDeleteMe vdm(wszClassName); hres = Implications.ImproveKnownNot(GetEmbeddedObjPropName(), wszClassName); if(FAILED(hres)) { // Contradicts known information --- fail recording // ================================================ return hres; } } } else { // Normal branch --- record the class // ================================== BSTR strClassName = m_apcsTestPoints[lBranchIndex - 1]-> CreateBSTRCopy(); _IWmiObject* pObj = NULL; hres = pNamespace->GetClass(strClassName, &pObj); SysFreeString(strClassName); if(FAILED(hres)) return hres; hres = Implications.ImproveKnown(GetEmbeddedObjPropName(), pObj); pObj->Release(); } return hres; } int CInheritanceNode::ComparePrecedence(CBranchingNode* pOther) { CInheritanceNode* pInhOther = (CInheritanceNode*)pOther; return (m_lDerivationIndex - pInhOther->m_lDerivationIndex); } int CInheritanceNode::SubCompare(CEvalNode* pOther) { CInheritanceNode* pInhOther = (CInheritanceNode*)pOther; int nCompare; nCompare = m_lDerivationIndex - pInhOther->m_lDerivationIndex; if(nCompare) return nCompare; nCompare = m_lNumPoints - pInhOther->m_lNumPoints; if(nCompare) return nCompare; for(int i = 0; i < m_lNumPoints; i++) { nCompare = m_apcsTestPoints[i]->CompareNoCase( *pInhOther->m_apcsTestPoints[i]); if(nCompare) return nCompare; } return 0; } void CInheritanceNode::RemoveTestPoint(int i) { delete [] m_apcsTestPoints[i]; memcpy((void*)(m_apcsTestPoints + i), (void*)(m_apcsTestPoints + i + 1), sizeof(CCompressedString*) * (m_lNumPoints - i - 1)); m_lNumPoints--; } HRESULT CInheritanceNode::OptimizeSelf() { for(int i = 0; i < m_lNumPoints; i++) { // Compare this branch to the "nothing" branch // =========================================== if(CEvalNode::IsInvalid(m_apBranches[i+1]) || CEvalTree::Compare(m_apBranches[0], m_apBranches[i+1]) == 0) { RemoveTestPoint(i); m_apBranches.RemoveAt(i+1); i--; continue; } // Check if this node is another class check on the same object // ============================================================ if(CEvalNode::GetType(m_apBranches[i+1]) != EVAL_NODE_TYPE_BRANCH) continue; CBranchingNode* pBranch = (CBranchingNode*)(m_apBranches[i+1]); if(pBranch->GetSubType() == EVAL_NODE_TYPE_INHERITANCE && pBranch->GetEmbeddedObjPropName() == GetEmbeddedObjPropName()) { // If the "none-of-the-above" branch of this child is the same // as the "none-of-the-above" branch of ourselves, we can replace // our "none-of-the-above" branch with this node, since anything // that is falling under none-of-the-above now will fall under // none-of-the-above of our child (otherwise there is an // optimization flaw in the child). // IMPORTANT: this will no longer be true if we change the // precedence order of inheritance nodes!!! if(CEvalTree::Compare(m_apBranches[0], pBranch->GetBranches()[0]) == 0) { m_apBranches.SetAt(0, pBranch); m_apBranches.GetArrayPtr()[i+1] = NULL; m_apBranches.RemoveAt(i+1); RemoveTestPoint(i); i--; } } } return S_OK; } HRESULT CInheritanceNode::Optimize(CContextMetaData* pNamespace, CEvalNode** ppNew) { // Delegate to the normal branch optimization process // ================================================== *ppNew = NULL; HRESULT hres = CBranchingNode::Optimize(pNamespace, ppNew); if(FAILED(hres) || *ppNew != this) return hres; // Specific post-processing // ======================== if(m_apBranches.GetSize() == 1) { // We are reduced to checking for NULL. If our non-NULL branch is // talking about the same property, push the test there. // ============================================================== if (CEvalNode::GetType(m_apBranches[0]) != EVAL_NODE_TYPE_BRANCH) return hres; CBranchingNode* pBranch = (CBranchingNode*)(m_apBranches[0]); if(pBranch && pBranch->GetSubType() == EVAL_NODE_TYPE_INHERITANCE && pBranch->GetEmbeddedObjPropName() == GetEmbeddedObjPropName()) { pBranch->SetNullBranch(m_pNullBranch); pBranch->Optimize(pNamespace, ppNew); if(*ppNew != pBranch) m_apBranches.RemoveAll(); else m_apBranches.GetArrayPtr()[0] = NULL; m_pNullBranch = NULL; return S_OK; } } return S_OK; } HRESULT CInheritanceNode::Evaluate(CObjectInfo& ObjInfo, INTERNAL CEvalNode** ppNext) { _IWmiObject* pInst; HRESULT hres = GetContainerObject(ObjInfo, &pInst); if(FAILED(hres)) return hres; if(pInst == NULL) { *ppNext = m_pNullBranch; return WBEM_S_NO_ERROR; } // Get the parent at the right index // ================================= CCompressedString* pcs; ULONG lNumParents; HRESULT hRes = CoreGetNumParents(pInst, &lNumParents); if (FAILED(hRes)) return hRes; if(lNumParents < m_lDerivationIndex) { if (m_apBranches.GetSize()) *ppNext = m_apBranches[0]; else *ppNext = NULL; return WBEM_S_NO_ERROR; } else if(lNumParents == m_lDerivationIndex) { pcs = CoreGetClassInternal(pInst); } else { pcs = CoreGetParentAtIndex(pInst, m_lDerivationIndex); } if(pcs == NULL) { // // This class does not even have that long an ancestry --- clearly // not derived from any of those // if (m_apBranches.GetSize()) *ppNext = m_apBranches[0]; else *ppNext = NULL; return WBEM_S_NO_ERROR; } // Search for the value // ==================== long lLeft = -1; long lRight = m_lNumPoints; while(lRight > lLeft + 1) { long lMiddle = (lRight + lLeft) >> 1; int nCompare = pcs->CheapCompare(*m_apcsTestPoints[lMiddle]); if(nCompare < 0) { lRight = lMiddle; } else if(nCompare > 0) { lLeft = lMiddle; } else { *ppNext = m_apBranches[lMiddle+1]; return WBEM_S_NO_ERROR; } } if (m_apBranches.GetSize()) *ppNext = m_apBranches[0]; else *ppNext = NULL; return WBEM_S_NO_ERROR; } HRESULT CInheritanceNode::Compile(CContextMetaData* pNamespace, CImplicationList& Implications) { if (!m_pInfo) { m_pInfo = new CEmbeddingInfo; if(m_pInfo == NULL) return WBEM_E_OUT_OF_MEMORY; } HRESULT hres = CompileEmbeddingPortion(pNamespace, Implications, NULL); return hres; } // Computes preliminary parameters for merging two inheritance nodes at the // same level --- which children will be used, and how many times. HRESULT CInheritanceNode::ComputeUsageForMerge(CInheritanceNode* pArg2, CContextMetaData* pNamespace, CImplicationList& OrigImplications, bool bDeleteThis, bool bDeleteArg2, DWORD* pdwFirstNoneCount, DWORD* pdwSecondNoneCount, bool* pbBothNonePossible) { HRESULT hres; *pdwFirstNoneCount = 0; *pdwSecondNoneCount = 0; *pbBothNonePossible = false; try { CImplicationList Implications(OrigImplications); BOOL bFirstNonePossible, bSecondNonePossible; CImplicationList NoneImplications(Implications); hres = RecordBranch(pNamespace, NoneImplications, 0); if(FAILED(hres)) { bFirstNonePossible = FALSE; bSecondNonePossible = SUCCEEDED(pArg2->RecordBranch(pNamespace, NoneImplications, 0)); } else { bFirstNonePossible = TRUE; hres = pArg2->RecordBranch(pNamespace, NoneImplications, 0); if(FAILED(hres)) { // Check if the second one can survive in isolation // ================================================ CImplicationList NoneImplications1(Implications); bSecondNonePossible = SUCCEEDED(pArg2->RecordBranch(pNamespace, NoneImplications1, 0)); } else { bSecondNonePossible = TRUE; } } if(bFirstNonePossible && bSecondNonePossible) { // // Both of them will be used at least once: with each other! // *pdwFirstNoneCount = *pdwSecondNoneCount = 1; *pbBothNonePossible = true; } // // If we are not deleting something, the usage count should be infinite! // if(!bDeleteThis) *pdwFirstNoneCount = 0xFFFFFFFF; if(!bDeleteArg2) *pdwSecondNoneCount = 0xFFFFFFFF; // // Merge lookup lists // long lFirstIndex = 0; long lSecondIndex = 0; while(lFirstIndex < m_lNumPoints || lSecondIndex < pArg2->m_lNumPoints) { // // Retrieve the test points from both lists and compare them, // taking care of boundary conditions // int nCompare; CCompressedString* pcsFirstVal = NULL; CCompressedString* pcsSecondVal = NULL; if(lFirstIndex == m_lNumPoints) { nCompare = 1; pcsSecondVal = pArg2->m_apcsTestPoints[lSecondIndex]; } else if(lSecondIndex == pArg2->m_lNumPoints) { pcsFirstVal = m_apcsTestPoints[lFirstIndex]; nCompare = -1; } else { pcsFirstVal = m_apcsTestPoints[lFirstIndex]; pcsSecondVal = pArg2->m_apcsTestPoints[lSecondIndex]; nCompare = pcsFirstVal->CheapCompare(*pcsSecondVal); } if(nCompare < 0) { // // At this index is the first value combined with second none // if(!bDeleteArg2) // not interesting { lFirstIndex++; continue; } if(!bSecondNonePossible) { lFirstIndex++; continue; } CImplicationList BranchImplications(Implications); if(FAILED(RecordBranch(pNamespace, BranchImplications, lFirstIndex+1))) { lFirstIndex++; continue; } if(FAILED(pArg2->RecordBranch(pNamespace, BranchImplications, 0))) { lFirstIndex++; continue; } (*pdwSecondNoneCount)++; lFirstIndex++; } else if(nCompare > 0) { // At this index is the second value combined with first none // ========================================================== if(!bDeleteThis) // not interesting { lSecondIndex++; continue; } if(!bFirstNonePossible) { lSecondIndex++; continue; } CImplicationList BranchImplications(Implications); if(FAILED(pArg2->RecordBranch(pNamespace, BranchImplications, lSecondIndex+1))) { lSecondIndex++; continue; } if(FAILED(RecordBranch(pNamespace, BranchImplications, 0))) { lSecondIndex++; continue; } (*pdwFirstNoneCount)++; lSecondIndex++; } else { // At this index is the combinations of the ats // ============================================ lFirstIndex++; lSecondIndex++; } } } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } return S_OK; } HRESULT CInheritanceNode::CombineBranchesWith(CBranchingNode* pRawArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& OrigImplications, bool bDeleteThis, bool bDeleteArg2, CEvalNode** ppRes) { HRESULT hres; CInheritanceNode* pArg2 = (CInheritanceNode*)pRawArg2; if(!bDeleteThis && bDeleteArg2) { // It is easier to combine in the other direction // ============================================== return pArg2->CombineBranchesWith(this, FlipEvalOp(nOp), pNamespace, OrigImplications, bDeleteArg2, bDeleteThis, ppRes); } try { // Either clone or use our node // ============================ CInheritanceNode* pNew = NULL; if(bDeleteThis) { pNew = this; } else { // Clone this node without cloning the branches. // ============================================= pNew = (CInheritanceNode*)CloneSelf(); if(pNew == NULL) return WBEM_E_OUT_OF_MEMORY; } CImplicationList Implications(OrigImplications); hres = pNew->AdjustCompile(pNamespace, Implications); if(FAILED(hres)) { if(!bDeleteThis) delete pNew; return hres; } // // Get overall information // DWORD dwFirstNoneCount, dwSecondNoneCount; bool bBothNonePossible; hres = ComputeUsageForMerge(pArg2, pNamespace, Implications, bDeleteThis, bDeleteArg2, &dwFirstNoneCount, &dwSecondNoneCount, &bBothNonePossible); if(FAILED(hres)) { if(!bDeleteThis) delete pNew; return hres; } bool bFirstNonePossible = (dwFirstNoneCount > 0); bool bSecondNonePossible = (dwSecondNoneCount > 0); // // Allocate a new array of test points and a new array of branches. We // can't use the ones in pNew because we are using some of the // elements in those arrays in this many times and don't want to trample // them (since pNew and this might be the same). Therefore, we create // and fill these arrays outside of the node and then place them into // pNew when we are done. // // As we delete the child nodes in the Branches array, we will set them // to NULL so that we can clear the branch array in the end // CCompressedString** apcsTestPoints = new CCompressedString*[m_lNumPoints + pArg2->m_lNumPoints]; if(apcsTestPoints == NULL) { if(!bDeleteThis) delete pNew; return WBEM_E_OUT_OF_MEMORY; } CUniquePointerArray apBranches; // // Merge none-of-the-above branches // if(bBothNonePossible) { // // They have to be merged // CImplicationList NoneImplications(Implications); hres = RecordBranch(pNamespace, NoneImplications, 0); if(FAILED(hres)) { if(!bDeleteThis) delete pNew; return hres; } hres = pArg2->RecordBranch(pNamespace, NoneImplications, 0); if(FAILED(hres)) { if(!bDeleteThis) delete pNew; return hres; } // // We may delete our None nodes if and only if there predicted usage // count (adjusted for usage that has already occurred) is dropping // to 0 --- that is, noone will use these nodes further during this // merge // CEvalNode* pNone = NULL; bool bDeleteFirstBranch = (--dwFirstNoneCount == 0); bool bDeleteSecondBranch = (--dwSecondNoneCount == 0); hres = CEvalTree::Combine(m_apBranches[0], pArg2->m_apBranches[0], nOp, pNamespace, NoneImplications, bDeleteFirstBranch, bDeleteSecondBranch, &pNone); if ( FAILED( hres ) ) { if(!bDeleteThis) delete pNew; return hres; } if(bDeleteSecondBranch) pArg2->m_apBranches.Discard(0); if(bDeleteFirstBranch) m_apBranches.Discard(0); if(apBranches.Add(pNone) < 0) { if(!bDeleteThis) delete pNew; return WBEM_E_OUT_OF_MEMORY; } } else { // // Since both none is not possible, we set this branch to FALSE // if(apBranches.Add(NULL) < 0) { if(!bDeleteThis) delete pNew; return WBEM_E_OUT_OF_MEMORY; } } // // Merge lookup lists // long lFirstIndex = 0; long lSecondIndex = 0; long lNewIndex = 0; while(lFirstIndex < m_lNumPoints || lSecondIndex < pArg2->m_lNumPoints) { // // Retrieve the test points from both lists and compare them, // taking care of boundary conditions // int nCompare; CCompressedString* pcsFirstVal = NULL; CCompressedString* pcsSecondVal = NULL; if(lFirstIndex == m_lNumPoints) { nCompare = 1; pcsSecondVal = pArg2->m_apcsTestPoints[lSecondIndex]; } else if(lSecondIndex == pArg2->m_lNumPoints) { pcsFirstVal = m_apcsTestPoints[lFirstIndex]; nCompare = -1; } else { pcsFirstVal = m_apcsTestPoints[lFirstIndex]; pcsSecondVal = pArg2->m_apcsTestPoints[lSecondIndex]; nCompare = pcsFirstVal->CheapCompare(*pcsSecondVal); } // // Compute the branch to be added and its test point // CEvalNode* pNewBranch = NULL; CCompressedString* pcsCurrentVal = NULL; if(nCompare < 0) { // // At this index is the first value combined with second none // if(!bSecondNonePossible) { lFirstIndex++; continue; } CImplicationList BranchImplications(Implications); if(FAILED(RecordBranch(pNamespace, BranchImplications, lFirstIndex+1))) { lFirstIndex++; continue; } pArg2->RecordBranch(pNamespace, BranchImplications, 0); // // We may delete our None nodes if and only if there predicted // usage count (adjusted for usage that has already occurred) is // dropping to 0 --- that is, noone will use these nodes further // during this merge // bool bDeleteOtherBranch = (--dwSecondNoneCount == 0); hres = CEvalTree::Combine(m_apBranches[lFirstIndex+1], pArg2->m_apBranches[0], nOp, pNamespace, BranchImplications, bDeleteThis, bDeleteOtherBranch, &pNewBranch); if ( FAILED( hres ) ) { if(!bDeleteThis) delete pNew; return hres; } if(bDeleteOtherBranch) pArg2->m_apBranches.Discard(0); if(bDeleteThis) m_apBranches.Discard(lFirstIndex+1); pcsCurrentVal = pcsFirstVal; lFirstIndex++; } else if(nCompare > 0) { // At this index is the second value combined with first none // ========================================================== if(!bFirstNonePossible) { lSecondIndex++; continue; } CImplicationList BranchImplications(Implications); if(FAILED(pArg2->RecordBranch(pNamespace, BranchImplications, lSecondIndex+1))) { lSecondIndex++; continue; } if(FAILED(RecordBranch(pNamespace, BranchImplications, 0))) { lSecondIndex++; continue; } // // We may delete our None nodes if and only if there predicted // usage count (adjusted for usage that has already occurred) is // dropping to 0 --- that is, noone will use these nodes further // during this merge // bool bDeleteThisBranch = (--dwFirstNoneCount == 0); hres = CEvalTree::Combine(m_apBranches[0], pArg2->m_apBranches[lSecondIndex+1], nOp, pNamespace, BranchImplications, bDeleteThisBranch, bDeleteArg2, &pNewBranch); if ( FAILED( hres ) ) { if(!bDeleteThis) delete pNew; return hres; } if(bDeleteArg2) pArg2->m_apBranches.Discard(lSecondIndex+1); if(bDeleteThisBranch) m_apBranches.Discard(0); pcsCurrentVal = pcsSecondVal; lSecondIndex++; } else { // At this index is the combinations of the ats // ============================================ CImplicationList BranchImplications(Implications); if(FAILED(RecordBranch(pNamespace, BranchImplications, lFirstIndex+1))) { lSecondIndex++; lFirstIndex++; continue; } hres = CEvalTree::Combine(m_apBranches[lFirstIndex+1], pArg2->m_apBranches[lSecondIndex+1], nOp, pNamespace, BranchImplications, bDeleteThis, bDeleteArg2, &pNewBranch); if ( FAILED( hres ) ) { if(!bDeleteThis) delete pNew; return hres; } if(bDeleteArg2) pArg2->m_apBranches.Discard(lSecondIndex+1); if(bDeleteThis) m_apBranches.Discard(lFirstIndex+1); pcsCurrentVal = pcsFirstVal; // doesn't matter --- same lFirstIndex++; lSecondIndex++; } // // Add the newely constructed branch // if(apBranches.Add(pNewBranch) < 0) { if ( !bDeleteThis ) delete pNew; return WBEM_E_OUT_OF_MEMORY; } // // Add the newely constructed test point // apcsTestPoints[lNewIndex] = (CCompressedString*)_new BYTE[pcsCurrentVal->GetLength()]; if(apcsTestPoints[lNewIndex] == NULL) { if ( !bDeleteThis ) delete pNew; return WBEM_E_OUT_OF_MEMORY; } memcpy((void*)apcsTestPoints[lNewIndex], (void*)pcsCurrentVal, pcsCurrentVal->GetLength()); lNewIndex++; } // // Now that we are done with testing, place the test-point array into // pNew // pNew->RemoveAllTestPoints(); pNew->m_apcsTestPoints = apcsTestPoints; pNew->m_lNumPoints = lNewIndex; // // Replace the array of branches that we may have had (guaranteed to // all be NULL, since we were NULLing them out as we went). // pNew->m_apBranches.RemoveAll(); for(int i = 0; i < apBranches.GetSize(); i++) { pNew->m_apBranches.Add(apBranches[i]); apBranches.Discard(i); } // // Merge the nulls // CImplicationList NullImplications(Implications); hres = RecordBranch(pNamespace, Implications, -1); if(SUCCEEDED(hres)) { CEvalNode* pNewBranch = NULL; hres = CEvalTree::Combine(m_pNullBranch, pArg2->m_pNullBranch, nOp, pNamespace, NullImplications, bDeleteThis, bDeleteArg2, &pNewBranch); if ( FAILED( hres ) ) { if(!bDeleteThis) delete pNew; return hres; } // // Clear the old new branch, whatever it was, (it has been deleted, // if it ever were there at all) and replace it with the new one. // pNew->m_pNullBranch = pNewBranch; // Clear deleted branches // ====================== if(bDeleteArg2) pArg2->m_pNullBranch = NULL; } // Delete Arg2, if needed (reused portions have been nulled out) // ============================================================= if(bDeleteArg2) delete pArg2; *ppRes = pNew; return WBEM_S_NO_ERROR; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } } HRESULT CInheritanceNode::Project(CContextMetaData* pMeta, CImplicationList& Implications, CProjectionFilter* pFilter, EProjectionType eType, bool bDeleteThis, CEvalNode** ppNewNode) { // // There are two choices here: either it is about us, or we are not // interested. // if(pFilter->IsInSet(this)) { return CBranchingNode::Project(pMeta, Implications, pFilter, eType, bDeleteThis, ppNewNode); } else { if(eType == e_Sufficient) *ppNewNode = CValueNode::GetStandardFalse(); else { *ppNewNode = CValueNode::GetStandardTrue(); if(*ppNewNode == NULL) return WBEM_E_OUT_OF_MEMORY; } if(bDeleteThis) delete this; return S_OK; } } void CInheritanceNode::Dump(FILE* f, int nOffset) { CBranchingNode::Dump(f, nOffset); PrintOffset(f, nOffset); fprintf(f, "inheritance index %d: (0x%p)\n", m_lDerivationIndex, this); for(int i = 0; i < m_lNumPoints; i++) { WString ws = m_apcsTestPoints[i]->CreateWStringCopy(); PrintOffset(f, nOffset); fprintf(f, "%S->\n", (LPWSTR)ws); DumpNode(f, nOffset+1, m_apBranches[i+1]); } PrintOffset(f, nOffset); fprintf(f, "none of the above->\n"); DumpNode(f, nOffset+1, m_apBranches[0]); PrintOffset(f, nOffset); fprintf(f, "NULL->\n"); DumpNode(f, nOffset+1, m_pNullBranch); } #ifdef CHECK_TREES void CBranchingNode::CheckNode(CTreeChecker *pCheck) { pCheck->CheckoffNode(this); int nItems = m_apBranches.GetSize(); for (int i = 0; i < nItems; i++) { CEvalNode *pNode = m_apBranches[i]; if (pNode) m_apBranches[i]->CheckNode(pCheck); } if (m_pNullBranch) m_pNullBranch->CheckNode(pCheck); } #endif //****************************************************************************** //****************************************************************************** // // OR NODE // //****************************************************************************** //****************************************************************************** void COrNode::operator=(const COrNode& Other) { // Remove all our children // ======================= m_apBranches.RemoveAll(); // Clone all the branches from the other // ===================================== for(int i = 0; i < Other.m_apBranches.GetSize(); i++) { CEvalNode* pNewBranch = CloneNode(Other.m_apBranches[i]); if(pNewBranch == NULL && Other.m_apBranches[i] != NULL) throw CX_MemoryException(); if(m_apBranches.Add(pNewBranch) < 0) throw CX_MemoryException(); } } HRESULT COrNode::CombineWith(CEvalNode* pArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& Implications, bool bDeleteThis, bool bDeleteArg2, CEvalNode** ppRes) { HRESULT hres; *ppRes = NULL; // We don't support AND combines on OR nodes // ========================================= if(nOp == EVAL_OP_AND) return WBEM_E_CRITICAL_ERROR; // If the other is another OR node, delegate to the iterator // ========================================================= if(CEvalNode::GetType(pArg2) == EVAL_NODE_TYPE_OR) { return CombineWithOrNode((COrNode*)pArg2, nOp, pNamespace, Implications, bDeleteThis, bDeleteArg2, ppRes); } // Make a copy --- the new node will be mostly us // ============================================== COrNode* pNew = NULL; if(!bDeleteThis) { pNew = (COrNode*)Clone(); if(pNew == NULL) return WBEM_E_OUT_OF_MEMORY; } else { pNew = this; } // Combining an OR node with a non-OR --- try all branches // ======================================================= for(int i = 0; i < m_apBranches.GetSize(); i++) { // Check if this branch is a good fit for the other node // ===================================================== if(CEvalTree::IsMergeAdvisable(m_apBranches[i], pArg2, Implications) == WBEM_S_NO_ERROR) { // It is --- merge it in // ===================== CEvalNode* pNewBranch = NULL; hres = CEvalTree::Combine(m_apBranches[i], pArg2, nOp, pNamespace, Implications, bDeleteThis, bDeleteArg2, &pNewBranch); if(FAILED(hres)) { if ( !bDeleteThis ) delete pNew; return hres; } if(bDeleteThis) m_apBranches.Discard(i); pNew->m_apBranches.SetAt(i, pNewBranch); *ppRes = pNew; return WBEM_S_NO_ERROR; } } // No branch was a good fit --- add the node to our list // ===================================================== if(bDeleteArg2) { hres = pNew->AddBranch(pArg2); } else { CEvalNode* pNode = pArg2->Clone(); if(pNode == NULL) { if ( !bDeleteThis ) delete pNew; return WBEM_E_OUT_OF_MEMORY; } hres = pNew->AddBranch(pNode); } if(FAILED(hres)) { if ( !bDeleteThis ) delete pNew; return hres; } *ppRes = pNew; return WBEM_S_NO_ERROR; } HRESULT COrNode::AddBranch(CEvalNode* pNewBranch) { // Search for a place in our array of branches // =========================================== for(int i = 0; i < m_apBranches.GetSize(); i++) { int nCompare = CEvalTree::Compare(pNewBranch, m_apBranches[i]); if(nCompare == 0) { // Could happen: sometimes we force an OR-merge nCompare = -1; } if(nCompare < 0) { // pNewBranch comes before this branch, so insert it right here // ============================================================ if(!m_apBranches.InsertAt(i, pNewBranch)) return WBEM_E_OUT_OF_MEMORY; return WBEM_S_NO_ERROR; } } // It is after all branches --- append // =================================== if(m_apBranches.Add(pNewBranch) < 0) return WBEM_E_OUT_OF_MEMORY; return WBEM_S_NO_ERROR; } HRESULT COrNode::CombineWithOrNode(COrNode* pArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& Implications, bool bDeleteThis, bool bDeleteArg2, CEvalNode** ppRes) { *ppRes = NULL; // NOTE: this function may delete THIS in the middle of execution!! // ================================================================ // Combine us with every branch of the other // ========================================= CEvalNode* pCurrent = this; bool bDeleteCurrent = bDeleteThis; for(int i = 0; i < pArg2->m_apBranches.GetSize(); i++) { CEvalNode* pNew = NULL; HRESULT hres = pCurrent->CombineWith(pArg2->m_apBranches[i], nOp, pNamespace, Implications, bDeleteCurrent, bDeleteArg2, &pNew); if(FAILED(hres)) return hres; pCurrent = pNew; // At this point, we can safely delete pCurrent if needed --- it's ours // ==================================================================== bDeleteCurrent = TRUE; // If pArg2's branch has been deleted, reset it // ============================================ if(bDeleteArg2) pArg2->m_apBranches.Discard(i); } *ppRes = pCurrent; if(bDeleteArg2) delete pArg2; return WBEM_S_NO_ERROR; } int COrNode::Compare(CEvalNode* pOther) { COrNode* pOtherNode = (COrNode*)pOther; // Compare array sizes // =================== if(m_apBranches.GetSize() != pOtherNode->m_apBranches.GetSize()) return m_apBranches.GetSize() - pOtherNode->m_apBranches.GetSize(); // Compare individual nodes // ======================== for(int i = 0; i < m_apBranches.GetSize(); i++) { int nCompare = CEvalTree::Compare(m_apBranches[i], pOtherNode->m_apBranches[i]); if(nCompare != 0) return nCompare; } return 0; } DWORD COrNode::ApplyPredicate(CLeafPredicate* pPred) { for(int i = 0; i < m_apBranches.GetSize(); i++) { if (m_apBranches[i]) m_apBranches[i]->ApplyPredicate(pPred); } return WBEM_DISPOSITION_NORMAL; } HRESULT COrNode::Optimize(CContextMetaData* pNamespace, CEvalNode** ppNew) { HRESULT hres = WBEM_S_NO_ERROR; // First, optimize all its branches // ================================ for(int i = 0; i < m_apBranches.GetSize(); i++) { CEvalNode* pNew = NULL; if (m_apBranches[i]) { hres = m_apBranches[i]->Optimize(pNamespace, &pNew); if(FAILED(hres)) return hres; } if(pNew != m_apBranches[i]) { // Replace, but check for emptiness first // ====================================== if(!CEvalNode::IsAllFalse(pNew)) m_apBranches.SetAt(i, pNew); else { delete pNew; m_apBranches.RemoveAt(i); i--; } } } if(m_apBranches.GetSize() == 0) { // We have no branches --- equivalent to no successes // ================================================== *ppNew = CValueNode::GetStandardFalse(); return WBEM_S_NO_ERROR; } else if(m_apBranches.GetSize() == 1) { // One branch --- equivalent to that branch // ======================================== m_apBranches.RemoveAt(0, ppNew); } else { *ppNew = this; } return WBEM_S_NO_ERROR; } void COrNode::Dump(FILE* f, int nOffset) { PrintOffset(f, nOffset); fprintf(f, "FOREST\n"); for(int i = 0; i < m_apBranches.GetSize(); i++) { if (m_apBranches[i]) m_apBranches[i]->Dump(f, nOffset+1); else fprintf(f, "all false ValueNode (or error?)\n"); } } HRESULT COrNode::Evaluate(CObjectInfo& Info, CSortedArray& trueIDs) { for(int i = 0; i < m_apBranches.GetSize(); i++) { if (m_apBranches[i]) { HRESULT hres = CEvalTree::Evaluate(Info, m_apBranches[i], trueIDs); if(FAILED(hres)) return hres; } } return WBEM_S_NO_ERROR; } HRESULT COrNode::Project(CContextMetaData* pMeta, CImplicationList& Implications, CProjectionFilter* pFilter, EProjectionType eType, bool bDeleteThis, CEvalNode** ppNewNode) { *ppNewNode = NULL; COrNode* pNew; if(bDeleteThis) pNew = this; else pNew = (COrNode*)Clone(); if(pNew == NULL) return WBEM_E_OUT_OF_MEMORY; // // Simply project all the branches // for(int i = 0; i < pNew->m_apBranches.GetSize(); i++) { CEvalNode* pProjected = NULL; HRESULT hres = CEvalTree::Project(pMeta, Implications, pNew->m_apBranches[i], pFilter, eType, true, // always delete --- has been cloned &pProjected); if(FAILED(hres)) return hres; pNew->m_apBranches.Discard(i); pNew->m_apBranches.SetAt(i, pProjected); } *ppNewNode = pNew; return S_OK; } //****************************************************************************** //****************************************************************************** // PREDICATES //****************************************************************************** //****************************************************************************** // NOTE: Not checking for NULL leaves, should be checked by caller DWORD CEvalTree::CRemoveIndexPredicate::operator()(CValueNode* pLeaf) { if(pLeaf) { pLeaf->RemoveQueryID(m_nIndex); if(pLeaf->GetNumTrues() == 0) return WBEM_DISPOSITION_FLAG_DELETE; } return WBEM_DISPOSITION_NORMAL; } // NOTE: Not checking for NULL leaves, should be checked by caller DWORD CEvalTree::CRebasePredicate::operator()(CValueNode* pLeaf) { if(pLeaf) pLeaf->Rebase(m_newBase); return WBEM_DISPOSITION_NORMAL; } // NOTE: Not checking for NULL leaves, should be checked by caller DWORD CEvalTree::CRemoveFailureAtIndexPredicate::operator()(CValueNode* pLeaf) { if(pLeaf == NULL || pLeaf->GetAt(m_nIndex) != EVAL_VALUE_TRUE) return WBEM_DISPOSITION_FLAG_INVALIDATE; pLeaf->RemoveQueryID(m_nIndex); if(pLeaf->GetNumTrues() == 0) return WBEM_DISPOSITION_FLAG_DELETE; return WBEM_DISPOSITION_NORMAL; } //****************************************************************************** //****************************************************************************** // EVALUATION TREE //****************************************************************************** //****************************************************************************** CEvalTree::CEvalTree() : m_lRef(0), m_pStart(NULL), m_nNumValues(0) { #ifdef CHECK_TREES g_treeChecker.AddTree(this); #endif } CEvalTree::CEvalTree(const CEvalTree& Other) : m_lRef(0), m_pStart(NULL), m_nNumValues(0) { #ifdef CHECK_TREES g_treeChecker.AddTree(this); #endif *this = Other; } CEvalTree::~CEvalTree() { #ifdef CHECK_TREES g_treeChecker.RemoveTree(this); #endif delete m_pStart; } bool CEvalTree::SetBool(BOOL bVal) { CInCritSec ics(&m_cs); delete m_pStart; CValueNode* pNode; if (bVal) { pNode = CValueNode::GetStandardTrue(); if(pNode == NULL) return false; } else pNode = CValueNode::GetStandardFalse(); m_pStart = pNode; m_nNumValues = 1; if(!m_ObjectInfo.SetLength(1)) return false; return true; } bool CEvalTree::IsFalse() { return (m_pStart == NULL); } bool CEvalTree::IsValid() { return !CEvalNode::IsInvalid(m_pStart); } int CEvalTree::Compare(CEvalNode* pArg1, CEvalNode* pArg2) { if(pArg1 == NULL) { if(pArg2 == NULL) return 0; else return 1; } else if(pArg2 == NULL) return -1; else if(CEvalNode::GetType(pArg1) != CEvalNode::GetType(pArg2)) return CEvalNode::GetType(pArg1) - CEvalNode::GetType(pArg2); else return pArg1->Compare(pArg2); } HRESULT CEvalTree::CreateFromQuery(CContextMetaData* pNamespace, LPCWSTR wszQuery, long lFlags, long lMaxTokens) { CTextLexSource src((LPWSTR)wszQuery); QL1_Parser parser(&src); QL_LEVEL_1_RPN_EXPRESSION *pExp = 0; int nRes = parser.Parse(&pExp); CDeleteMe deleteMe(pExp); if (nRes) return WBEM_E_INVALID_QUERY; HRESULT hres = CreateFromQuery(pNamespace, pExp, lFlags, lMaxTokens); return hres; } HRESULT CEvalTree::CreateFromQuery(CContextMetaData* pNamespace, QL_LEVEL_1_RPN_EXPRESSION* pQuery, long lFlags, long lMaxTokens) { return CreateFromQuery(pNamespace, pQuery->bsClassName, pQuery->nNumTokens, pQuery->pArrayOfTokens, lFlags, lMaxTokens); } HRESULT CEvalTree::CreateFromQuery(CContextMetaData* pNamespace, LPCWSTR wszClassName, int nNumTokens, QL_LEVEL_1_TOKEN* apTokens, long lFlags, long lMaxTokens) { CInCritSec ics(&m_cs); HRESULT hres; // Create basic implication list // ============================= _IWmiObject* pObj = NULL; hres = pNamespace->GetClass(wszClassName, &pObj); if(FAILED(hres)) { return hres; } CReleaseMe rm1(pObj); try { CImplicationList Implications(lFlags); CPropertyName EmptyName; Implications.ImproveKnown(&EmptyName, pObj); #ifdef CHECK_TREES CheckTrees(); #endif CEvalNode* pWhere = NULL; if(nNumTokens) { // Convert the token list to DNF // ============================= CDNFExpression DNF; QL_LEVEL_1_TOKEN* pEnd = apTokens + nNumTokens - 1; hres = DNF.CreateFromTokens(pEnd, 0, lMaxTokens); if(FAILED(hres)) return hres; if(pEnd != apTokens - 1) { return WBEM_E_CRITICAL_ERROR; } DNF.Sort(); // Build a tree for the token list // =============================== hres = CreateFromDNF(pNamespace, Implications, &DNF, &pWhere); if(FAILED(hres)) { return hres; } } else { pWhere = CValueNode::GetStandardTrue(); if(pWhere == NULL) return WBEM_E_OUT_OF_MEMORY; } // Add inheritance check // ===================== CInheritanceNode* pInhNode = new CInheritanceNode; if (!pInhNode) return WBEM_E_OUT_OF_MEMORY; hres = pInhNode->AddClass(pNamespace, wszClassName, (_IWmiObject*)pObj, pWhere); if(FAILED(hres)) { delete pWhere; delete pInhNode; return hres; } if(!m_ObjectInfo.SetLength(Implications.GetRequiredDepth())) { delete pInhNode; return WBEM_E_OUT_OF_MEMORY; } delete m_pStart; m_pStart = pInhNode; m_nNumValues = 1; #ifdef CHECK_TREES CheckTrees(); #endif Optimize(pNamespace); #ifdef CHECK_TREES CheckTrees(); #endif return WBEM_S_NO_ERROR; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } } // extension of BuildFromToken to build nodes that have two properties // e.g. this would service "select * from class where prop1 < prop2" HRESULT CEvalTree::BuildTwoPropFromToken(CContextMetaData* pNamespace, CImplicationList& Implications, QL_LEVEL_1_TOKEN& Token, CEvalNode** ppRes) { HRESULT hres; CEmbeddingInfo leftInfo, rightInfo; if(!leftInfo.SetPropertyNameButLast(Token.PropertyName)) return WBEM_E_OUT_OF_MEMORY; if(!rightInfo.SetPropertyNameButLast(Token.PropertyName2)) return WBEM_E_OUT_OF_MEMORY; _IWmiObject* pLeftClass; hres = leftInfo.Compile(pNamespace, Implications, &pLeftClass); if(FAILED(hres)) return hres; if(pLeftClass == NULL) return WBEMESS_E_REGISTRATION_TOO_BROAD; _IWmiObject* pRightClass; hres = rightInfo.Compile(pNamespace, Implications, &pRightClass); if(FAILED(hres)) return hres; if(pRightClass == NULL) { pLeftClass->Release(); return WBEMESS_E_REGISTRATION_TOO_BROAD; } // Get the property types and handles // ================================== LPCWSTR wszLeftPropName = Token.PropertyName.GetStringAt( Token.PropertyName.GetNumElements() - 1); LPCWSTR wszRightPropName = Token.PropertyName2.GetStringAt( Token.PropertyName2.GetNumElements() - 1); CIMTYPE ctLeft, ctRight; long lLeftPropHandle, lRightPropHandle; hres = pLeftClass->GetPropertyHandleEx(wszLeftPropName, 0L, &ctLeft, &lLeftPropHandle); pLeftClass->Release(); if(FAILED(hres)) return hres; hres = pRightClass->GetPropertyHandleEx(wszRightPropName, 0L, &ctRight, &lRightPropHandle); pRightClass->Release(); if(FAILED(hres)) return hres; if( ((ctLeft & CIM_FLAG_ARRAY) != 0) || (ctLeft == CIM_OBJECT) || ((ctRight & CIM_FLAG_ARRAY) != 0) || (ctRight == CIM_OBJECT) ) return WBEM_E_NOT_SUPPORTED; // if the type of either node is reference or date, go to dumb if ( (ctLeft == CIM_DATETIME) || (ctLeft == CIM_REFERENCE) || (ctRight == CIM_DATETIME) || (ctRight == CIM_REFERENCE) ) { if(ctLeft != ctRight) return WBEM_E_TYPE_MISMATCH; CDumbNode* pDumb = NULL; try { pDumb = new CDumbNode(Token); if(pDumb == NULL) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } hres = pDumb->Validate(pLeftClass); if(FAILED(hres)) { delete pDumb; return hres; } *ppRes = pDumb; return WBEM_S_NO_ERROR; } // if the node is mismatched (different types in either node) // we'll want to create the most flexible node needed to hold both types CIMTYPE ctNode = ctLeft; CPropertyNode* pNode = NULL; try { bool bMismatchedNode = (ctLeft != ctRight); if (bMismatchedNode) { // we'll be real forgiving about strings matching strings if ( (ctLeft == CIM_STRING) || (ctRight == CIM_STRING) ) pNode = new CTwoMismatchedStringNode; else if ( (ctRight == CIM_REAL32) || (ctRight == CIM_REAL64) || (ctLeft == CIM_REAL32) || (ctLeft == CIM_REAL64) ) pNode = new CTwoMismatchedFloatNode; else if ( (ctLeft == CIM_UINT64) || (ctRight == CIM_UINT64) ) pNode = new CTwoMismatchedUInt64Node; else if ( (ctLeft == CIM_SINT64 ) || (ctRight == CIM_SINT64 ) ) pNode = new CTwoMismatchedInt64Node; else if ( (ctRight == CIM_UINT32) || (ctLeft == CIM_UINT32) ) pNode = new CTwoMismatchedUIntNode; else pNode = new CTwoMismatchedIntNode; } else // not mistmatched - go with the exact type { // Create the Proper node // ===================== switch(ctNode) { case CIM_SINT8: pNode = new TTwoScalarPropNode; break; case CIM_UINT8: pNode = new TTwoScalarPropNode; break; case CIM_SINT16: pNode = new TTwoScalarPropNode; break; case CIM_UINT16: case CIM_CHAR16: pNode = new TTwoScalarPropNode; break; case CIM_SINT32: pNode = new TTwoScalarPropNode; break; case CIM_UINT32: pNode = new TTwoScalarPropNode; break; case CIM_SINT64: pNode = new TTwoScalarPropNode<__int64>; break; case CIM_UINT64: pNode = new TTwoScalarPropNode; break; case CIM_REAL32: pNode = new TTwoScalarPropNode; break; case CIM_REAL64: pNode = new TTwoScalarPropNode; break; case CIM_BOOLEAN: pNode = new TTwoScalarPropNode; break; case CIM_STRING: pNode = new CTwoStringPropNode; break; case CIM_DATETIME: case CIM_REFERENCE: { CDumbNode* pDumb = new CDumbNode(Token); hres = pDumb->Validate(pLeftClass); if(FAILED(hres)) { delete pDumb; return hres; } else { *ppRes = pDumb; return WBEM_S_NO_ERROR; } } return WBEM_S_NO_ERROR; default: return WBEM_E_CRITICAL_ERROR; } } } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } if (!pNode) return WBEM_E_OUT_OF_MEMORY; if(!pNode->SetEmbeddingInfo(&leftInfo)) return WBEM_E_OUT_OF_MEMORY; if(!pNode->SetPropertyInfo(wszLeftPropName, lLeftPropHandle)) return WBEM_E_OUT_OF_MEMORY; ((CTwoPropNode*)pNode)->SetRightEmbeddingInfo(&rightInfo); ((CTwoPropNode*)pNode)->SetRightPropertyInfo(wszRightPropName, lRightPropHandle); hres = pNode->SetOperator(Token.nOperator); if(FAILED(hres)) return hres; *ppRes = pNode; return WBEM_S_NO_ERROR; } HRESULT CEvalTree::BuildFromToken(CContextMetaData* pNamespace, CImplicationList& Implications, QL_LEVEL_1_TOKEN& Token, CEvalNode** ppRes) { HRESULT hres; *ppRes = NULL; if (Token.m_bPropComp) { hres = BuildTwoPropFromToken(pNamespace, Implications, Token, ppRes); if(hres == WBEMESS_E_REGISTRATION_TOO_BROAD || hres == WBEM_E_INVALID_PROPERTY) { // // Not enough information to use efficient evaluation // CDumbNode* pNode = NULL; try { pNode = new CDumbNode(Token); if(pNode == NULL) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } *ppRes = pNode; return WBEM_S_NO_ERROR; } else return hres; } // // Retrieve event class definition // _IWmiObject* pEventClass; hres = Implications.FindClassForProp(NULL, 0, &pEventClass); if(FAILED(hres)) return hres; if(pEventClass == NULL) return WBEM_E_INVALID_QUERY; CReleaseMe rm1((IWbemClassObject*)pEventClass); if(Token.nOperator == QL1_OPERATOR_ISA) { // // Inheritance node are rarely applicable in Nova --- we have no way // of telling *which definition* of the class is being referenced. Thus, // we only construct an inheritance node in one case --- where the // embedded object in question is a property of an instance operation // event. These we trust. // if(pEventClass->InheritsFrom(L"__InstanceOperationEvent") == S_OK) { // // OK, can use an inheritance node // if(V_VT(&Token.vConstValue) != VT_BSTR) return WBEM_E_INVALID_QUERY; BSTR strClassName = V_BSTR(&Token.vConstValue); CInheritanceNode* pNode = NULL; try { pNode = new CInheritanceNode; if (!pNode) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } CDeleteMe deleteMe(pNode); CEvalNode* pTrue = CValueNode::GetStandardTrue(); if(pTrue == NULL) return WBEM_E_OUT_OF_MEMORY; hres = pNode->AddClass(pNamespace, strClassName, pTrue); if(FAILED(hres)) return hres; if(!pNode->SetPropertyInfo(pNamespace, Token.PropertyName)) return WBEM_E_OUT_OF_MEMORY; hres = pNode->Compile(pNamespace, Implications); if(FAILED(hres)) return hres; // Record the fact that TRUE branch is being taken // =============================================== pNode->RecordBranch(pNamespace, Implications, 1); // in the event that we made it this far, // we no longer WANT to delete node deleteMe = NULL; *ppRes = pNode; return WBEM_S_NO_ERROR; } else { // // May not use an inheritance node --- use a dumb one instead // CDumbNode* pNode = NULL; try { pNode = new CDumbNode(Token); if(pNode == NULL) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } hres = pNode->Validate(pEventClass); if(FAILED(hres)) { delete pNode; return hres; } *ppRes = pNode; return WBEM_S_NO_ERROR; } } else { // // Attempt to compile the embedding portion. This will only succeed if // the rest of the query implies enough information for us to know // exactly what class the embedded object is // CEmbeddingInfo Info; if(!Info.SetPropertyNameButLast(Token.PropertyName)) return WBEM_E_OUT_OF_MEMORY; _IWmiObject* pClass; hres = Info.Compile(pNamespace, Implications, &pClass); if(hres == WBEMESS_E_REGISTRATION_TOO_BROAD || hres == WBEM_E_INVALID_PROPERTY || // invalid or not yet known? pClass == NULL) { // // Not enough information --- have to use the dumb node // CDumbNode* pNode = NULL; try { pNode = new CDumbNode(Token); if(pNode == NULL) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } hres = pNode->Validate(pEventClass); if(FAILED(hres)) { delete pNode; return hres; } *ppRes = pNode; return WBEM_S_NO_ERROR; } if(FAILED(hres)) return hres; // // We do know the class definition. Check if this is a system property, // though, in which case we still have toi use a dumb node // LPCWSTR wszPropName = Token.PropertyName.GetStringAt( Token.PropertyName.GetNumElements() - 1); if(wszPropName == NULL) return WBEM_E_OUT_OF_MEMORY; if(wszPropName[0] == '_') { CDumbNode* pNode = NULL; try { pNode = new CDumbNode(Token); if(pNode == NULL) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } hres = pNode->Validate(pEventClass); if(FAILED(hres)) { delete pNode; return hres; } *ppRes = pNode; return WBEM_S_NO_ERROR; } // Get the property type and handle // ================================ CIMTYPE ct; long lPropHandle; hres = pClass->GetPropertyHandleEx(wszPropName, 0L, &ct, &lPropHandle); pClass->Release(); if(FAILED(hres)) return hres; if(((ct & CIM_FLAG_ARRAY) != 0) || (ct == CIM_OBJECT)) return WBEM_E_NOT_SUPPORTED; // Coerce the constant to the right type // ===================================== VARIANT v; VariantInit(&v); CClearMe cm(&v); if(V_VT(&Token.vConstValue) != VT_NULL) { hres = ChangeVariantToCIMTYPE(&v, &Token.vConstValue, ct); if(FAILED(hres)) return hres; } else { V_VT(&v) = VT_NULL; } // // Create the right node // CPropertyNode* pNode = NULL; try { if ( Token.nOperator != QL1_OPERATOR_LIKE && Token.nOperator != QL1_OPERATOR_UNLIKE ) { switch(ct) { case CIM_SINT8: pNode = new CScalarPropNode; break; case CIM_UINT8: pNode = new CScalarPropNode; break; case CIM_SINT16: pNode = new CScalarPropNode; break; case CIM_UINT16: case CIM_CHAR16: pNode = new CScalarPropNode; break; case CIM_SINT32: pNode = new CScalarPropNode; break; case CIM_UINT32: pNode = new CScalarPropNode; break; case CIM_SINT64: pNode = new CScalarPropNode<__int64>; break; case CIM_UINT64: pNode = new CScalarPropNode; break; case CIM_REAL32: pNode = new CScalarPropNode; break; case CIM_REAL64: pNode = new CScalarPropNode; break; case CIM_BOOLEAN: pNode = new CScalarPropNode; break; case CIM_STRING: pNode = new CStringPropNode; break; case CIM_DATETIME: case CIM_REFERENCE: { CDumbNode* pDumb = new CDumbNode(Token); if(pDumb == NULL) return WBEM_E_OUT_OF_MEMORY; hres = pDumb->Validate(pEventClass); if(FAILED(hres)) { delete pDumb; return hres; } else { *ppRes = pDumb; return WBEM_S_NO_ERROR; } } default: return WBEM_E_CRITICAL_ERROR; } } else { if ( V_VT(&v) != VT_BSTR ) return WBEM_E_INVALID_QUERY; pNode = new CLikeStringPropNode; } } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } if (!pNode) return WBEM_E_OUT_OF_MEMORY; if(!pNode->SetEmbeddingInfo(&Info)) { delete pNode; return WBEM_E_OUT_OF_MEMORY; } if(!pNode->SetPropertyInfo(wszPropName, lPropHandle)) { delete pNode; return WBEM_E_OUT_OF_MEMORY; } if(V_VT(&v) == VT_NULL) { pNode->SetNullTest(Token.nOperator); } else { // // Check if the operator makes sense for the type // if(ct == CIM_BOOLEAN && (Token.nOperator != QL1_OPERATOR_EQUALS && Token.nOperator != QL1_OPERATOR_NOTEQUALS)) { // No < > for booleans return WBEM_E_INVALID_QUERY; } hres = pNode->SetOperator(Token.nOperator); if(FAILED(hres)) return hres; hres = pNode->SetTest(v); if(FAILED(hres)) return hres; } *ppRes = pNode; return WBEM_S_NO_ERROR; } } HRESULT CEvalTree::Combine(CEvalNode* pArg1, CEvalNode* pArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& Implications, bool bDeleteArg1, bool bDeleteArg2, CEvalNode** ppRes) { HRESULT hres; try { // // Apply the extra implications of the nodes being combined // CImplicationList* pArg1List = NULL; if(pArg1 && pArg1->GetExtraImplications()) { pArg1List = new CImplicationList(*pArg1->GetExtraImplications(), false); if(pArg1List == NULL) return WBEM_E_OUT_OF_MEMORY; } CDeleteMe dm1(pArg1List); CImplicationList* pArg2List = NULL; if(pArg2 && pArg2->GetExtraImplications()) { pArg2List = new CImplicationList(*pArg2->GetExtraImplications(), false); if(pArg2List == NULL) return WBEM_E_OUT_OF_MEMORY; } CDeleteMe dm2(pArg2List); if(pArg1List || pArg2List) { CImplicationList TheseImplications(Implications); if(pArg1List) { hres = TheseImplications.MergeIn(pArg1List); if(FAILED(hres)) return hres; } if(pArg2List) { hres = TheseImplications.MergeIn(pArg2List); if(FAILED(hres)) return hres; } // // Call inner combine to do everything but the implications // hres = InnerCombine(pArg1, pArg2, nOp, pNamespace, TheseImplications, bDeleteArg1, bDeleteArg2, ppRes); } else { // // Call inner combine to do everything but the implications // hres = InnerCombine(pArg1, pArg2, nOp, pNamespace, Implications, bDeleteArg1, bDeleteArg2, ppRes); } if(FAILED(hres)) return hres; // // The implication of the combined node is the combination of the // individual node implications. It does not matter what the operation // is: by the time we have arrived here, we have arrived to these // respective // points in the individual trees, so the implications have commenced. // OK, I am convinced :-) // if(*ppRes) { CImplicationList* pResultList = NULL; if(pArg1List || pArg2List) { // // There is actually some implication info in one of them --- // merge them // if(pArg1List == NULL) { pResultList = new CImplicationList(*pArg2List, false); if(pResultList == NULL) return WBEM_E_OUT_OF_MEMORY; } else { pResultList = new CImplicationList(*pArg1List, false); if(pResultList == NULL) return WBEM_E_OUT_OF_MEMORY; if(pArg2List != NULL) { hres = pResultList->MergeIn(pArg2List); if(FAILED(hres)) { delete pResultList; return hres; } } } } return (*ppRes)->SetExtraImplications(pResultList); // acquires } else return S_OK; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } } HRESULT CEvalTree::InnerCombine(CEvalNode* pArg1, CEvalNode* pArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& Implications, bool bDeleteArg1, bool bDeleteArg2, CEvalNode** ppRes) { HRESULT hres; *ppRes = NULL; if ((pArg1 == NULL) && (pArg2 == NULL)) return WBEM_S_NO_ERROR; if(CEvalNode::IsInvalid(pArg1) || CEvalNode::IsInvalid(pArg2)) { // // Invalid branches cannot be taken, so the result is invalid // *ppRes = CValueNode::GetStandardInvalid(); if(bDeleteArg1) delete pArg1; if(bDeleteArg2) delete pArg2; return S_OK; } int arg1Type = CEvalNode::GetType(pArg1); int arg2Type = CEvalNode::GetType(pArg2); // Check if merging the nodes is called for // ======================================== if(nOp != EVAL_OP_AND && IsMergeAdvisable(pArg1, pArg2, Implications) != WBEM_S_NO_ERROR) { // Create an OR node instead // ========================= COrNode* pNew = NULL; try { pNew = new COrNode; if(pNew == NULL) return WBEM_E_OUT_OF_MEMORY; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } if(bDeleteArg1) { hres = pNew->AddBranch(pArg1); } else { CEvalNode* pClone = pArg1->Clone(); if(pClone == NULL) { delete pNew; return WBEM_E_OUT_OF_MEMORY; } hres = pNew->AddBranch(pClone); } if(FAILED(hres)) { delete pNew; return hres; } if(bDeleteArg2) { hres = pNew->AddBranch(pArg2); } else { CEvalNode* pClone = pArg2->Clone(); if(pClone == NULL) { delete pNew; return WBEM_E_OUT_OF_MEMORY; } hres = pNew->AddBranch(pClone); } if(FAILED(hres)) { delete pNew; return hres; } *ppRes = pNew; return WBEM_S_NO_ERROR; } // Delegate same-type operations to the type // ========================================= if(arg1Type == arg2Type) { if ( ((pArg1 == NULL) || (pArg2 == NULL)) // well, gosh - if we've already decided they're the same type, no reason for redundant checks... && (arg1Type == EVAL_NODE_TYPE_VALUE)) { if(nOp == EVAL_OP_AND) { // FALSE AND anything is FALSE // =========================== *ppRes = NULL; if(bDeleteArg1) delete pArg1; if(bDeleteArg2) delete pArg2; return WBEM_S_NO_ERROR; } // FALSE combined in any other way with anything is that thing // =========================================================== if (pArg1) { if (bDeleteArg1) *ppRes = pArg1; else *ppRes = pArg1->Clone(); if(*ppRes == NULL) return WBEM_E_OUT_OF_MEMORY; } else if (pArg2) { if (bDeleteArg2) *ppRes = pArg2; else *ppRes = pArg2->Clone(); if(*ppRes == NULL) return WBEM_E_OUT_OF_MEMORY; } else // can't touch this *ppRes = NULL; return WBEM_S_NO_ERROR; } else // not value nodes return pArg1->CombineWith(pArg2, nOp, pNamespace, Implications, bDeleteArg1, bDeleteArg2, ppRes); } // Check if one is an OR // ===================== if(arg1Type == EVAL_NODE_TYPE_OR) return pArg1->CombineWith(pArg2, nOp, pNamespace, Implications, bDeleteArg1, bDeleteArg2, ppRes); if(arg2Type == EVAL_NODE_TYPE_OR) return pArg2->CombineWith(pArg1, FlipEvalOp(nOp), pNamespace, Implications, bDeleteArg2, bDeleteArg1, ppRes); // One leaf, one branch // ==================== if(arg1Type == EVAL_NODE_TYPE_VALUE) { return CombineLeafWithBranch((CValueNode*)pArg1, (CBranchingNode*)pArg2, nOp, pNamespace, Implications, bDeleteArg1, bDeleteArg2, ppRes); } else // it's pArg2 { return CombineLeafWithBranch((CValueNode*)pArg2, (CBranchingNode*)pArg1, FlipEvalOp(nOp), pNamespace, Implications, bDeleteArg2, bDeleteArg1, ppRes); } } // static HRESULT CEvalTree::CombineLeafWithBranch(CValueNode* pArg1, CBranchingNode* pArg2, int nOp, CContextMetaData* pNamespace, CImplicationList& Implications, bool bDeleteArg1, bool bDeleteArg2, CEvalNode** ppRes) { HRESULT hres; if (pArg1 == NULL) { *ppRes = NULL; if(nOp == EVAL_OP_AND) { // Anding a FALSE with something --- getting a FALSE! // ================================================== if(bDeleteArg2) delete pArg2; return WBEM_S_NO_ERROR; } else { // Anything else combined with FALSE gives itself! // =============================================== // Well, this is true, but the problem is with optimizations. // Some branches in X might not be valid under the implications in // this branch of the tree, and so need to be removed. For now, I // will simply turn off this short-circuiting path. It may turn out // that there are some critical performance gains to be had by // keeping it, in which case we would need to put this back and // make an efficient pass through it, checking branches. // /* if(bDeleteArg2) *ppRes = pArg2; else *ppRes = pArg2->Clone(); return WBEM_S_NO_ERROR; */ } } else { // Try to short-circuit // ==================== hres = pArg1->TryShortCircuit(pArg2, nOp, bDeleteArg1, bDeleteArg2, ppRes); if(FAILED(hres)) return hres; // hard-failure if(hres == WBEM_S_NO_ERROR) return WBEM_S_NO_ERROR; // short-circuit succeeded } // Didn't short-circuit // ==================== return ((CBranchingNode*)pArg2)->CombineInOrderWith(pArg1, FlipEvalOp(nOp), pNamespace, Implications, bDeleteArg2, bDeleteArg1, ppRes); } HRESULT CEvalTree::Evaluate(CObjectInfo& Info, CEvalNode* pStart, CSortedArray& trueIDs) { HRESULT hres; // Loop as long as we are still seeing branching nodes // =================================================== CEvalNode* pCurrent = pStart; int nType; while((nType = CEvalNode::GetType(pCurrent)) == EVAL_NODE_TYPE_BRANCH) { hres = ((CBranchingNode*)pCurrent)->Evaluate(Info, &pCurrent); if(FAILED(hres)) return hres; } if(nType == EVAL_NODE_TYPE_OR) { hres = ((COrNode*)pCurrent)->Evaluate(Info, trueIDs); if(FAILED(hres)) return hres; } else // VALUE { if (CValueNode::AreAnyTrue((CValueNode*)pCurrent)) ((CValueNode*)pCurrent)->AddTruesTo(trueIDs); } return WBEM_S_NO_ERROR; } HRESULT CEvalTree::Evaluate(IWbemObjectAccess* pObj, CSortedArray& trueIDs) { CInCritSec ics(&m_cs); trueIDs.SetSize(0); HRESULT hres = WBEM_S_NO_ERROR; if(m_pStart != NULL) { m_ObjectInfo.SetObjectAt(0, (_IWmiObject*)pObj); hres = Evaluate(m_ObjectInfo, m_pStart, trueIDs); m_ObjectInfo.Clear(); } return hres; } HRESULT CEvalTree::Optimize(CContextMetaData* pNamespace) { CInCritSec ics(&m_cs); if(m_pStart == NULL) return WBEM_S_NO_ERROR; CEvalNode* pNew = NULL; HRESULT hres = m_pStart->Optimize(pNamespace, &pNew); if(pNew != m_pStart) { delete m_pStart; m_pStart = pNew; } if(CEvalNode::GetType(m_pStart) == EVAL_NODE_TYPE_VALUE) { if(!m_ObjectInfo.SetLength(1)) return WBEM_E_OUT_OF_MEMORY; } return hres; } HRESULT CEvalTree::CombineWith(CEvalTree& Other, CContextMetaData* pNamespace, int nOp, long lFlags) { CInCritSec ics(&m_cs); HRESULT hres; try { CImplicationList Implications(lFlags); // // Compute required object info depth. We are not set up to configure // it properly, so we'll estimate the upper bound as the sum of the // depths of the trees being merged. Except that the first object // doesn't count --- it's the event itself. Unless one of the objects // is empty --- in that case it doesn't mention the event itself, and // so we should not subtract that 1. // long lRequiredDepth = m_ObjectInfo.GetLength() + Other.m_ObjectInfo.GetLength(); if(m_ObjectInfo.GetLength() > 0 && Other.m_ObjectInfo.GetLength() > 0) lRequiredDepth--; // // Combine our Start node with the new tree's. Ours will be deleted in // the process. // CEvalNode* pNew; hres = CEvalTree::Combine(m_pStart, Other.m_pStart, nOp, pNamespace, Implications, true, // delete ours false, // don't touch theirs &pNew); if(FAILED(hres)) { m_pStart = NULL; return hres; } m_pStart = pNew; if(!m_ObjectInfo.SetLength(lRequiredDepth)) return WBEM_E_OUT_OF_MEMORY; if(nOp == EVAL_OP_COMBINE || nOp == EVAL_OP_INVERSE_COMBINE) m_nNumValues += Other.m_nNumValues; return WBEM_S_NO_ERROR; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } } HRESULT CEvalTree::IsMergeAdvisable(CEvalNode* pArg1, CEvalNode* pArg2, CImplicationList& Implications) { if(Implications.IsMergeMandatory()) return S_OK; int arg1Type = CEvalNode::GetType(pArg1); int arg2Type = CEvalNode::GetType(pArg2); // if we have ONE non-false ValueNode, and ONE Branch Node, do not merge if ( ((arg1Type == EVAL_NODE_TYPE_VALUE) && (arg2Type == EVAL_NODE_TYPE_BRANCH) && !CValueNode::IsAllFalse((CValueNode*)pArg1)) || ((arg2Type == EVAL_NODE_TYPE_VALUE) && (arg1Type == EVAL_NODE_TYPE_BRANCH) && !CValueNode::IsAllFalse((CValueNode*)pArg2)) ) return WBEM_S_FALSE; else // otherwise, if one of the nodes is not Branching, certainly yes (how hard can it be?) if(arg1Type != EVAL_NODE_TYPE_BRANCH || arg2Type != EVAL_NODE_TYPE_BRANCH) { return WBEM_S_NO_ERROR; } // They are both branching. If not about the same property, then certainly // inadvisable, since there is very little to be gained // ======================================================================== CBranchingNode* pBranching1 = (CBranchingNode*)pArg1; CBranchingNode* pBranching2 = (CBranchingNode*)pArg2; if(CBranchingNode::ComparePrecedence(pBranching1, pBranching2)) return WBEM_S_FALSE; // Check if the nodes are inheritance --- in that case we can only merge if // they have identical checks // ======================================================================== if(pBranching1->GetSubType() == EVAL_NODE_TYPE_INHERITANCE) { // So is the other one, given that precedence is identical // ======================================================= if(((CInheritanceNode*)pBranching1)->SubCompare( (CInheritanceNode*)pBranching2) != 0) { return WBEM_S_FALSE; } else { return WBEM_S_NO_ERROR; } } else if(pBranching1->GetSubType() == EVAL_NODE_TYPE_DUMB) { // // Only merge if identical // if(((CDumbNode*)pBranching1)->SubCompare( (CDumbNode*)pBranching2) != 0) { return WBEM_S_FALSE; } else { return WBEM_S_NO_ERROR; } } // Same property. TBD: better checks // ================================== return WBEM_S_NO_ERROR; } HRESULT CEvalTree::RemoveIndex(int nIndex) { CInCritSec ics(&m_cs); if(m_pStart != NULL) { CRemoveIndexPredicate P(nIndex); m_pStart->ApplyPredicate(&P); m_nNumValues--; } return S_OK; } HRESULT CEvalTree::UtilizeGuarantee(CEvalTree& Guaranteed, CContextMetaData* pNamespace) { CInCritSec ics(&m_cs); #ifdef DUMP_EVAL_TREES FILE* f; f = fopen("c:\\log", "a"); fprintf(f, "\n\nORIGINAL:\n"); Dump(f); fprintf(f, "\n\nGUARANTEE:\n"); Guaranteed.Dump(f); fflush(f); #endif #ifdef CHECK_TREES CheckTrees(); #endif // // Combine them together // // // This is a single-valued tree -- rebase it to 1 to distinguish from // the guarantee // Rebase(1); HRESULT hres = CombineWith(Guaranteed, pNamespace, EVAL_OP_COMBINE, WBEM_FLAG_MANDATORY_MERGE); if(FAILED(hres)) return hres; #ifdef DUMP_EVAL_TREES fprintf(f, "AFTER MERGE:\n"); Dump(f); fflush(f); #endif // Eliminate all nodes where Guaranteed is failing // =============================================== if(m_pStart) { CRemoveFailureAtIndexPredicate P(0); hres = m_pStart->ApplyPredicate(&P); if(FAILED(hres)) return hres; } m_nNumValues--; #ifdef CHECK_TREES CheckTrees(); #endif #ifdef DUMP_EVAL_TREES fprintf(f, "AFTER REMOVE:\n"); Dump(f); fflush(f); #endif hres = Optimize(pNamespace); if(FAILED(hres)) return hres; Rebase((QueryID)-1); #ifdef CHECK_TREES CheckTrees(); #endif #ifdef DUMP_EVAL_TREES fprintf(f, "AFTER OPTIMIZE:\n"); Dump(f); fclose(f); #endif return S_OK; } HRESULT CEvalTree::ApplyPredicate(CLeafPredicate* pPred) { CInCritSec ics(&m_cs); if(m_pStart != NULL) m_pStart->ApplyPredicate(pPred); return S_OK; } void CEvalTree::operator=(const CEvalTree& Other) { CInCritSec ics(&m_cs); delete m_pStart; m_pStart = (Other.m_pStart ? Other.m_pStart->Clone() : NULL); if(m_pStart == NULL && Other.m_pStart != NULL) throw CX_MemoryException(); m_nNumValues = Other.m_nNumValues; if(!m_ObjectInfo.SetLength(m_nNumValues)) throw CX_MemoryException(); } // renumber the QueryIDs in the leaves of the tree void CEvalTree::Rebase(QueryID newBase) { CRebasePredicate predRebase(newBase); ApplyPredicate(&predRebase); } bool CEvalTree::Clear() { CInCritSec ics(&m_cs); delete m_pStart; m_pStart = CValueNode::GetStandardFalse(); if(!m_ObjectInfo.SetLength(1)) return false; m_nNumValues = 0; return true; } void CEvalTree::Dump(FILE* f) { CEvalNode::DumpNode(f, 0, m_pStart); } #ifdef CHECK_TREES void CEvalTree::CheckNodes(CTreeChecker *pChecker) { CInCritSec ics2(&m_cs); if (m_pStart) m_pStart->CheckNode(pChecker); } #endif HRESULT CEvalTree::CreateFromConjunction(CContextMetaData* pNamespace, CImplicationList& Implications, CConjunction* pConj, CEvalNode** ppRes) { HRESULT hres; *ppRes = NULL; // Build them for all tokens and AND together // ========================================== try { CImplicationList BranchImplications(Implications); for(int i = 0; i < pConj->GetNumTokens(); i++) { CEvalNode* pNew = NULL; hres = CEvalTree::BuildFromToken(pNamespace, BranchImplications, *pConj->GetTokenAt(i), &pNew); if(FAILED(hres)) { delete *ppRes; return hres; } if(i > 0) { CEvalNode* pOld = *ppRes; hres = CEvalTree::Combine(pOld, pNew, EVAL_OP_AND, pNamespace, Implications, true, true, ppRes); // delete both if ( FAILED( hres ) ) { delete pOld; return hres; } } else { *ppRes = pNew; } } return WBEM_S_NO_ERROR; } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } } HRESULT CEvalTree::CreateFromDNF(CContextMetaData* pNamespace, CImplicationList& Implications, CDNFExpression* pDNF, CEvalNode** ppRes) { HRESULT hres; *ppRes = NULL; // Check if there is only one conjunction to talk about // ==================================================== if(pDNF->GetNumTerms() == 1) { // Just build that one // =================== return CreateFromConjunction(pNamespace, Implications, pDNF->GetTermAt(0), ppRes); } // Build them for all conjunctions and OR together // =============================================== CEvalNode* pRes = NULL; for(int i = 0; i < pDNF->GetNumTerms(); i++) { CEvalNode* pNew; hres = CreateFromConjunction(pNamespace, Implications, pDNF->GetTermAt(i), &pNew); if(FAILED(hres)) { delete pRes; return hres; } if(pRes == NULL) { pRes = pNew; } else { CEvalNode* pNewRes = NULL; hres = CEvalTree::Combine(pRes, pNew, EVAL_OP_COMBINE, pNamespace, Implications, true, true, &pNewRes); if(FAILED(hres)) { delete pRes; delete pNew; return hres; } pRes = pNewRes; } } *ppRes = pRes; return WBEM_S_NO_ERROR; } HRESULT CEvalTree::CreateProjection(CEvalTree& Old, CContextMetaData* pMeta, CProjectionFilter* pFilter, EProjectionType eType, bool bDeleteOld) { delete m_pStart; m_pStart = NULL; try { CImplicationList Implications; return CEvalTree::Project(pMeta, Implications, Old.m_pStart, pFilter, eType, bDeleteOld, &m_pStart); } catch (CX_MemoryException) { return WBEM_E_OUT_OF_MEMORY; } } HRESULT CEvalTree::Project(CContextMetaData* pMeta, CImplicationList& Implications, CEvalNode* pOldNode, CProjectionFilter* pFilter, EProjectionType eType, bool bDeleteOld, CEvalNode** ppNewNode) { if(pOldNode == NULL) { *ppNewNode = NULL; return WBEM_S_NO_ERROR; } return pOldNode->Project(pMeta, Implications, pFilter, eType, bDeleteOld, ppNewNode); } CPropertyProjectionFilter::CPropertyProjectionFilter() { m_papProperties = new CUniquePointerArray; if(m_papProperties == NULL) throw CX_MemoryException(); } CPropertyProjectionFilter::~CPropertyProjectionFilter() { delete m_papProperties; } bool CPropertyProjectionFilter::IsInSet(CEvalNode* pNode) { if(CEvalNode::GetType(pNode) != EVAL_NODE_TYPE_BRANCH) return false; CBranchingNode* pBranchingNode = (CBranchingNode*)pNode; CPropertyName* pEmbeddedObjName = pBranchingNode->GetEmbeddedObjPropName(); CPropertyName ThisName; if(pEmbeddedObjName) ThisName = *pEmbeddedObjName; int nSubType = pBranchingNode->GetSubType(); if(nSubType == EVAL_NODE_TYPE_SCALAR || nSubType == EVAL_NODE_TYPE_STRING) { // // Derived from CPropertyNode --- get its property name // ThisName.AddElement( ((CPropertyNode*)pBranchingNode)->GetPropertyName()); } else if(nSubType == EVAL_NODE_TYPE_INHERITANCE) { // No extra name parts } else { // // Two-prop, perhaps. Just say no // return false; } // // Search for the name in our list // for(int i = 0; i < m_papProperties->GetSize(); i++) { if(*(m_papProperties->GetAt(i)) == ThisName) return true; } return false; } bool CPropertyProjectionFilter::AddProperty(const CPropertyName& Prop) { CPropertyName* pProp = NULL; try { pProp = new CPropertyName(Prop); if(pProp == NULL) return false; } catch (CX_MemoryException) { return false; } if(m_papProperties->Add(pProp) < 0) return false; return true; }