/*-------------------------------------------------------------------------
* drawElements Quality Program Test Executor
* ------------------------------------------
*
* Copyright 2014 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*//*!
* \file
* \brief Test case list parser.
*//*--------------------------------------------------------------------*/
#include "xeTestCaseListParser.hpp"
#include "deString.h"
using std::vector;
using std::string;
namespace xe
{
static TestCaseType getTestCaseType (const char* caseType)
{
// \todo [2012-06-11 pyry] Use hashes for speedup.
static const struct
{
const char* name;
TestCaseType caseType;
} s_caseTypeMap[] =
{
{ "SelfValidate", TESTCASETYPE_SELF_VALIDATE },
{ "Capability", TESTCASETYPE_CAPABILITY },
{ "Accuracy", TESTCASETYPE_ACCURACY },
{ "Performance", TESTCASETYPE_PERFORMANCE }
};
for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(s_caseTypeMap); ndx++)
{
if (deStringEqual(caseType, s_caseTypeMap[ndx].name))
return s_caseTypeMap[ndx].caseType;
}
XE_FAIL((string("Unknown test case type '") + caseType + "'").c_str());
}
TestCaseListParser::TestCaseListParser (void)
: m_root(DE_NULL)
{
}
TestCaseListParser::~TestCaseListParser (void)
{
}
void TestCaseListParser::clear (void)
{
m_xmlParser.clear();
m_nodeStack.clear();
m_root = DE_NULL;
}
void TestCaseListParser::init (TestGroup* rootGroup)
{
clear();
m_root = rootGroup;
}
void TestCaseListParser::parse (const deUint8* bytes, int numBytes)
{
DE_ASSERT(m_root);
m_xmlParser.feed(bytes, numBytes);
for (;;)
{
xml::Element element = m_xmlParser.getElement();
if (element == xml::ELEMENT_INCOMPLETE ||
element == xml::ELEMENT_END_OF_STRING)
break;
if (element == xml::ELEMENT_START || element == xml::ELEMENT_END)
{
bool isStart = element == xml::ELEMENT_START;
const char* elemName = m_xmlParser.getElementName();
if (deStringEqual(elemName, "TestCase"))
{
if (isStart)
{
XE_CHECK_MSG(!m_nodeStack.empty(), "<TestCase> outside of <TestCaseList>");
TestNode* parent = m_nodeStack.back();
const char* name = m_xmlParser.hasAttribute("Name") ? m_xmlParser.getAttribute("Name") : DE_NULL;
const char* description = m_xmlParser.hasAttribute("Description") ? m_xmlParser.getAttribute("Description") : DE_NULL;
const char* caseType = m_xmlParser.hasAttribute("CaseType") ? m_xmlParser.getAttribute("CaseType") : DE_NULL;
XE_CHECK_MSG(name && description && caseType, "Missing attribute in <TestCase>");
XE_CHECK_MSG(parent->getNodeType() == TESTNODETYPE_GROUP, "Only TestGroups are allowed to have child nodes");
bool isGroup = deStringEqual(caseType, "TestGroup") == DE_TRUE;
TestNode* node = isGroup ? static_cast<TestNode*>(static_cast<TestGroup*>(parent)->createGroup(name, description))
: static_cast<TestNode*>(static_cast<TestGroup*>(parent)->createCase(getTestCaseType(caseType), name, description));
m_nodeStack.push_back(node);
}
else
{
XE_CHECK_MSG(m_nodeStack.size() >= 2, "Unexpected </TestCase>");
m_nodeStack.pop_back();
}
}
else if (deStringEqual(elemName, "TestCaseList"))
{
if (isStart)
{
XE_CHECK_MSG(m_nodeStack.empty(), "Unexpected <TestCaseList>");
m_nodeStack.push_back(m_root);
}
else
{
XE_CHECK_MSG(m_nodeStack.size() == 1, "Unexpected </TestCaseList>");
m_nodeStack.pop_back();
}
}
else
XE_FAIL((string("Unexpected <") + elemName + ">").c_str());
}
else if (element != xml::ELEMENT_DATA)
DE_ASSERT(false); // \note Data elements are just ignored, they should be whitespace anyway.
m_xmlParser.advance();
}
}
} // xe