/*++ Copyright (c) 2000 Microsoft Corporation Module Name: rules.c Abstract: Implements the "rule" algorithm that scores script patterns and keeps them in reverse-sorted order. Author: Jim Schmidt (jimschm) 12-May-2000 Revision History: --*/ // // Includes // #include "pch.h" #include "v1p.h" #define DBG_V1 "v1" // // Strings // // None // // Constants // // None // // Macros // // None // // Types // typedef struct _TAG_RULE { PCTSTR ObjectBase; MIG_OBJECTSTRINGHANDLE ObjectPattern; POBSPARSEDPATTERN ParsedPattern; UINT Score; ACTIONGROUP ActionGroup; DWORD ActionFlags; ACTION_STRUCT ActionStruct; struct _TAG_RULE *NextRule; } RULE, *PRULE; typedef struct _TAG_CHAR_NODE { PRULE RuleList; WORD Char; WORD Flags; struct _TAG_CHAR_NODE *NextLevel; struct _TAG_CHAR_NODE *NextPeer; } CHAR_NODE, *PCHAR_NODE; typedef struct { MIG_OBJECTTYPEID ObjectTypeId; PRULE RuleList; PCHAR_NODE FirstLevel; } TYPE_RULE, *PTYPE_RULE; typedef struct { MIG_OBJECTTYPEID ObjectTypeId; TYPE_RULE TypeRule; } TYPETORULE, *PTYPETORULE; // // Globals // GROWBUFFER g_TypeToIncRule = INIT_GROWBUFFER; GROWBUFFER g_TypeToPriRule = INIT_GROWBUFFER; GROWBUFFER g_TypeToCollRule = INIT_GROWBUFFER; GROWBUFFER g_TypeRuleList = INIT_GROWBUFFER; PMHANDLE g_RulePool; // // Macro expansion list // // None // // Private function prototypes // // None // // Macro expansion definition // // None // // Code // BOOL pAddRuleToTypeRule ( IN PTYPE_RULE TypeRule, IN PRULE Rule ) { PCHAR_NODE node, currNode, prevNode; PTSTR objectBase; PCTSTR p; WORD w; BOOL found; if (Rule->ObjectBase) { currNode = TypeRule->FirstLevel; prevNode = currNode; objectBase = DuplicatePathString (Rule->ObjectBase, 0); CharLower (objectBase); p = objectBase; while (*p) { w = (WORD) _tcsnextc (p); p = _tcsinc (p); if (currNode) { if (currNode->Char == w) { if (!*p) { Rule->NextRule = currNode->RuleList; currNode->RuleList = Rule; } prevNode = currNode; currNode = currNode->NextLevel; } else { found = FALSE; while (!found && currNode->NextPeer) { if (currNode->NextPeer->Char == w) { if (!*p) { Rule->NextRule = currNode->NextPeer->RuleList; currNode->NextPeer->RuleList = Rule; } prevNode = currNode->NextPeer; currNode = prevNode->NextLevel; found = TRUE; break; } currNode = currNode->NextPeer; } if (!found) { node = PmGetMemory (g_RulePool, sizeof (CHAR_NODE)); ZeroMemory (node, sizeof (CHAR_NODE)); if (!*p) { node->RuleList = Rule; } node->Char = w; node->NextPeer = currNode->NextPeer; currNode->NextPeer = node; prevNode = node; currNode = node->NextLevel; } } } else { node = PmGetMemory (g_RulePool, sizeof (CHAR_NODE)); ZeroMemory (node, sizeof (CHAR_NODE)); if (!*p) { node->RuleList = Rule; } node->Char = w; if (prevNode) { prevNode->NextLevel = node; } else { TypeRule->FirstLevel = node; } prevNode = node; currNode = prevNode->NextLevel; } } FreePathString (objectBase); } else { Rule->NextRule = TypeRule->RuleList; TypeRule->RuleList = Rule; } return TRUE; } UINT pComputeSubScore ( IN BOOL ExcludeLastPatSeg, IN PCTSTR String, IN UINT WildcardScore, IN UINT FixedCharScore, IN UINT WackScoreBeforeWildcard, IN UINT WackScoreAfterWildcard ) { PCTSTR p; BOOL skipExtendedPattern; UINT score = 0; BOOL patternEncountered = FALSE; CHARTYPE ch; UINT state = 0; UINT delScore = 0; for (p = String ; *p ; p = _tcsinc (p)) { skipExtendedPattern = FALSE; switch (_tcsnextc (p)) { case TEXT('*'): if (state == 1) { state = 2; } if (state == 2) { delScore += WildcardScore; } score += WildcardScore; skipExtendedPattern = TRUE; patternEncountered = TRUE; break; case TEXT('?'): if (state == 1) { state = 2; } if (state == 2) { delScore += WildcardScore; } score += WildcardScore; skipExtendedPattern = TRUE; patternEncountered = TRUE; break; case TEXT('\\'): state = 1; if (patternEncountered) { score += WackScoreAfterWildcard; delScore = WackScoreAfterWildcard; } else { score += WackScoreBeforeWildcard; delScore = WackScoreBeforeWildcard; } break; case TEXT(':'): // // Special case: if just a drive spec with no wacks, // add the wack score // if (p[1] == 0) { score += WackScoreBeforeWildcard; break; } // fall through default: state = 0; delScore = 0; score += FixedCharScore; break; } if (skipExtendedPattern) { if (_tcsnextc (_tcsinc (p)) == TEXT('[')) { do { p = _tcsinc (p); ch = (CHARTYPE) _tcsnextc (p); if (ch == TEXT('^')) { p++; } } while (ch != TEXT(']')); } } } if (ExcludeLastPatSeg) { if (score > delScore) { score -= delScore; if (delScore && score > 0) { score -= 1; } } else { score = 0; } } return score; } UINT pComputeScore ( IN MIG_OBJECTSTRINGHANDLE EncodedString ) { PCTSTR node; PCTSTR leaf; UINT score = 0; IsmCreateObjectStringsFromHandle (EncodedString, &node, &leaf); if (node) { score += 1000 * pComputeSubScore (TRUE, node, 0, 10, 1000, 500); } if (leaf) { score += pComputeSubScore (FALSE, leaf, 0, 10, 0, 0); } IsmDestroyObjectString (node); IsmDestroyObjectString (leaf); return score; } PTYPE_RULE pGetListForType ( IN PGROWBUFFER TypeToRule, IN MIG_OBJECTTYPEID ObjectTypeId ) { PTYPE_RULE pos; PTYPE_RULE end; ObjectTypeId &= ~(PLATFORM_MASK); pos = (PTYPE_RULE) TypeToRule->Buf; end = (PTYPE_RULE) (TypeToRule->Buf + TypeToRule->End); while (pos < end) { if (pos->ObjectTypeId == ObjectTypeId) { return pos; } pos++; } pos = (PTYPE_RULE) GbGrow (TypeToRule, sizeof (TYPE_RULE)); ZeroMemory (pos, sizeof (TYPE_RULE)); pos->ObjectTypeId = ObjectTypeId; return pos; } VOID InitRules ( VOID ) { g_RulePool = PmCreateNamedPoolEx ("Rule Data", 32768); PmDisableTracking (g_RulePool); } VOID TerminateRules ( VOID ) { GbFree (&g_TypeToIncRule); GbFree (&g_TypeToPriRule); GbFree (&g_TypeToCollRule); GbFree (&g_TypeRuleList); PmEmptyPool (g_RulePool); PmDestroyPool (g_RulePool); INVALID_POINTER (g_RulePool); } UINT pGetActionWeight ( IN ACTIONGROUP ActionGroup ) { switch (ActionGroup) { case ACTIONGROUP_DEFAULTPRIORITY: return ACTIONWEIGHT_DEFAULTPRIORITY; case ACTIONGROUP_SPECIFICPRIORITY: return ACTIONWEIGHT_SPECIFICPRIORITY; case ACTIONGROUP_EXCLUDE: return ACTIONWEIGHT_EXCLUDE; case ACTIONGROUP_EXCLUDEEX: return ACTIONWEIGHT_EXCLUDEEX; case ACTIONGROUP_INCLUDE: return ACTIONWEIGHT_INCLUDE; case ACTIONGROUP_INCLUDEEX: return ACTIONWEIGHT_INCLUDEEX; case ACTIONGROUP_RENAME: return ACTIONWEIGHT_RENAME; case ACTIONGROUP_RENAMEEX: return ACTIONWEIGHT_RENAMEEX; case ACTIONGROUP_INCLUDERELEVANT: return ACTIONWEIGHT_INCLUDERELEVANT; case ACTIONGROUP_INCLUDERELEVANTEX: return ACTIONWEIGHT_INCLUDERELEVANTEX; case ACTIONGROUP_RENAMERELEVANT: return ACTIONWEIGHT_RENAMERELEVANT; case ACTIONGROUP_RENAMERELEVANTEX: return ACTIONWEIGHT_RENAMERELEVANTEX; case ACTIONGROUP_INCFILE: return ACTIONWEIGHT_INCFILE; case ACTIONGROUP_INCFILEEX: return ACTIONWEIGHT_INCFILEEX; case ACTIONGROUP_INCFOLDER: return ACTIONWEIGHT_INCFOLDER; case ACTIONGROUP_INCFOLDEREX: return ACTIONWEIGHT_INCFOLDEREX; case ACTIONGROUP_INCICON: return ACTIONWEIGHT_INCICON; case ACTIONGROUP_INCICONEX: return ACTIONWEIGHT_INCICONEX; case ACTIONGROUP_DELREGKEY: return ACTIONWEIGHT_DELREGKEY; default: return 0; } } BOOL AddRuleEx ( IN MIG_OBJECTTYPEID Type, IN MIG_OBJECTSTRINGHANDLE ObjectBase, IN MIG_OBJECTSTRINGHANDLE ObjectPattern, IN ACTIONGROUP ActionGroup, IN DWORD ActionFlags, IN PACTION_STRUCT ActionStruct, OPTIONAL IN RULEGROUP RuleGroup ) { PTYPE_RULE typeRule; MIG_SEGMENTS nodeSegment; MIG_SEGMENTS leafSegment; PCTSTR ourEncodedString; PCTSTR node; PCTSTR leaf; PRULE rule; PGROWBUFFER ruleStruct; switch (RuleGroup) { case RULEGROUP_NORMAL: ruleStruct = &g_TypeToIncRule; break; case RULEGROUP_PRIORITY: ruleStruct = &g_TypeToPriRule; break; case RULEGROUP_COLLPATTERN: ruleStruct = &g_TypeToCollRule; break; } typeRule = pGetListForType (ruleStruct, Type); if (!typeRule) { return FALSE; } // // Make sure both node an leaf are specified // IsmCreateObjectStringsFromHandleEx (ObjectPattern, &node, &leaf, TRUE); nodeSegment.Segment = node ? node : TEXT("*"); nodeSegment.IsPattern = TRUE; leafSegment.Segment = leaf ? leaf : TEXT("*"); leafSegment.IsPattern = TRUE; ourEncodedString = IsmCreateObjectPattern ( &nodeSegment, 1, &leafSegment, 1 ); IsmDestroyObjectString (node); IsmDestroyObjectString (leaf); // // Insert the rule based on score // rule = PmGetMemory (g_RulePool, sizeof (RULE)); ZeroMemory (rule, sizeof (RULE)); if (ObjectBase) { IsmCreateObjectStringsFromHandle (ObjectBase, &node, &leaf); if (node) { rule->ObjectBase = PmDuplicateString (g_RulePool, node); } IsmDestroyObjectString (node); IsmDestroyObjectString (leaf); } rule->ObjectPattern = PmDuplicateString (g_RulePool, ourEncodedString); rule->ParsedPattern = ObsCreateParsedPatternEx (g_RulePool, ourEncodedString, FALSE); MYASSERT (rule->ParsedPattern); if (rule->ParsedPattern) { rule->Score = pComputeScore (ourEncodedString); rule->Score = rule->Score * 10 + pGetActionWeight (ActionGroup); rule->ActionGroup = ActionGroup; rule->ActionFlags = ActionFlags; if (ActionStruct) { if (ActionStruct->ObjectBase) { rule->ActionStruct.ObjectBase = PmDuplicateString (g_RulePool, ActionStruct->ObjectBase); } if (ActionStruct->ObjectDest) { rule->ActionStruct.ObjectDest = PmDuplicateString (g_RulePool, ActionStruct->ObjectDest); } if (ActionStruct->AddnlDest) { rule->ActionStruct.AddnlDest = PmDuplicateString (g_RulePool, ActionStruct->AddnlDest); } if (ActionStruct->ObjectHint) { rule->ActionStruct.ObjectHint = PmDuplicateString (g_RulePool, ActionStruct->ObjectHint); } } pAddRuleToTypeRule (typeRule, rule); } else { IsmCreateObjectStringsFromHandleEx (ourEncodedString, &node, &leaf, TRUE); LOG ((LOG_ERROR, (PCSTR) MSG_OBJECT_SPEC_ERROR, node, leaf)); IsmDestroyObjectString (node); IsmDestroyObjectString (leaf); } IsmDestroyObjectHandle (ourEncodedString); return TRUE; } BOOL QueryRuleEx ( IN MIG_OBJECTTYPEID Type, IN MIG_OBJECTSTRINGHANDLE EncodedString, IN PCTSTR ObjectNode, OUT PACTIONGROUP ActionGroup, OUT PDWORD ActionFlags, OUT PACTION_STRUCT ActionStruct, OPTIONAL IN RULEGROUP RuleGroup ) { PTYPE_RULE typeRule; PCHAR_NODE charNode; PTSTR objectBase; PCTSTR p; WORD w; PRULE rule; PRULE foundRule; UINT maxScore; ACTIONGROUP initialAction = ACTIONGROUP_NONE; DWORD matchingFlags = 0; PGROWBUFFER ruleStruct; switch (RuleGroup) { case RULEGROUP_NORMAL: ruleStruct = &g_TypeToIncRule; break; case RULEGROUP_PRIORITY: ruleStruct = &g_TypeToPriRule; break; case RULEGROUP_COLLPATTERN: ruleStruct = &g_TypeToCollRule; break; } if (ActionStruct) { ZeroMemory (ActionStruct, sizeof (ACTION_STRUCT)); } typeRule = pGetListForType (ruleStruct, Type); if (!typeRule) { return FALSE; } // // Let's walk the structure finding all possible rules and put them in a grow buffer // Later we'll examine the rules one by one. Note that the root rules always qualify // if (ObjectNode) { objectBase = DuplicatePathString (ObjectNode, 0); CharLower (objectBase); } g_TypeRuleList.End = 0; p = objectBase; if (p) { w = (WORD) _tcsnextc (p); charNode = typeRule->FirstLevel; while (charNode && *p) { if (charNode->Char == w) { if (charNode->RuleList) { CopyMemory (GbGrow (&g_TypeRuleList, sizeof (PRULE)), &(charNode->RuleList), sizeof (PRULE)); } charNode = charNode->NextLevel; p = _tcsinc (p); w = (WORD) _tcsnextc (p); } else { charNode = charNode->NextPeer; } } } maxScore = 0; foundRule = NULL; while (g_TypeRuleList.End) { CopyMemory (&rule, &(g_TypeRuleList.Buf[g_TypeRuleList.End - sizeof (PRULE)]), sizeof (PRULE)); while (rule) { if (maxScore <= rule->Score) { if (IsmParsedPatternMatch ((MIG_PARSEDPATTERN)rule->ParsedPattern, Type, EncodedString)) { if (foundRule && (maxScore == rule->Score)) { if (initialAction == rule->ActionGroup) { matchingFlags |= rule->ActionFlags; } } else { foundRule = rule; initialAction = foundRule->ActionGroup; matchingFlags = foundRule->ActionFlags; maxScore = rule->Score; } } } rule = rule->NextRule; } g_TypeRuleList.End -= sizeof (PRULE); } rule = typeRule->RuleList; while (rule) { if (maxScore <= rule->Score) { if (IsmParsedPatternMatch ((MIG_PARSEDPATTERN)rule->ParsedPattern, Type, EncodedString)) { if (foundRule && (maxScore == rule->Score)) { if (initialAction == rule->ActionGroup) { matchingFlags |= rule->ActionFlags; } } else { foundRule = rule; initialAction = foundRule->ActionGroup; matchingFlags = foundRule->ActionFlags; maxScore = rule->Score; } } } rule = rule->NextRule; } if (foundRule) { *ActionGroup = initialAction; *ActionFlags = matchingFlags; if (ActionStruct && (!ActionStruct->ObjectBase) && (foundRule->ActionStruct.ObjectBase)) { ActionStruct->ObjectBase = foundRule->ActionStruct.ObjectBase; } if (ActionStruct && (!ActionStruct->ObjectDest) && (foundRule->ActionStruct.ObjectDest)) { ActionStruct->ObjectDest = foundRule->ActionStruct.ObjectDest; } if (ActionStruct && (!ActionStruct->AddnlDest) && (foundRule->ActionStruct.AddnlDest)) { ActionStruct->AddnlDest = foundRule->ActionStruct.AddnlDest; } if (ActionStruct && (!ActionStruct->ObjectHint) && (foundRule->ActionStruct.ObjectHint)) { ActionStruct->ObjectHint = foundRule->ActionStruct.ObjectHint; } } FreePathString (objectBase); return (foundRule != NULL); }