blob: a2802f06d40141f34951c0b16b253d04cb54b094 [file] [log] [blame]
commit-bot@chromium.org50a30432013-10-24 17:44:27 +00001/*
2 * Copyright 2013 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#ifndef SkMessageBus_DEFINED
9#define SkMessageBus_DEFINED
10
mtklein1b249332015-07-07 12:21:21 -070011#include "SkMutex.h"
reed959a2932016-05-05 01:36:43 -070012#include "SkOncePtr.h"
bsalomon23e619c2015-02-06 11:54:28 -080013#include "SkTArray.h"
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000014#include "SkTDArray.h"
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000015#include "SkTypes.h"
16
17template <typename Message>
18class SkMessageBus : SkNoncopyable {
19public:
20 // Post a message to be received by all Inboxes for this Message type. Threadsafe.
21 static void Post(const Message& m);
22
23 class Inbox {
24 public:
25 Inbox();
26 ~Inbox();
27
28 // Overwrite out with all the messages we've received since the last call. Threadsafe.
bsalomon23e619c2015-02-06 11:54:28 -080029 void poll(SkTArray<Message>* out);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000030
31 private:
bsalomon23e619c2015-02-06 11:54:28 -080032 SkTArray<Message> fMessages;
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000033 SkMutex fMessagesMutex;
34
35 friend class SkMessageBus;
36 void receive(const Message& m); // SkMessageBus is a friend only to call this.
37 };
38
39private:
40 SkMessageBus();
41 static SkMessageBus* Get();
mtklein148ec592014-10-13 13:17:56 -070042
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000043 SkTDArray<Inbox*> fInboxes;
44 SkMutex fInboxesMutex;
45};
46
commit-bot@chromium.orgc6658042014-01-15 23:09:01 +000047// This must go in a single .cpp file, not some .h, or we risk creating more than one global
mtklein148ec592014-10-13 13:17:56 -070048// SkMessageBus per type when using shared libraries. NOTE: at most one per file will compile.
mtklein6c59d802015-09-09 09:09:53 -070049#define DECLARE_SKMESSAGEBUS_MESSAGE(Message) \
reed959a2932016-05-05 01:36:43 -070050 SK_DECLARE_STATIC_ONCE_PTR(SkMessageBus<Message>, bus); \
mtklein6c59d802015-09-09 09:09:53 -070051 template <> \
52 SkMessageBus<Message>* SkMessageBus<Message>::Get() { \
reed959a2932016-05-05 01:36:43 -070053 return bus.get([]{ return new SkMessageBus<Message>(); }); \
commit-bot@chromium.orgc6658042014-01-15 23:09:01 +000054 }
55
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000056// ----------------------- Implementation of SkMessageBus::Inbox -----------------------
57
58template<typename Message>
59SkMessageBus<Message>::Inbox::Inbox() {
60 // Register ourselves with the corresponding message bus.
61 SkMessageBus<Message>* bus = SkMessageBus<Message>::Get();
62 SkAutoMutexAcquire lock(bus->fInboxesMutex);
63 bus->fInboxes.push(this);
64}
65
66template<typename Message>
67SkMessageBus<Message>::Inbox::~Inbox() {
68 // Remove ourselves from the corresponding message bus.
69 SkMessageBus<Message>* bus = SkMessageBus<Message>::Get();
70 SkAutoMutexAcquire lock(bus->fInboxesMutex);
71 // This is a cheaper fInboxes.remove(fInboxes.find(this)) when order doesn't matter.
72 for (int i = 0; i < bus->fInboxes.count(); i++) {
73 if (this == bus->fInboxes[i]) {
74 bus->fInboxes.removeShuffle(i);
75 break;
76 }
77 }
78}
79
80template<typename Message>
81void SkMessageBus<Message>::Inbox::receive(const Message& m) {
82 SkAutoMutexAcquire lock(fMessagesMutex);
bsalomon23e619c2015-02-06 11:54:28 -080083 fMessages.push_back(m);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000084}
85
86template<typename Message>
bsalomon23e619c2015-02-06 11:54:28 -080087void SkMessageBus<Message>::Inbox::poll(SkTArray<Message>* messages) {
bsalomon49f085d2014-09-05 13:34:00 -070088 SkASSERT(messages);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000089 messages->reset();
90 SkAutoMutexAcquire lock(fMessagesMutex);
bsalomon23e619c2015-02-06 11:54:28 -080091 fMessages.swap(messages);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000092}
93
94// ----------------------- Implementation of SkMessageBus -----------------------
95
96template <typename Message>
97SkMessageBus<Message>::SkMessageBus() {}
98
99template <typename Message>
commit-bot@chromium.org50a30432013-10-24 17:44:27 +0000100/*static*/ void SkMessageBus<Message>::Post(const Message& m) {
101 SkMessageBus<Message>* bus = SkMessageBus<Message>::Get();
102 SkAutoMutexAcquire lock(bus->fInboxesMutex);
103 for (int i = 0; i < bus->fInboxes.count(); i++) {
104 bus->fInboxes[i]->receive(m);
105 }
106}
107
108#endif // SkMessageBus_DEFINED