| #ifndef _TCUTESTCASE_HPP |
| #define _TCUTESTCASE_HPP |
| /*------------------------------------------------------------------------- |
| * drawElements Quality Program Tester Core |
| * ---------------------------------------- |
| * |
| * 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 Base class for a test case. |
| *//*--------------------------------------------------------------------*/ |
| |
| #include "tcuDefs.hpp" |
| #include "tcuTestContext.hpp" |
| |
| #include <string> |
| #include <vector> |
| |
| namespace tcu |
| { |
| |
| enum TestNodeType |
| { |
| NODETYPE_ROOT = 0, //!< Root for all test packages. |
| NODETYPE_PACKAGE, //!< Test case package -- same as group, but is omitted from XML dump. |
| NODETYPE_GROUP, //!< Test case container -- cannot be executed. |
| NODETYPE_SELF_VALIDATE, //!< Self-validating test case -- can be executed |
| NODETYPE_PERFORMANCE, //!< Performace test case -- can be executed |
| NODETYPE_CAPABILITY, //!< Capability score case -- can be executed |
| NODETYPE_ACCURACY //!< Accuracy test case -- can be executed |
| }; |
| |
| enum TestNodeClass |
| { |
| NODECLASS_GROUP = 0, //!< Root or non-leaf in the test hierarchy tree |
| NODECLASS_EXECUTABLE, //!< Non-root leaf in the test hierarchy tree |
| |
| NODECLASS_LAST |
| }; |
| |
| enum TestRunnerType |
| { |
| RUNNERTYPE_ANY = 0u, |
| RUNNERTYPE_NONE = (1u << 0), |
| RUNNERTYPE_AMBER = (1u << 1) |
| }; |
| |
| inline TestNodeClass getTestNodeTypeClass (TestNodeType type) |
| { |
| switch (type) |
| { |
| case NODETYPE_ROOT: return NODECLASS_GROUP; |
| case NODETYPE_PACKAGE: return NODECLASS_GROUP; |
| case NODETYPE_GROUP: return NODECLASS_GROUP; |
| case NODETYPE_SELF_VALIDATE: return NODECLASS_EXECUTABLE; |
| case NODETYPE_PERFORMANCE: return NODECLASS_EXECUTABLE; |
| case NODETYPE_CAPABILITY: return NODECLASS_EXECUTABLE; |
| case NODETYPE_ACCURACY: return NODECLASS_EXECUTABLE; |
| default: |
| DE_ASSERT(false); |
| return NODECLASS_LAST; |
| } |
| } |
| |
| inline bool isTestNodeTypeExecutable (TestNodeType type) |
| { |
| return getTestNodeTypeClass(type) == NODECLASS_EXECUTABLE; |
| } |
| |
| inline bool isValidTestCaseNameChar (char c) |
| { |
| return de::inRange(c, 'a', 'z') || |
| de::inRange(c, 'A', 'Z') || |
| de::inRange(c, '0', '9') || |
| c == '_' || c == '-'; |
| } |
| |
| /*--------------------------------------------------------------------*//*! |
| * \brief Test case hierarchy node |
| * |
| * Test node forms the backbone of the test case hierarchy. All objects |
| * in the hierarchy are derived from this class. |
| * |
| * Each test node has a type and all except the root node have name and |
| * description. Root and test group nodes have a list of children. |
| * |
| * During test execution TestExecutor iterates the hierarchy. Upon entering |
| * the node (both groups and test cases) init() is called. When exiting the |
| * node deinit() is called respectively. |
| *//*--------------------------------------------------------------------*/ |
| class TestNode |
| { |
| public: |
| enum IterateResult |
| { |
| STOP = 0, |
| CONTINUE = 1 |
| }; |
| |
| // Methods. |
| TestNode (TestContext& testCtx, TestNodeType nodeType, const char* name, const char* description); |
| TestNode (TestContext& testCtx, TestNodeType nodeType, const char* name, const char* description, const std::vector<TestNode*>& children); |
| virtual ~TestNode (void); |
| |
| TestNodeType getNodeType (void) const { return m_nodeType; } |
| TestContext& getTestContext (void) const { return m_testCtx; } |
| const char* getName (void) const { return m_name.c_str(); } |
| const char* getDescription (void) const { return m_description.c_str(); } |
| void getChildren (std::vector<TestNode*>& children); |
| void addChild (TestNode* node); |
| bool empty () const { return m_children.empty(); } |
| |
| virtual void init (void); |
| virtual void deinit (void); |
| virtual IterateResult iterate (void) = 0; |
| virtual TestRunnerType getRunnerType (void) const { return RUNNERTYPE_NONE; } |
| virtual bool validateRequirements () { return true; } |
| protected: |
| TestContext& m_testCtx; |
| std::string m_name; |
| std::string m_description; |
| |
| private: |
| const TestNodeType m_nodeType; |
| std::vector<TestNode*> m_children; |
| }; |
| |
| /*--------------------------------------------------------------------*//*! |
| * \brief Test case group node |
| * |
| * Test case group implementations must inherit this class. To save resources |
| * during test execution the group must delay creation of any child groups |
| * until init() is called. |
| * |
| * Default deinit() for test group will destroy all child nodes. |
| *//*--------------------------------------------------------------------*/ |
| class TestCaseGroup : public TestNode |
| { |
| public: |
| TestCaseGroup (TestContext& testCtx, const char* name, const char* description); |
| TestCaseGroup (TestContext& testCtx, const char* name, const char* description, const std::vector<TestNode*>& children); |
| virtual ~TestCaseGroup (void); |
| |
| virtual IterateResult iterate (void); |
| }; |
| |
| /*--------------------------------------------------------------------*//*! |
| * \brief Test case class |
| * |
| * Test case implementations must inherit this class. |
| * |
| * Test case objects are usually constructed when TestExecutor enters parent |
| * group. Allocating any non-parameter resources, especially target API objects |
| * must be delayed to init(). |
| * |
| * Upon entering the test case TestExecutor calls init(). If initialization |
| * is successful (no exception is thrown) the executor will then call iterate() |
| * until test case returns STOP. After that deinit() will be called. |
| * |
| * Before exiting the execution phase (i.e. at returning STOP from iterate()) |
| * the test case must set valid status code to test context (m_testCtx). |
| * |
| * Test case can also signal error condition by throwing an exception. In |
| * that case the framework will set result code and details based on the |
| * exception. |
| *//*--------------------------------------------------------------------*/ |
| class TestCase : public TestNode |
| { |
| public: |
| TestCase (TestContext& testCtx, const char* name, const char* description); |
| TestCase (TestContext& testCtx, TestNodeType nodeType, const char* name, const char* description); |
| virtual ~TestCase (void); |
| }; |
| |
| class TestStatus |
| { |
| public: |
| TestStatus (qpTestResult code, const std::string& description) : m_code(code), m_description(description) {} |
| |
| bool isComplete (void) const { return m_code != QP_TEST_RESULT_LAST; } |
| qpTestResult getCode (void) const { DE_ASSERT(isComplete()); return m_code; } |
| const std::string& getDescription (void) const { DE_ASSERT(isComplete()); return m_description; } |
| |
| static TestStatus pass (const std::string& description) { return TestStatus(QP_TEST_RESULT_PASS, description); } |
| static TestStatus fail (const std::string& description) { return TestStatus(QP_TEST_RESULT_FAIL, description); } |
| static TestStatus incomplete (void) { return TestStatus(QP_TEST_RESULT_LAST, ""); } |
| |
| private: |
| qpTestResult m_code; |
| std::string m_description; |
| } DE_WARN_UNUSED_TYPE; |
| |
| } // tcu |
| |
| #endif // _TCUTESTCASE_HPP |