summaryrefslogtreecommitdiffstats
path: root/libs/surfaceflinger/MessageQueue.h
diff options
context:
space:
mode:
authorMathias Agopian <mathias@google.com>2009-04-20 19:39:12 -0700
committerMathias Agopian <mathias@google.com>2009-04-24 16:22:36 -0700
commitf1d8e87b09abf963cd5b6a026194c1940fadb7b4 (patch)
treef26b490726ff7548b08cd705f1411fe13528e6a4 /libs/surfaceflinger/MessageQueue.h
parent9a11206fe793363c0e8897b478cbe6ef8c52b543 (diff)
downloadframeworks_native-f1d8e87b09abf963cd5b6a026194c1940fadb7b4.zip
frameworks_native-f1d8e87b09abf963cd5b6a026194c1940fadb7b4.tar.gz
frameworks_native-f1d8e87b09abf963cd5b6a026194c1940fadb7b4.tar.bz2
a brand new MessageQueue for SurfaceFlinger.
Diffstat (limited to 'libs/surfaceflinger/MessageQueue.h')
-rw-r--r--libs/surfaceflinger/MessageQueue.h224
1 files changed, 224 insertions, 0 deletions
diff --git a/libs/surfaceflinger/MessageQueue.h b/libs/surfaceflinger/MessageQueue.h
new file mode 100644
index 0000000..d894fe2
--- /dev/null
+++ b/libs/surfaceflinger/MessageQueue.h
@@ -0,0 +1,224 @@
+/*
+ * Copyright (C) 2009 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.
+ */
+
+#ifndef ANDROID_MESSAGE_QUEUE_H
+#define ANDROID_MESSAGE_QUEUE_H
+
+#include <stdint.h>
+#include <errno.h>
+#include <sys/types.h>
+
+#include <utils/threads.h>
+#include <utils/Timers.h>
+
+
+namespace android {
+
+// ---------------------------------------------------------------------------
+
+template<typename NODE_PTR_TYPE>
+class DoublyLinkedList
+{
+protected:
+ typedef NODE_PTR_TYPE NODE_PTR;
+
+ NODE_PTR mFirst;
+ NODE_PTR mLast;
+
+public:
+ class Node {
+ friend class DoublyLinkedList;
+ mutable NODE_PTR next;
+ mutable NODE_PTR prev;
+ public:
+ typedef NODE_PTR PTR;
+ inline NODE_PTR getNext() const { return next; }
+ inline NODE_PTR getPrev() const { return prev; }
+ void detach() {
+ prev = 0;
+ next = 0;
+ }
+ };
+
+ DoublyLinkedList() : mFirst(0), mLast(0) { }
+ ~DoublyLinkedList() { }
+
+ bool isEmpty() const { return mFirst == 0; }
+ const NODE_PTR& head() const { return mFirst; }
+ const NODE_PTR& tail() const { return mLast; }
+ const NODE_PTR& head() { return mFirst; }
+ const NODE_PTR& tail() { return mLast; }
+
+ void insertAfter(const NODE_PTR& node, const NODE_PTR& newNode) {
+ newNode->prev = node;
+ newNode->next = node->next;
+ if (node->next == 0) mLast = newNode;
+ else node->next->prev = newNode;
+ node->next = newNode;
+ }
+
+ void insertBefore(const NODE_PTR& node, const NODE_PTR& newNode) {
+ newNode->prev = node->prev;
+ newNode->next = node;
+ if (node->prev == 0) mFirst = newNode;
+ else node->prev->next = newNode;
+ node->prev = newNode;
+ }
+
+ void insertHead(const NODE_PTR& newNode) {
+ if (mFirst == 0) {
+ mFirst = mLast = newNode;
+ newNode->prev = newNode->next = 0;
+ } else {
+ newNode->prev = 0;
+ newNode->next = mFirst;
+ mFirst->prev = newNode;
+ mFirst = newNode;
+ }
+ }
+
+ void insertTail(const NODE_PTR& newNode) {
+ if (mLast == 0) {
+ insertHead(newNode);
+ } else {
+ newNode->prev = mLast;
+ newNode->next = 0;
+ mLast->next = newNode;
+ mLast = newNode;
+ }
+ }
+
+ NODE_PTR remove(const NODE_PTR& node) {
+ if (node->prev == 0) mFirst = node->next;
+ else node->prev->next = node->next;
+ if (node->next == 0) mLast = node->prev;
+ else node->next->prev = node->prev;
+ return node;
+ }
+};
+
+// ---------------------------------------------------------------------------
+
+template<typename NODE_PTR_TYPE>
+class SortedList : protected DoublyLinkedList< NODE_PTR_TYPE >
+{
+ typedef DoublyLinkedList< NODE_PTR_TYPE > forward;
+public:
+ typedef NODE_PTR_TYPE NODE_PTR;
+ inline bool isEmpty() const { return forward::isEmpty(); }
+ inline const NODE_PTR& head() const { return forward::head(); }
+ inline const NODE_PTR& tail() const { return forward::tail(); }
+ inline const NODE_PTR& head() { return forward::head(); }
+ inline const NODE_PTR& tail() { return forward::tail(); }
+ inline NODE_PTR remove(const NODE_PTR& node) { return forward::remove(node); }
+ void insert(const NODE_PTR& node) {
+ NODE_PTR l(head());
+ while (l != 0) {
+ if (*node < *l) {
+ insertBefore(l, node);
+ return;
+ }
+ l = l->getNext();
+ }
+ insertTail(node);
+ }
+};
+
+// ============================================================================
+
+class MessageBase :
+ public LightRefBase<MessageBase>,
+ public DoublyLinkedList< sp<MessageBase> >::Node
+{
+public:
+ nsecs_t when;
+ uint32_t what;
+ int32_t arg0;
+
+ MessageBase(uint32_t what=0, int32_t arg0=0)
+ : when(0), what(what), arg0(arg0) { }
+
+ // return true if message has a handler
+ virtual bool handler() { return false; }
+
+protected:
+ virtual ~MessageBase() { }
+
+private:
+ friend class LightRefBase<MessageBase>;
+};
+
+inline bool operator < (const MessageBase& lhs, const MessageBase& rhs) {
+ return lhs.when < rhs.when;
+}
+
+// ---------------------------------------------------------------------------
+
+/*
+ * MessageList is a sorted list of sp<MessageBase>
+ */
+
+typedef SortedList< MessageBase::Node::PTR > MessageList;
+
+// ---------------------------------------------------------------------------
+
+class MessageQueue
+{
+public:
+
+ // this is a work-around the multichar constant warning. A macro would
+ // work too, but would pollute the namespace.
+ template <int a, int b, int c, int d>
+ struct WHAT {
+ static const uint32_t Value =
+ (uint32_t(a&0xff)<<24)|(uint32_t(b&0xff)<<16)|
+ (uint32_t(c&0xff)<<8)|uint32_t(d&0xff);
+ };
+
+ MessageQueue();
+ ~MessageQueue();
+
+ // pre-defined messages
+ enum {
+ INVALIDATE = WHAT<'_','p','d','t'>::Value
+ };
+
+ MessageList::NODE_PTR waitMessage(nsecs_t timeout = -1);
+
+ status_t postMessage(const MessageList::NODE_PTR& message,
+ nsecs_t reltime=0, uint32_t flags = 0);
+
+ status_t invalidate();
+
+ void dump(const MessageList::NODE_PTR& message);
+
+private:
+ status_t queueMessage(const MessageList::NODE_PTR& message,
+ nsecs_t reltime, uint32_t flags);
+ void dumpLocked(const MessageList::NODE_PTR& message);
+
+ Mutex mLock;
+ Condition mCondition;
+ MessageList mMessages;
+ bool mInvalidate;
+ MessageList::NODE_PTR mInvalidateMessage;
+};
+
+// ---------------------------------------------------------------------------
+
+}; // namespace android
+
+#endif /* ANDROID_MESSAGE_QUEUE_H */