From d69814028ebe5ec50ccedb7e64cba7e452371099 Mon Sep 17 00:00:00 2001 From: "scherkus@chromium.org" Date: Thu, 4 Nov 2010 23:14:18 +0000 Subject: Initial StateMatrix idea using composition and void* BUG=none TEST=media_unittests Review URL: http://codereview.chromium.org/3416001 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@65128 0039d316-1c4b-4281-b951-d872f2087c98 --- media/base/state_matrix.cc | 26 ++++++++++++++++ media/base/state_matrix.h | 61 +++++++++++++++++++++++++++++++++++++ media/base/state_matrix_unittest.cc | 53 ++++++++++++++++++++++++++++++++ media/media.gyp | 3 ++ 4 files changed, 143 insertions(+) create mode 100644 media/base/state_matrix.cc create mode 100644 media/base/state_matrix.h create mode 100644 media/base/state_matrix_unittest.cc diff --git a/media/base/state_matrix.cc b/media/base/state_matrix.cc new file mode 100644 index 0000000..f0ef07b --- /dev/null +++ b/media/base/state_matrix.cc @@ -0,0 +1,26 @@ +// Copyright (c) 2010 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "media/base/state_matrix.h" + +#include "base/stl_util-inl.h" + +namespace media { + +StateMatrix::StateMatrix() { +} + +StateMatrix::~StateMatrix() { + STLDeleteValues(&states_); +} + +bool StateMatrix::IsStateDefined(int state) { + return states_.find(state) != states_.end(); +} + +int StateMatrix::ExecuteHandler(void* instance, int state) { + return states_.find(state)->second->ExecuteHandler(instance); +} + +} // namespace media diff --git a/media/base/state_matrix.h b/media/base/state_matrix.h new file mode 100644 index 0000000..779646f --- /dev/null +++ b/media/base/state_matrix.h @@ -0,0 +1,61 @@ +// Copyright (c) 2010 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#ifndef MEDIA_BASE_STATE_MATRIX_H_ +#define MEDIA_BASE_STATE_MATRIX_H_ + +#include + +#include "base/logging.h" + +namespace media { + +class StateMatrix { + public: + StateMatrix(); + ~StateMatrix(); + + template + void AddState(int state, int (HandlerType::* handler)()) { + CHECK(!IsStateDefined(state)); + StateEntryBase* entry = new StateEntry(handler); + states_.insert(std::make_pair(state, entry)); + } + + bool IsStateDefined(int state); + + int ExecuteHandler(void* instance, int state); + + private: + class StateEntryBase { + public: + StateEntryBase() {} + virtual ~StateEntryBase() {} + + virtual int ExecuteHandler(void* instance) = 0; + }; + + template + class StateEntry : public StateEntryBase { + public: + explicit StateEntry(int (HandlerType::* handler)()) : handler_(handler) {} + virtual ~StateEntry() {} + + virtual int ExecuteHandler(void* instance) { + return (reinterpret_cast(instance)->*handler_)(); + } + + private: + int (HandlerType::* handler_)(); + }; + + typedef std::map StateMap; + StateMap states_; + + DISALLOW_COPY_AND_ASSIGN(StateMatrix); +}; + +} // namespace media + +#endif // MEDIA_BASE_STATE_MATRIX_H_ diff --git a/media/base/state_matrix_unittest.cc b/media/base/state_matrix_unittest.cc new file mode 100644 index 0000000..599ea6f --- /dev/null +++ b/media/base/state_matrix_unittest.cc @@ -0,0 +1,53 @@ +// Copyright (c) 2010 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "media/base/state_matrix.h" +#include "testing/gmock/include/gmock/gmock.h" + +using ::testing::Return; + +namespace media { + +class MockStateHandler { + public: + enum State { + A, + B, + C, + }; + + MOCK_METHOD0(HandleA, int()); + MOCK_METHOD0(HandleB, int()); + MOCK_METHOD0(HandleC, int()); +}; + +TEST(StateMatrixTest, AddState) { + MockStateHandler handler; + StateMatrix sm; + + EXPECT_FALSE(sm.IsStateDefined(MockStateHandler::A)); + EXPECT_FALSE(sm.IsStateDefined(MockStateHandler::B)); + EXPECT_FALSE(sm.IsStateDefined(MockStateHandler::C)); + + sm.AddState(MockStateHandler::A, &MockStateHandler::HandleA); + + EXPECT_TRUE(sm.IsStateDefined(MockStateHandler::A)); + EXPECT_FALSE(sm.IsStateDefined(MockStateHandler::B)); + EXPECT_FALSE(sm.IsStateDefined(MockStateHandler::C)); +} + +TEST(StateMatrixTest, ExecuteHandler) { + MockStateHandler handler; + StateMatrix sm; + + sm.AddState(MockStateHandler::A, &MockStateHandler::HandleA); + + EXPECT_CALL(handler, HandleA()) + .WillOnce(Return(MockStateHandler::B)); + + EXPECT_EQ(MockStateHandler::B, + sm.ExecuteHandler(&handler, MockStateHandler::A)); +} + +} // namespace media diff --git a/media/media.gyp b/media/media.gyp index 53c59f4..5ec1c47 100644 --- a/media/media.gyp +++ b/media/media.gyp @@ -95,6 +95,8 @@ 'base/pts_heap.h', 'base/seekable_buffer.cc', 'base/seekable_buffer.h', + 'base/state_matrix.cc', + 'base/state_matrix.h', 'base/synchronizer.cc', 'base/synchronizer.h', 'base/video_frame.cc', @@ -276,6 +278,7 @@ 'base/pts_heap_unittest.cc', 'base/run_all_unittests.cc', 'base/seekable_buffer_unittest.cc', + 'base/state_matrix_unittest.cc', 'base/video_frame_unittest.cc', 'base/yuv_convert_unittest.cc', 'filters/audio_renderer_algorithm_ola_unittest.cc', -- cgit v1.1