blob: 01145ecb537c0d59217e8c31e59439bc51e6a2af [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
Ben Wagnerd5148e32018-07-16 17:44:06 -040011#include "../private/SkNoncopyable.h"
mtklein1b249332015-07-07 12:21:21 -070012#include "SkMutex.h"
mtkleinffa4a922016-05-05 16:05:56 -070013#include "SkOnce.h"
bsalomon23e619c2015-02-06 11:54:28 -080014#include "SkTArray.h"
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000015#include "SkTDArray.h"
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000016#include "SkTypes.h"
17
Brian Salomon238069b2018-07-11 15:58:57 -040018/**
19 * Message must implement bool Message::shouldSend(uint32_t inboxID) const. Perhaps someday we
20 * can use std::experimental::is_detected to avoid this requirement by sending to all inboxes when
21 * the method is not detected on Message.
22 */
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000023template <typename Message>
24class SkMessageBus : SkNoncopyable {
25public:
Brian Salomon238069b2018-07-11 15:58:57 -040026 // Post a message to be received by Inboxes for this Message type. Checks
27 // Message::shouldSend() for each inbox. Threadsafe.
28 static void Post(const Message& m);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000029
30 class Inbox {
31 public:
Jim Van Verth474d6872017-12-14 13:00:05 -050032 Inbox(uint32_t uniqueID = SK_InvalidUniqueID);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000033 ~Inbox();
34
35 // Overwrite out with all the messages we've received since the last call. Threadsafe.
bsalomon23e619c2015-02-06 11:54:28 -080036 void poll(SkTArray<Message>* out);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000037
38 private:
bsalomon23e619c2015-02-06 11:54:28 -080039 SkTArray<Message> fMessages;
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000040 SkMutex fMessagesMutex;
Jim Van Verth474d6872017-12-14 13:00:05 -050041 uint32_t fUniqueID;
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000042
43 friend class SkMessageBus;
44 void receive(const Message& m); // SkMessageBus is a friend only to call this.
45 };
46
47private:
48 SkMessageBus();
49 static SkMessageBus* Get();
mtklein148ec592014-10-13 13:17:56 -070050
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000051 SkTDArray<Inbox*> fInboxes;
52 SkMutex fInboxesMutex;
53};
54
commit-bot@chromium.orgc6658042014-01-15 23:09:01 +000055// 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 -070056// SkMessageBus per type when using shared libraries. NOTE: at most one per file will compile.
mtklein6c59d802015-09-09 09:09:53 -070057#define DECLARE_SKMESSAGEBUS_MESSAGE(Message) \
mtklein6c59d802015-09-09 09:09:53 -070058 template <> \
59 SkMessageBus<Message>* SkMessageBus<Message>::Get() { \
mtkleinffa4a922016-05-05 16:05:56 -070060 static SkOnce once; \
61 static SkMessageBus<Message>* bus; \
62 once([] { bus = new SkMessageBus<Message>(); }); \
63 return bus; \
commit-bot@chromium.orgc6658042014-01-15 23:09:01 +000064 }
65
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000066// ----------------------- Implementation of SkMessageBus::Inbox -----------------------
67
68template<typename Message>
Jim Van Verth474d6872017-12-14 13:00:05 -050069SkMessageBus<Message>::Inbox::Inbox(uint32_t uniqueID) : fUniqueID(uniqueID) {
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000070 // Register ourselves with the corresponding message bus.
71 SkMessageBus<Message>* bus = SkMessageBus<Message>::Get();
72 SkAutoMutexAcquire lock(bus->fInboxesMutex);
Mike Reed5edcd312018-08-08 11:23:41 -040073 bus->fInboxes.push_back(this);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000074}
75
76template<typename Message>
77SkMessageBus<Message>::Inbox::~Inbox() {
78 // Remove ourselves from the corresponding message bus.
79 SkMessageBus<Message>* bus = SkMessageBus<Message>::Get();
80 SkAutoMutexAcquire lock(bus->fInboxesMutex);
81 // This is a cheaper fInboxes.remove(fInboxes.find(this)) when order doesn't matter.
82 for (int i = 0; i < bus->fInboxes.count(); i++) {
83 if (this == bus->fInboxes[i]) {
84 bus->fInboxes.removeShuffle(i);
85 break;
86 }
87 }
88}
89
90template<typename Message>
91void SkMessageBus<Message>::Inbox::receive(const Message& m) {
92 SkAutoMutexAcquire lock(fMessagesMutex);
bsalomon23e619c2015-02-06 11:54:28 -080093 fMessages.push_back(m);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000094}
95
96template<typename Message>
bsalomon23e619c2015-02-06 11:54:28 -080097void SkMessageBus<Message>::Inbox::poll(SkTArray<Message>* messages) {
bsalomon49f085d2014-09-05 13:34:00 -070098 SkASSERT(messages);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +000099 messages->reset();
100 SkAutoMutexAcquire lock(fMessagesMutex);
Ben Wagnerf08d1d02018-06-18 15:11:00 -0400101 fMessages.swap(*messages);
commit-bot@chromium.org50a30432013-10-24 17:44:27 +0000102}
103
104// ----------------------- Implementation of SkMessageBus -----------------------
105
106template <typename Message>
107SkMessageBus<Message>::SkMessageBus() {}
108
109template <typename Message>
Brian Salomon238069b2018-07-11 15:58:57 -0400110/*static*/ void SkMessageBus<Message>::Post(const Message& m) {
commit-bot@chromium.org50a30432013-10-24 17:44:27 +0000111 SkMessageBus<Message>* bus = SkMessageBus<Message>::Get();
112 SkAutoMutexAcquire lock(bus->fInboxesMutex);
113 for (int i = 0; i < bus->fInboxes.count(); i++) {
Brian Salomon238069b2018-07-11 15:58:57 -0400114 if (m.shouldSend(bus->fInboxes[i]->fUniqueID)) {
Jim Van Verth474d6872017-12-14 13:00:05 -0500115 bus->fInboxes[i]->receive(m);
116 }
commit-bot@chromium.org50a30432013-10-24 17:44:27 +0000117 }
118}
119
120#endif // SkMessageBus_DEFINED