/* * safe_list.h - safe list template * * Copyright (c) 2014 Intel Corporation * * 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. * * Author: Wind Yuan <feng.yuan@intel.com> */ #ifndef XCAM_SAFE_LIST_H #define XCAM_SAFE_LIST_H #include <base/xcam_defs.h> #include <base/xcam_common.h> #include <errno.h> #include <list> #include <xcam_mutex.h> namespace XCam { template<class OBj> class SafeList { public: typedef SmartPtr<OBj> ObjPtr; typedef std::list<ObjPtr> ObjList; typedef typename std::list<typename SafeList<OBj>::ObjPtr>::iterator ObjIter; SafeList () : _pop_paused (false) {} ~SafeList () { } /* * timeout, -1, wait until wakeup * >=0, wait for @timeout microsseconds */ inline ObjPtr pop (int32_t timeout = -1); inline bool push (const ObjPtr &obj); inline bool erase (const ObjPtr &obj); inline ObjPtr front (); uint32_t size () { SmartLock lock(_mutex); return _obj_list.size(); } bool is_empty () { SmartLock lock(_mutex); return _obj_list.empty(); } void wakeup () { _new_obj_cond.broadcast (); } void pause_pop () { SmartLock lock(_mutex); _pop_paused = true; wakeup (); } void resume_pop () { SmartLock lock(_mutex); _pop_paused = false; } inline void clear (); protected: ObjList _obj_list; Mutex _mutex; XCam::Cond _new_obj_cond; volatile bool _pop_paused; }; template<class OBj> typename SafeList<OBj>::ObjPtr SafeList<OBj>::pop (int32_t timeout) { SmartLock lock (_mutex); int code = 0; while (!_pop_paused && _obj_list.empty() && code == 0) { if (timeout < 0) code = _new_obj_cond.wait(_mutex); else code = _new_obj_cond.timedwait(_mutex, timeout); } if (_pop_paused) return NULL; if (_obj_list.empty()) { if (code == ETIMEDOUT) { XCAM_LOG_DEBUG ("safe list pop timeout"); } else { XCAM_LOG_ERROR ("safe list pop failed, code:%d", code); } return NULL; } SafeList<OBj>::ObjPtr obj = *_obj_list.begin (); _obj_list.erase (_obj_list.begin ()); return obj; } template<class OBj> bool SafeList<OBj>::push (const SafeList<OBj>::ObjPtr &obj) { SmartLock lock (_mutex); _obj_list.push_back (obj); _new_obj_cond.signal (); return true; } template<class OBj> bool SafeList<OBj>::erase (const SafeList<OBj>::ObjPtr &obj) { XCAM_ASSERT (obj.ptr ()); SmartLock lock (_mutex); for (SafeList<OBj>::ObjIter i_obj = _obj_list.begin (); i_obj != _obj_list.end (); ++i_obj) { if ((*i_obj).ptr () == obj.ptr ()) { _obj_list.erase (i_obj); return true; } } return false; } template<class OBj> typename SafeList<OBj>::ObjPtr SafeList<OBj>::front () { SmartLock lock (_mutex); SafeList<OBj>::ObjIter i = _obj_list.begin (); if (i == _obj_list.end ()) return NULL; return *i; } template<class OBj> void SafeList<OBj>::clear () { SmartLock lock (_mutex); SafeList<OBj>::ObjIter i_obj = _obj_list.begin (); while (i_obj != _obj_list.end ()) { _obj_list.erase (i_obj++); } } }; #endif //XCAM_SAFE_LIST_H