1676 lines
47 KiB
C++
1676 lines
47 KiB
C++
/*++
|
|
|
|
Copyright (c) 1995 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
qryparse.cxx
|
|
|
|
Abstract:
|
|
|
|
Author:
|
|
|
|
Felix Wong [t-FelixW] 05-Nov-1996
|
|
|
|
++*/
|
|
#include "nds.hxx"
|
|
#pragma hdrstop
|
|
|
|
//#define DEBUG_DUMPSTACK
|
|
//#define DEBUG_DUMPRULE
|
|
|
|
#if (defined(DEBUG_DUMPSTACK) || defined (DEBUG_DUMPRULE))
|
|
#include "stdio.h"
|
|
#endif
|
|
|
|
#define MAPHEXTODIGIT(x) ( x >= '0' && x <= '9' ? (x-'0') : \
|
|
x >= 'A' && x <= 'F' ? (x-'A'+10) : \
|
|
x >= 'a' && x <= 'f' ? (x-'a'+10) : 0 )
|
|
|
|
|
|
nuint16 g_MapTokenToNdsToken[] = {
|
|
|
|
FTOK_END, // TOKEN_ERROR
|
|
FTOK_LPAREN, // TOKEN_LPARAN
|
|
FTOK_RPAREN, // TOKEN_RPARAN
|
|
FTOK_OR, // TOKEN_OR
|
|
FTOK_AND, // TOKEN_AND
|
|
FTOK_NOT, // TOKEN_NOT
|
|
FTOK_APPROX, // TOKEN_APPROX_EQ
|
|
FTOK_EQ, // TOKEN_EQ
|
|
FTOK_LE, // TOKEN_LE
|
|
FTOK_GE, // TOKEN_GE
|
|
FTOK_PRESENT, // TOKEN_PRESENT
|
|
FTOK_ANAME, // TOKEN_ATTRTYPE
|
|
FTOK_AVAL, // TOKEN_ATTRVAL
|
|
FTOK_END // TOKEN_ENDINPUT
|
|
};
|
|
|
|
// Action Table
|
|
typedef struct _action{
|
|
DWORD type;
|
|
DWORD dwState;
|
|
}action;
|
|
|
|
// Rule Table
|
|
typedef struct _rule{
|
|
DWORD dwNumber;
|
|
DWORD dwA;
|
|
}rule;
|
|
|
|
enum types {
|
|
N,
|
|
S,
|
|
R,
|
|
A
|
|
};
|
|
|
|
#define X 99
|
|
|
|
action g_action[28][14] = {
|
|
// ERROR ,LPARAN,RPARAN,OR, AND, NOT, APPROX,EQ, LE, GE, PRESNT,ATYPE, VAL, END,
|
|
/*00*/ { {N,X },{S,2 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*01*/ { {N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{A,X } },
|
|
/*02*/ { {N,X },{N,X },{N,X },{S,12},{S,11},{S,13},{N,X },{N,X },{N,X },{N,X },{N,X },{S,14},{N,X },{N,X } },
|
|
/*03*/ { {N,X },{N,X },{R,2 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*04*/ { {N,X },{N,X },{R,3 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*05*/ { {N,X },{N,X },{R,4 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*06*/ { {N,X },{N,X },{R,5 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*07*/ { {N,X },{N,X },{S,15},{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*08*/ { {N,X },{N,X },{R,11},{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*09*/ { {N,X },{N,X },{R,12},{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*10*/ { {N,X },{N,X },{R,13},{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*11*/ { {N,X },{S,2 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*12*/ { {N,X },{S,2 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*13*/ { {N,X },{S,2 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*14*/ { {N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{S,20},{S,26},{S,22},{S,21},{S,23},{N,X },{N,X },{N,X } },
|
|
/*15*/ { {N,X },{R,1 },{R,1 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{R,1 } },
|
|
/*16*/ { {N,X },{N,X },{R,6 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*17*/ { {N,X },{S,2 },{R,9 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*18*/ { {N,X },{N,X },{R,7 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*19*/ { {N,X },{N,X },{R,8 },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*20*/ { {N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{R,15},{N,X } },
|
|
/*21*/ { {N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{R,16},{N,X } },
|
|
/*22*/ { {N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{R,17},{N,X } },
|
|
/*23*/ { {N,X },{N,X },{R,18},{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*24*/ { {N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{S,25},{N,X } },
|
|
/*25*/ { {N,X },{N,X },{R,14},{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } },
|
|
/*26*/ { {N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{R,19},{N,X } },
|
|
/*27*/ { {N,X },{N,X },{R,10},{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X },{N,X } }
|
|
};
|
|
|
|
enum non_terminals {
|
|
NONTERM_F,
|
|
NONTERM_FC,
|
|
NONTERM_AND,
|
|
NONTERM_OR,
|
|
NONTERM_NOT,
|
|
NONTERM_FL,
|
|
NONTERM_ITM,
|
|
NONTERM_SMP,
|
|
NONTERM_FT,
|
|
NONTERM_PRS
|
|
};
|
|
|
|
|
|
rule g_rule[] = {
|
|
// 1)No. of non-terminals and terminals on the right hand side
|
|
// 2)The Parent
|
|
/*00*/ {0, 0, },
|
|
/*01*/ {3, NONTERM_F, },
|
|
/*02*/ {1, NONTERM_FC, },
|
|
/*03*/ {1, NONTERM_FC, },
|
|
/*04*/ {1, NONTERM_FC, },
|
|
/*05*/ {1, NONTERM_FC, },
|
|
/*06*/ {2, NONTERM_AND, },
|
|
/*07*/ {2, NONTERM_OR, },
|
|
/*08*/ {2, NONTERM_NOT, },
|
|
/*09*/ {1, NONTERM_FL, },
|
|
/*10*/ {2, NONTERM_FL, },
|
|
/*11*/ {1, NONTERM_ITM, },
|
|
/*12*/ {1, NONTERM_ITM, },
|
|
/*13*/ {1, NONTERM_ITM, },
|
|
/*14*/ {3, NONTERM_SMP, },
|
|
/*15*/ {1, NONTERM_FT, },
|
|
/*16*/ {1, NONTERM_FT, },
|
|
/*17*/ {1, NONTERM_FT, },
|
|
/*18*/ {2, NONTERM_PRS, },
|
|
/*19*/ {1, NONTERM_FT, }
|
|
};
|
|
|
|
#ifdef DEBUG_DUMPRULE
|
|
LPWSTR g_rgszRule[] = {
|
|
/*00*/ L"",
|
|
/*01*/ L"F->(FC)",
|
|
/*02*/ L"FC->AND",
|
|
/*03*/ L"FC->OR",
|
|
/*04*/ L"FC->NOT",
|
|
/*05*/ L"FC->ITM",
|
|
/*06*/ L"AND->&FL",
|
|
/*07*/ L"OR->|FL",
|
|
/*08*/ L"NOT->!F",
|
|
/*09*/ L"FL->F",
|
|
/*10*/ L"FL->F FL",
|
|
/*11*/ L"ITM->SMP",
|
|
/*12*/ L"ITM->PRS",
|
|
/*13*/ L"ITM->STR",
|
|
/*14*/ L"SMP->ATR FT VAL",
|
|
/*15*/ L"FT->~=",
|
|
/*16*/ L"FT->>=",
|
|
/*17*/ L"FT-><=",
|
|
/*18*/ L"PRS->ATR=*",
|
|
/*19*/ L"FT->="
|
|
};
|
|
#endif
|
|
|
|
DWORD g_goto[28][10] = {
|
|
// F, FC, AND, OR, NOT, FL, ITM, SMP, FT, PRS,
|
|
/*00*/ {1, X, X, X, X, X, X, X, X, X },
|
|
/*01*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*02*/ {X, 7, 3, 4, 5, X, 6, 8, X, 9 },
|
|
/*03*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*04*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*05*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*06*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*07*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*08*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*09*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*10*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*11*/ {17, X, X, X, X, 16, X, X, X, X },
|
|
/*12*/ {17, X, X, X, X, 18, X, X, X, X },
|
|
/*13*/ {19, X, X, X, X, X, X, X, X, X },
|
|
/*14*/ {X, X, X, X, X, X, X, X, 24, X },
|
|
/*15*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*16*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*17*/ {17, X, X, X, X, 27, X, X, X, X },
|
|
/*18*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*19*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*20*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*21*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*22*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*23*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*24*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*25*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*26*/ {X, X, X, X, X, X, X, X, X, X },
|
|
/*27*/ {X, X, X, X, X, X, X, X, X, X }
|
|
};
|
|
|
|
HRESULT MapTokenToType(
|
|
DWORD dwToken,
|
|
DWORD *pdwType
|
|
)
|
|
{
|
|
DWORD dwType;
|
|
switch(dwToken) {
|
|
case TOKEN_EQ:
|
|
dwType = QUERY_EQUAL;
|
|
break;
|
|
case TOKEN_LE:
|
|
dwType = QUERY_LE;
|
|
break;
|
|
case TOKEN_GE:
|
|
dwType = QUERY_GE;
|
|
break;
|
|
case TOKEN_APPROX_EQ:
|
|
dwType = QUERY_APPROX;
|
|
break;
|
|
default:
|
|
return (E_ADS_INVALID_FILTER);
|
|
}
|
|
*pdwType = dwType;
|
|
return (S_OK);
|
|
}
|
|
|
|
HRESULT Parse(
|
|
LPWSTR szQuery,
|
|
CQueryNode **ppNode,
|
|
CAttrList **ppAttrList
|
|
)
|
|
{
|
|
CStack Stack;
|
|
CQryLexer Query(szQuery);
|
|
LPWSTR lexeme;
|
|
DWORD dwToken;
|
|
DWORD dwState;
|
|
HRESULT hr = E_ADS_INVALID_FILTER;
|
|
|
|
CAttrList* pAttrList = new CAttrList;
|
|
if (!pAttrList)
|
|
return E_OUTOFMEMORY;
|
|
|
|
CSyntaxNode *pSynNode = NULL;
|
|
CQueryNode *pNode1 = NULL;
|
|
CQueryNode *pNode2 = NULL;
|
|
CQueryNode *pNode3 = NULL;
|
|
|
|
// Push in State 0
|
|
pSynNode = new CSyntaxNode;
|
|
Stack.Push(pSynNode);
|
|
pSynNode = NULL;
|
|
|
|
#ifdef DEBUG_DUMPSTACK
|
|
Stack.Dump();
|
|
#endif
|
|
|
|
while (1) {
|
|
// Getting information for this iteration, dwToken and dwState
|
|
hr = Query.GetCurrentToken(
|
|
&lexeme,
|
|
&dwToken
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = Stack.Current(&pSynNode);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
dwState = pSynNode->_dwState;
|
|
pSynNode = NULL;
|
|
|
|
// Analysing and processing the data
|
|
if (g_action[dwState][dwToken].type == S) {
|
|
pSynNode = new CSyntaxNode;
|
|
pSynNode->_dwState = g_action[dwState][dwToken].dwState;
|
|
pSynNode->_dwToken = dwToken;
|
|
switch (dwToken) {
|
|
case TOKEN_ATTRTYPE:
|
|
{
|
|
hr = pAttrList->Add(lexeme);
|
|
BAIL_ON_FAILURE(hr);
|
|
}
|
|
case TOKEN_ATTRVAL:
|
|
// both TOKEN_ATTRTYPE and TOKEN_ATTRVAL will get here
|
|
{
|
|
LPWSTR szValue = AllocADsStr(lexeme);
|
|
if (!szValue) {
|
|
hr = E_OUTOFMEMORY;
|
|
goto error;
|
|
}
|
|
pSynNode->SetNode(szValue);
|
|
break;
|
|
}
|
|
}
|
|
hr = Stack.Push(pSynNode);
|
|
BAIL_ON_FAILURE(hr);
|
|
pSynNode = NULL;
|
|
|
|
hr = Query.GetNextToken(
|
|
&lexeme,
|
|
&dwToken
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
#ifdef DEBUG_DUMPSTACK
|
|
Stack.Dump();
|
|
#endif
|
|
}
|
|
else if (g_action[dwState][dwToken].type == R) {
|
|
DWORD dwRule = g_action[dwState][dwToken].dwState;
|
|
DWORD dwNumber = g_rule[dwRule].dwNumber;
|
|
#ifdef DEBUG_DUMPRULE
|
|
wprintf(L"%s\n",g_rgszRule[dwRule]);
|
|
#endif
|
|
pSynNode = new CSyntaxNode;
|
|
CSyntaxNode *pSynNodeRed;
|
|
switch (dwRule) {
|
|
case 1: // Reduction of Basic Filter rule
|
|
{
|
|
// Getting the middle node
|
|
|
|
hr = Stack.Pop();
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
pSynNode->SetNode(
|
|
pSynNodeRed->_pNode
|
|
);
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
hr = Stack.Pop();
|
|
BAIL_ON_FAILURE(hr);
|
|
break;
|
|
}
|
|
case 18: // Reduction of PRESENT rule
|
|
{
|
|
// Getting second node
|
|
LPWSTR szType;
|
|
|
|
hr = Stack.Pop();
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
szType = pSynNodeRed->_szValue;
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
hr = MakeLeaf(
|
|
szType,
|
|
&pNode1
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = MakeNode(
|
|
QUERY_PRESENT,
|
|
pNode1,
|
|
NULL,
|
|
&pNode2
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
pNode1 = NULL;
|
|
|
|
pSynNode->SetNode(
|
|
pNode2
|
|
);
|
|
pNode2 = NULL;
|
|
break;
|
|
}
|
|
case 14: // Reduction of SMP rule
|
|
{
|
|
LPWSTR szType;
|
|
LPWSTR szValue;
|
|
DWORD dwType;
|
|
DWORD dwToken;
|
|
|
|
hr = Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
szValue = pSynNodeRed->_szValue;
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
dwToken = (DWORD)pSynNodeRed->_dwFilterType;
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
hr = Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
szType = pSynNodeRed->_szValue;
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
hr = MakeLeaf(
|
|
szType,
|
|
&pNode1
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = MakeLeaf(
|
|
szValue,
|
|
&pNode2
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = MapTokenToType(
|
|
dwToken,
|
|
&dwType
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = MakeNode(
|
|
dwType,
|
|
pNode1,
|
|
pNode2,
|
|
&pNode3
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
pSynNode->SetNode(
|
|
pNode3
|
|
);
|
|
pNode1 = NULL;
|
|
pNode2 = NULL;
|
|
pNode3 = NULL;
|
|
|
|
break;
|
|
}
|
|
case 6: // Reduction of AND, OR rules
|
|
case 7:
|
|
{
|
|
DWORD dwType;
|
|
|
|
Stack.Pop(&pSynNodeRed);
|
|
pSynNode->SetNode(
|
|
pSynNodeRed->_pNode
|
|
);
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
Stack.Pop();
|
|
|
|
// Adding in the type information
|
|
if (dwRule == 6)
|
|
dwType = QUERY_AND;
|
|
else
|
|
dwType = QUERY_OR;
|
|
|
|
pSynNode->_pNode->_dwType = dwType;
|
|
break;
|
|
}
|
|
case 10: // Reduction of FL rule
|
|
{
|
|
DWORD dwType;
|
|
|
|
hr = Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
pNode2 = pSynNodeRed->_pNode;
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
hr = Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
pNode1 = pSynNodeRed->_pNode;
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
if (pNode2->_dwType == QUERY_UNKNOWN) {
|
|
// It's not new node, append to node1
|
|
hr = pNode2->AddChild(pNode1);
|
|
BAIL_ON_FAILURE(hr);
|
|
pSynNode->SetNode(
|
|
pNode2
|
|
);
|
|
pNode1 = NULL;
|
|
pNode2 = NULL;
|
|
}
|
|
else {
|
|
// New node
|
|
hr = MakeNode(
|
|
QUERY_UNKNOWN,
|
|
pNode1,
|
|
pNode2,
|
|
&pNode3
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
pSynNode->SetNode(
|
|
pNode3
|
|
);
|
|
pNode1 = NULL;
|
|
pNode2 = NULL;
|
|
pNode3 = NULL;
|
|
}
|
|
break;
|
|
}
|
|
case 9: // Reduction of FL rule
|
|
{
|
|
DWORD dwType;
|
|
|
|
hr = Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
pNode1 = pSynNodeRed->_pNode;
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
hr = MakeNode(
|
|
QUERY_UNKNOWN,
|
|
pNode1,
|
|
NULL,
|
|
&pNode3
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
pSynNode->SetNode(
|
|
pNode3
|
|
);
|
|
pNode1 = NULL;
|
|
pNode3 = NULL;
|
|
break;
|
|
}
|
|
case 8: // Reduction of NOT rule
|
|
{
|
|
Stack.Pop(&pSynNodeRed);
|
|
pNode1 = pSynNodeRed->_pNode;
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
|
|
hr = MakeNode(
|
|
QUERY_NOT,
|
|
pNode1,
|
|
NULL,
|
|
&pNode2
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
pNode1 = NULL;
|
|
hr = Stack.Pop();
|
|
BAIL_ON_FAILURE(hr);
|
|
pSynNode->SetNode(
|
|
pNode2
|
|
);
|
|
pNode2 = NULL;
|
|
break;
|
|
}
|
|
case 15: // Reduction of FT rule
|
|
case 16:
|
|
case 17:
|
|
case 19:
|
|
{
|
|
// Propagating the last entry
|
|
hr = Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
pSynNode->SetNode(
|
|
pSynNodeRed->_dwToken
|
|
);
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
// For all the other rules, we propogate the last entry
|
|
hr = Stack.Pop(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
pSynNode->SetNode(
|
|
pSynNodeRed->_pNode
|
|
);
|
|
pSynNodeRed->_dwType = SNODE_NULL;
|
|
delete pSynNodeRed;
|
|
for (DWORD i = 0;i<dwNumber-1;i++)
|
|
Stack.Pop();
|
|
}
|
|
}
|
|
hr = Stack.Current(&pSynNodeRed);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
dwState = pSynNodeRed->_dwState;
|
|
DWORD A = g_rule[dwRule].dwA;
|
|
pSynNode->_dwState = g_goto[dwState][A];
|
|
pSynNode->_dwToken = A;
|
|
hr = Stack.Push(pSynNode);
|
|
BAIL_ON_FAILURE(hr);
|
|
pSynNode = NULL;
|
|
#ifdef DEBUG_DUMPSTACK
|
|
Stack.Dump();
|
|
#endif
|
|
}
|
|
else if (g_action[dwState][dwToken].type == A){
|
|
hr = Stack.Pop(&pSynNode);
|
|
BAIL_ON_FAILURE(hr);
|
|
*ppNode = pSynNode->_pNode;
|
|
*ppAttrList = pAttrList;
|
|
pSynNode->_dwType = SNODE_NULL;
|
|
delete pSynNode;
|
|
return S_OK;
|
|
}
|
|
else {
|
|
hr = E_ADS_INVALID_FILTER;
|
|
goto error;
|
|
}
|
|
}
|
|
error:
|
|
if (pAttrList) {
|
|
delete pAttrList;
|
|
}
|
|
if (pSynNode) {
|
|
delete pSynNode;
|
|
}
|
|
if (pNode1) {
|
|
delete pNode1;
|
|
}
|
|
if (pNode2) {
|
|
delete pNode2;
|
|
}
|
|
if (pNode3) {
|
|
delete pNode3;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
CStack::CStack()
|
|
{
|
|
_dwStackIndex = 0;
|
|
}
|
|
|
|
CStack::~CStack()
|
|
{
|
|
DWORD dwIndex = _dwStackIndex;
|
|
while (dwIndex > 0) {
|
|
CSyntaxNode *pNode;
|
|
pNode = _Stack[--dwIndex];
|
|
delete pNode;
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG_DUMPSTACK
|
|
void CStack::Dump()
|
|
{
|
|
DWORD dwIndex = _dwStackIndex;
|
|
printf("Stack:\n");
|
|
while (dwIndex > 0) {
|
|
CSyntaxNode *pNode;
|
|
pNode = _Stack[--dwIndex];
|
|
printf(
|
|
"State=%5.0d, Token=%5.0d\n",
|
|
pNode->_dwState,
|
|
pNode->_dwToken
|
|
);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
HRESULT CStack::Push(CSyntaxNode* pNode)
|
|
{
|
|
if (_dwStackIndex < MAXVAL) {
|
|
_Stack[_dwStackIndex++] = pNode;
|
|
return S_OK;
|
|
}
|
|
else
|
|
return E_FAIL;
|
|
}
|
|
|
|
HRESULT CStack::Pop(CSyntaxNode** ppNode)
|
|
{
|
|
if (_dwStackIndex > 0) {
|
|
*ppNode = _Stack[--_dwStackIndex];
|
|
return S_OK;
|
|
}
|
|
else {
|
|
return E_FAIL;
|
|
}
|
|
}
|
|
|
|
HRESULT CStack::Pop()
|
|
{
|
|
if (_dwStackIndex > 0) {
|
|
CSyntaxNode *pNode;
|
|
pNode = _Stack[--_dwStackIndex];
|
|
delete pNode;
|
|
return S_OK;
|
|
}
|
|
else {
|
|
return E_FAIL;
|
|
}
|
|
}
|
|
|
|
HRESULT CStack::Current(CSyntaxNode **ppNode)
|
|
{
|
|
if (_dwStackIndex > 0) {
|
|
*ppNode = _Stack[_dwStackIndex-1];
|
|
return S_OK;
|
|
}
|
|
else {
|
|
return E_FAIL;
|
|
}
|
|
}
|
|
|
|
CAttrList::CAttrList()
|
|
{
|
|
_rgAttr = NULL;
|
|
_dwAttrCur = 0;
|
|
}
|
|
|
|
CAttrList::~CAttrList()
|
|
{
|
|
if (_rgAttr) {
|
|
for (DWORD i=0;i<_dwAttrCur;i++)
|
|
FreeADsStr(_rgAttr[i].szName);
|
|
FreeADsMem(_rgAttr);
|
|
}
|
|
}
|
|
|
|
HRESULT CAttrList::Add(LPWSTR szName)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fBinary = FALSE;
|
|
LPWSTR pszTemp = NULL;
|
|
|
|
for (DWORD i=0;i<_dwAttrCur;i++) {
|
|
if (_wcsicmp(
|
|
szName,
|
|
_rgAttr[i].szName
|
|
) == 0)
|
|
break;
|
|
}
|
|
|
|
if (i != _dwAttrCur) // does not loop till the end, entry exist already
|
|
return S_OK;
|
|
|
|
LPWSTR szAttr = AllocADsStr(szName);
|
|
if (!szAttr)
|
|
return E_OUTOFMEMORY;
|
|
|
|
//
|
|
// Check if the attribute comes with a ";binary" specifier. This means
|
|
// that the value has to be converted from the encoded form to
|
|
// octet strings
|
|
|
|
if (pszTemp = wcschr (szAttr, L';')) {
|
|
//
|
|
// Make sure binary appears at the end of the attribute name and
|
|
// immediately after ;
|
|
//
|
|
if (_wcsicmp(pszTemp+1, L"binary") == 0) {
|
|
fBinary = TRUE;
|
|
//
|
|
// Strip off the binary specifier.
|
|
//
|
|
wcstok(szAttr, L";");
|
|
}
|
|
}
|
|
|
|
if (_dwAttrCur == _dwAttrMax) {
|
|
if (!_rgAttr) {
|
|
_rgAttr = (AttrNode*)AllocADsMem(ATTRNODE_INITIAL*sizeof(AttrNode));
|
|
if (!_rgAttr) {
|
|
hr = E_OUTOFMEMORY;
|
|
goto error;
|
|
}
|
|
_dwAttrMax = ATTRNODE_INITIAL;
|
|
}
|
|
else {
|
|
_rgAttr = (AttrNode*)ReallocADsMem(
|
|
(void*)_rgAttr,
|
|
_dwAttrMax*sizeof(AttrNode),
|
|
(_dwAttrMax+ATTRNODE_INC)*sizeof(AttrNode)
|
|
);
|
|
if (!_rgAttr) {
|
|
hr = E_OUTOFMEMORY;
|
|
goto error;
|
|
}
|
|
_dwAttrMax+= ATTRNODE_INC;
|
|
}
|
|
}
|
|
_rgAttr[_dwAttrCur].szName = szAttr;
|
|
_rgAttr[_dwAttrCur].dwType = 0; //UNKNOWN at this point
|
|
_rgAttr[_dwAttrCur].fBinary = fBinary;
|
|
_dwAttrCur++;
|
|
return S_OK;
|
|
error:
|
|
if (szAttr)
|
|
FreeADsStr(szAttr);
|
|
return (hr);
|
|
}
|
|
|
|
HRESULT CAttrList::SetupType(NDS_CONTEXT_HANDLE hADsContext)
|
|
{
|
|
|
|
DWORD dwStatus;
|
|
HRESULT hr = S_OK;
|
|
DWORD dwNumberOfEntries;
|
|
DWORD dwInfoType;
|
|
LPNDS_ATTR_DEF lpAttrDefs = NULL;
|
|
DWORD i,j,k;
|
|
LPWSTR *ppszAttrs = NULL;
|
|
HANDLE hOperationData = NULL;
|
|
|
|
if (_dwAttrCur == 0) {
|
|
RRETURN(S_OK);
|
|
}
|
|
|
|
ppszAttrs = (LPWSTR *) AllocADsMem(_dwAttrCur * sizeof(LPWSTR *));
|
|
if (!ppszAttrs) {
|
|
BAIL_ON_FAILURE(E_OUTOFMEMORY);
|
|
}
|
|
|
|
for (i=0, j=0; i<_dwAttrCur; i++) {
|
|
ppszAttrs[j++] = _rgAttr[i].szName;
|
|
}
|
|
|
|
hr = ADsNdsReadAttrDef(
|
|
hADsContext,
|
|
DS_ATTR_DEFS,
|
|
ppszAttrs,
|
|
_dwAttrCur,
|
|
&hOperationData
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = ADsNdsGetAttrDefListFromBuffer(
|
|
hADsContext,
|
|
hOperationData,
|
|
&dwNumberOfEntries,
|
|
&dwInfoType,
|
|
&lpAttrDefs
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
if (dwNumberOfEntries != _dwAttrCur) {
|
|
hr = E_ADS_INVALID_FILTER;
|
|
goto error;
|
|
}
|
|
|
|
for (j = 0; j < dwNumberOfEntries ; j++ ) {
|
|
for (k = 0; k < dwNumberOfEntries; k++) {
|
|
if (_wcsicmp(
|
|
_rgAttr[k].szName,
|
|
lpAttrDefs[j].szAttributeName
|
|
) == 0) {
|
|
_rgAttr[k].dwType = lpAttrDefs[j].dwSyntaxID;
|
|
break;
|
|
}
|
|
}
|
|
if (k == dwNumberOfEntries) // cannot find entry
|
|
goto error;
|
|
}
|
|
|
|
error:
|
|
|
|
if (ppszAttrs)
|
|
FreeADsMem(ppszAttrs);
|
|
|
|
if (hOperationData)
|
|
ADsNdsFreeBuffer( hOperationData );
|
|
|
|
ADsNdsFreeAttrDefList(lpAttrDefs, dwNumberOfEntries);
|
|
|
|
RRETURN(hr);
|
|
}
|
|
|
|
HRESULT CAttrList::GetType(LPWSTR szName, DWORD *pdwType)
|
|
{
|
|
|
|
DWORD i = 0;
|
|
|
|
for (i=0;i<_dwAttrCur;i++) {
|
|
if (_wcsicmp(
|
|
szName,
|
|
_rgAttr[i].szName
|
|
) == 0)
|
|
break;
|
|
}
|
|
|
|
if (i == _dwAttrCur) // Cannot find attribute
|
|
return E_FAIL;
|
|
|
|
*pdwType = _rgAttr[i].dwType;
|
|
return S_OK;
|
|
}
|
|
|
|
CSyntaxNode::CSyntaxNode()
|
|
{
|
|
_dwType = SNODE_NULL;
|
|
_dwToken = 0;
|
|
_dwState = 0;
|
|
_pNode = 0;
|
|
}
|
|
|
|
CSyntaxNode::~CSyntaxNode()
|
|
{
|
|
switch (_dwType) {
|
|
case SNODE_SZ:
|
|
FreeADsStr(_szValue);
|
|
break;
|
|
case SNODE_NODE:
|
|
delete _pNode;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CSyntaxNode::SetNode(
|
|
CQueryNode *pNode
|
|
)
|
|
{
|
|
_pNode = pNode;
|
|
_dwType = SNODE_NODE;
|
|
}
|
|
|
|
void CSyntaxNode::SetNode(
|
|
LPWSTR szValue
|
|
)
|
|
{
|
|
_szValue = szValue;
|
|
_dwType = SNODE_SZ;
|
|
}
|
|
|
|
void CSyntaxNode::SetNode(
|
|
DWORD dwFilterType
|
|
)
|
|
{
|
|
_dwFilterType = dwFilterType;
|
|
_dwType = SNODE_FILTER;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: CQueryNode::CQueryNode
|
|
//
|
|
// Synopsis: Constructor of the CQueryNode
|
|
//
|
|
// Arguments:
|
|
//
|
|
// Returns:
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 11-12-96 Felix Wong Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
CQueryNode::CQueryNode()
|
|
{
|
|
_dwType = 0;
|
|
_szValue = NULL;
|
|
_dwQueryNode = 0;
|
|
_rgQueryNode = NULL;
|
|
_dwQueryNodeMax = 0;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: CQueryNode::SetToString
|
|
//
|
|
// Synopsis: Set the Node to be a String Node
|
|
//
|
|
// Arguments: szValue value of the string
|
|
//
|
|
// Returns:
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 11-12-96 Felix Wong Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT CQueryNode::SetToString(
|
|
LPWSTR szValue
|
|
)
|
|
{
|
|
_szValue = szValue;
|
|
/*
|
|
_szValue = AllocADsStr(szValue);
|
|
if (!_szValue) {
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
*/
|
|
_dwType = QUERY_STRING;
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: CQueryNode::~CQueryNode
|
|
//
|
|
// Synopsis: Destructor of the CQueryNode
|
|
//
|
|
// Arguments:
|
|
//
|
|
// Returns:
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 11-12-96 Felix Wong Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
CQueryNode::~CQueryNode()
|
|
{
|
|
if (_szValue)
|
|
FreeADsStr(_szValue);
|
|
if (_rgQueryNode) {
|
|
for (DWORD i=0;i<_dwQueryNode;i++) {
|
|
delete _rgQueryNode[i];
|
|
}
|
|
FreeADsMem(_rgQueryNode);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: CQueryNode::AddChild
|
|
//
|
|
// Synopsis: Add a child to the node
|
|
//
|
|
// Arguments: CQueryNode *pChild pointer to the child to be added
|
|
//
|
|
// Returns:
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 11-12-96 Felix Wong Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT CQueryNode::AddChild(CQueryNode *pChild)
|
|
{
|
|
if (_dwQueryNode == _dwQueryNodeMax) {
|
|
if (!_rgQueryNode) {
|
|
_rgQueryNode = (CQueryNode**)AllocADsMem(QUERYNODE_INITIAL*sizeof(CQueryNode*));
|
|
if (!_rgQueryNode) {
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
_dwQueryNodeMax = QUERYNODE_INITIAL;
|
|
}
|
|
else {
|
|
_rgQueryNode = (CQueryNode**)ReallocADsMem(
|
|
(void*)_rgQueryNode,
|
|
_dwQueryNodeMax*sizeof(CQueryNode*),
|
|
(_dwQueryNodeMax+QUERYNODE_INC)*sizeof(CQueryNode*)
|
|
);
|
|
if (!_rgQueryNode) {
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
_dwQueryNodeMax+= QUERYNODE_INC;
|
|
}
|
|
}
|
|
_rgQueryNode[_dwQueryNode] = pChild;
|
|
_dwQueryNode++;
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
HRESULT CQueryNode::AddToFilterBuf(
|
|
pFilter_Cursor_T pCur,
|
|
CAttrList *pAttrList
|
|
)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
nuint16 luTokenType;
|
|
DWORD dwStatus = 0;
|
|
|
|
LPWSTR szValue = NULL;
|
|
LPWSTR szAttr = NULL;
|
|
NWDSCCODE ccode;
|
|
void* pValue = NULL;
|
|
|
|
|
|
// Looking at type of operation
|
|
switch (_dwType) {
|
|
case QUERY_EQUAL:
|
|
case QUERY_LE:
|
|
case QUERY_GE:
|
|
case QUERY_APPROX:
|
|
case QUERY_PRESENT:
|
|
{
|
|
DWORD dwSyntax;
|
|
DWORD dwAttrType;
|
|
LPWSTR pszTemp = NULL;
|
|
|
|
// Getting left node
|
|
if (_rgQueryNode[0] &&
|
|
_rgQueryNode[0]->_dwType == QUERY_STRING) {
|
|
szAttr = AllocADsStr(_rgQueryNode[0]->_szValue);
|
|
if (!szAttr) {
|
|
hr = E_OUTOFMEMORY;
|
|
goto error;
|
|
}
|
|
|
|
}
|
|
else {
|
|
// No nodes available
|
|
goto error;
|
|
}
|
|
|
|
// Get syntax info of right node from attribute list
|
|
hr = pAttrList->GetType(
|
|
szAttr,
|
|
&dwAttrType
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
// Getting right node
|
|
if (_rgQueryNode[1] &&
|
|
_rgQueryNode[1]->_dwType == QUERY_STRING) {
|
|
|
|
// Format the node depending on the syntax
|
|
switch (dwAttrType) {
|
|
// WIDE STRING
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
case 4:
|
|
case 5:
|
|
case 10:
|
|
case 11:
|
|
case 20:
|
|
pValue = (void *) AllocADsStr(_rgQueryNode[1]->_szValue);
|
|
if (!pValue) {
|
|
hr = E_OUTOFMEMORY;
|
|
goto error;
|
|
}
|
|
break;
|
|
|
|
// BOOLEAN
|
|
case 7: {
|
|
Boolean_T *pBool = (Boolean_T *) AllocADsMem(sizeof(Boolean_T));
|
|
if (!pBool) {
|
|
BAIL_ON_FAILURE(hr = E_OUTOFMEMORY);
|
|
}
|
|
|
|
*pBool = (Boolean_T) _wtol(_rgQueryNode[1]->_szValue);
|
|
pValue = pBool;
|
|
break;
|
|
}
|
|
|
|
// Binary Strings
|
|
case 9: {
|
|
hr = E_ADS_CANT_CONVERT_DATATYPE;
|
|
goto error;
|
|
break;
|
|
}
|
|
|
|
// DWORD
|
|
case 8 :
|
|
case 22 :
|
|
case 27 : {
|
|
Integer_T *pInteger = (Integer_T *) AllocADsMem(sizeof(Integer_T));
|
|
if (!pInteger) {
|
|
BAIL_ON_FAILURE(hr = E_OUTOFMEMORY);
|
|
}
|
|
*pInteger = (Integer_T) _wtol(_rgQueryNode[1]->_szValue);
|
|
pValue = pInteger;
|
|
break;
|
|
}
|
|
|
|
case 6 :
|
|
case 13 :
|
|
case 14 :
|
|
case 15 :
|
|
case 16 :
|
|
case 17 :
|
|
case 18 :
|
|
case 19 :
|
|
case 23 :
|
|
case 24 :
|
|
case 25 :
|
|
case 26 :
|
|
default:
|
|
hr = E_ADS_CANT_CONVERT_DATATYPE;
|
|
goto error;
|
|
break;
|
|
}
|
|
}
|
|
|
|
hr = MapQueryToNDSType(
|
|
_dwType,
|
|
&luTokenType
|
|
);
|
|
BAIL_ON_FAILURE (hr);
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_LPAREN,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
if (_dwType == QUERY_PRESENT ) {
|
|
//
|
|
// First add the present token and then the attribute
|
|
//
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_PRESENT,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_ANAME,
|
|
szAttr,
|
|
dwAttrType
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_RPAREN,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
}
|
|
else {
|
|
|
|
// All the rest are binary operators. Add the attribute name,
|
|
// operator and then the attribute value
|
|
//
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_ANAME,
|
|
szAttr,
|
|
dwAttrType
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
luTokenType,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_AVAL,
|
|
pValue,
|
|
dwAttrType
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_RPAREN,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
}
|
|
|
|
break;
|
|
}
|
|
case QUERY_AND:
|
|
case QUERY_OR:
|
|
{
|
|
hr = MapQueryToNDSType(
|
|
_dwType,
|
|
&luTokenType
|
|
);
|
|
BAIL_ON_FAILURE (hr);
|
|
|
|
// Create first node
|
|
if (!_rgQueryNode[0])
|
|
goto error;
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_LPAREN,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
hr = _rgQueryNode[0]->AddToFilterBuf(
|
|
pCur,
|
|
pAttrList
|
|
);
|
|
BAIL_ON_FAILURE (hr);
|
|
|
|
// Go through a loop creating the rest
|
|
for (DWORD i=1;i<_dwQueryNode;i++) {
|
|
|
|
if (!_rgQueryNode[i])
|
|
goto error;
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
luTokenType,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
hr = _rgQueryNode[i]->AddToFilterBuf(
|
|
pCur,
|
|
pAttrList
|
|
);
|
|
BAIL_ON_FAILURE (hr);
|
|
|
|
}
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_RPAREN,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
break;
|
|
}
|
|
case QUERY_NOT:
|
|
{
|
|
hr = MapQueryToNDSType(
|
|
_dwType,
|
|
&luTokenType
|
|
);
|
|
BAIL_ON_FAILURE (hr);
|
|
|
|
// Create first node
|
|
if (!_rgQueryNode[0])
|
|
goto error;
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_LPAREN,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_NOT,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
hr = _rgQueryNode[0]->AddToFilterBuf(
|
|
pCur,
|
|
pAttrList
|
|
);
|
|
BAIL_ON_FAILURE (hr);
|
|
|
|
ccode = NWDSAddFilterToken(
|
|
pCur,
|
|
FTOK_RPAREN,
|
|
NULL,
|
|
0
|
|
);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
break;
|
|
}
|
|
default:
|
|
goto error;
|
|
}
|
|
|
|
RRETURN(hr);
|
|
|
|
error:
|
|
if (szAttr)
|
|
FreeADsStr(szAttr);
|
|
if (pValue)
|
|
FreeADsMem(pValue);
|
|
return hr;
|
|
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: CQueryNode::FreeFilterTokens
|
|
//
|
|
// Synopsis: Frees dynamically-allocated attributes passed to
|
|
// NWDSAddFilterToken by CQueryNode::AddToFilterBuffer
|
|
//
|
|
// Arguments: syntax NetWare syntax attribute ID
|
|
// val ptr. to memory to be freed
|
|
//
|
|
// Returns:
|
|
//
|
|
// Modifies: deallocates the memory
|
|
//
|
|
// History: 9-18-99 Matthew Rimer Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
void N_FAR N_CDECL CQueryNode::FreeFilterTokens(
|
|
nuint32 syntax,
|
|
nptr pVal
|
|
)
|
|
{
|
|
if (pVal) {
|
|
|
|
switch(syntax) {
|
|
// WIDE STRING
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
case 4:
|
|
case 5:
|
|
case 10:
|
|
case 11:
|
|
case 20:
|
|
FreeADsStr((LPWSTR) pVal);
|
|
break;
|
|
|
|
// BOOLEAN
|
|
case 7:
|
|
FreeADsMem(pVal);
|
|
break;
|
|
|
|
// DWORD
|
|
case 8:
|
|
case 22:
|
|
case 27:
|
|
FreeADsMem(pVal);
|
|
break;
|
|
|
|
// attribute name
|
|
case -1:
|
|
FreeADsStr((LPWSTR) pVal);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: CQueryNode::MapQueryToNDSType
|
|
//
|
|
// Synopsis: Maps the node type to the equivalent NDS types
|
|
//
|
|
// Arguments: dwType input type
|
|
// pdwNDSType output type
|
|
//
|
|
// Returns:
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 11-12-96 Felix Wong Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT CQueryNode::MapQueryToNDSType(
|
|
DWORD dwType,
|
|
nuint16 *pulNDSTokenType
|
|
)
|
|
{
|
|
nuint16 ulNDSTokenType;
|
|
switch(dwType) {
|
|
case QUERY_EQUAL:
|
|
ulNDSTokenType = FTOK_EQ;
|
|
break;
|
|
case QUERY_LE:
|
|
ulNDSTokenType = FTOK_LE;
|
|
break;
|
|
case QUERY_GE:
|
|
ulNDSTokenType = FTOK_GE;
|
|
break;
|
|
case QUERY_APPROX:
|
|
ulNDSTokenType = FTOK_APPROX;
|
|
break;
|
|
case QUERY_PRESENT:
|
|
ulNDSTokenType = FTOK_PRESENT;
|
|
break;
|
|
case QUERY_NOT:
|
|
ulNDSTokenType = FTOK_NOT;
|
|
break;
|
|
case QUERY_AND:
|
|
ulNDSTokenType = FTOK_AND;
|
|
break;
|
|
case QUERY_OR:
|
|
ulNDSTokenType = FTOK_OR;
|
|
break;
|
|
|
|
default:
|
|
return (E_ADS_INVALID_FILTER);
|
|
}
|
|
*pulNDSTokenType = ulNDSTokenType;
|
|
return (S_OK);
|
|
}
|
|
|
|
|
|
// Helper Functions for creating nodes using the CQueryNode Class
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: MakeNode
|
|
//
|
|
// Synopsis: Make a node with the input values
|
|
//
|
|
// Arguments: dwType type of node
|
|
// pLQueryNode pointer to left node
|
|
// pRQueryNode pointer to right node
|
|
// ppQueryNodeReturn pointer to Return Node
|
|
//
|
|
// Returns:
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 11-12-96 Felix Wong Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT MakeNode(
|
|
DWORD dwType,
|
|
CQueryNode *pLQueryNode,
|
|
CQueryNode *pRQueryNode,
|
|
CQueryNode **ppQueryNodeReturn
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
CQueryNode *pQueryNode = new CQueryNode();
|
|
if (!pQueryNode)
|
|
return E_OUTOFMEMORY;
|
|
|
|
pQueryNode->_dwType = dwType;
|
|
|
|
hr = pQueryNode->AddChild(pLQueryNode);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
if (pRQueryNode) {
|
|
pQueryNode->AddChild(pRQueryNode);
|
|
BAIL_ON_FAILURE(hr);
|
|
}
|
|
*ppQueryNodeReturn = pQueryNode;
|
|
|
|
RRETURN(hr);
|
|
|
|
error:
|
|
delete pQueryNode;
|
|
RRETURN(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: MakeLeaf
|
|
//
|
|
// Synopsis: Constructor of the CQueryNode
|
|
//
|
|
// Arguments: szValue value of the string
|
|
// ppQueryNodeReturn the return node
|
|
//
|
|
// Returns:
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 11-12-96 Felix Wong Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT MakeLeaf(
|
|
LPWSTR szValue,
|
|
CQueryNode **ppQueryNodeReturn
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
CQueryNode *pQueryNode = new CQueryNode();
|
|
if (!pQueryNode)
|
|
return E_OUTOFMEMORY;
|
|
|
|
hr = pQueryNode->SetToString(szValue);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
*ppQueryNodeReturn = pQueryNode;
|
|
RRETURN(hr);
|
|
|
|
error:
|
|
delete pQueryNode;
|
|
RRETURN(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
AdsNdsGenerateFilterBuffer(
|
|
NDS_CONTEXT_HANDLE hADsContext,
|
|
LPWSTR szSearchFilter,
|
|
NDS_BUFFER_HANDLE *phFilterBuf
|
|
)
|
|
{
|
|
NWDSContextHandle context;
|
|
|
|
HRESULT hr = E_ADS_INVALID_FILTER;
|
|
NWDSCCODE ccode;
|
|
BOOL fBufAllocated = FALSE;
|
|
DWORD i, j;
|
|
pFilter_Cursor_T pCur;
|
|
NDS_BUFFER_HANDLE hFilterBuf = NULL;
|
|
|
|
|
|
PNDS_CONTEXT pADsContext = (PNDS_CONTEXT) hADsContext;
|
|
PNDS_BUFFER_DATA pFilterBuf = (PNDS_BUFFER_DATA) hFilterBuf;
|
|
nuint16 luFilterToken;
|
|
|
|
CQueryNode *pNode = NULL;
|
|
CAttrList *pAttrList = NULL;
|
|
|
|
if (!hADsContext || !phFilterBuf) {
|
|
RRETURN (E_ADS_BAD_PARAMETER);
|
|
}
|
|
|
|
|
|
// Generate the parse tree and the attribute list
|
|
hr = Parse(
|
|
szSearchFilter,
|
|
&pNode,
|
|
&pAttrList
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
|
|
// Setup syntax information in the attribute list
|
|
hr = pAttrList->SetupType(hADsContext);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = ADsNdsCreateBuffer(
|
|
hADsContext,
|
|
DSV_SEARCH_FILTER,
|
|
&hFilterBuf
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
fBufAllocated = TRUE;
|
|
|
|
// Generate the parse tree and the attribute list
|
|
|
|
ccode = NWDSAllocFilter(&pCur);
|
|
CHECK_AND_SET_EXTENDED_ERROR(ccode, hr);
|
|
|
|
// Generate the NDS tree
|
|
hr = pNode->AddToFilterBuf(
|
|
pCur,
|
|
pAttrList
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
hr = ADsNdsPutFilter(
|
|
hADsContext,
|
|
hFilterBuf,
|
|
pCur,
|
|
CQueryNode::FreeFilterTokens
|
|
);
|
|
BAIL_ON_FAILURE(hr);
|
|
|
|
|
|
*phFilterBuf = hFilterBuf;
|
|
|
|
if (pNode)
|
|
delete pNode;
|
|
|
|
if (pAttrList)
|
|
delete pAttrList;
|
|
|
|
RRETURN(S_OK);
|
|
|
|
|
|
error:
|
|
|
|
if (fBufAllocated) {
|
|
ADsNdsFreeBuffer(hFilterBuf);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|