File: RequestCallbackManager.h

package info (click to toggle)
firefox 147.0-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 4,683,324 kB
  • sloc: cpp: 7,607,156; javascript: 6,532,492; ansic: 3,775,158; python: 1,415,368; xml: 634,556; asm: 438,949; java: 186,241; sh: 62,751; makefile: 18,079; objc: 13,092; perl: 12,808; yacc: 4,583; cs: 3,846; pascal: 3,448; lex: 1,720; ruby: 1,003; php: 436; lisp: 258; awk: 247; sql: 66; sed: 54; csh: 10; exp: 6
file content (136 lines) | stat: -rw-r--r-- 4,393 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

#ifndef mozilla_dom_RequestCallbackManager_h
#define mozilla_dom_RequestCallbackManager_h

#include <limits>

#include "mozilla/RefPtr.h"
#include "nsTArray.h"
#include "nsThreadUtils.h"

namespace mozilla::dom {

template <typename RequestCallback>
struct RequestCallbackEntry {
  RequestCallbackEntry(RequestCallback& aCallback, uint32_t aHandle)
      : mCallback(&aCallback), mHandle(aHandle) {
    LogTaskBase<RequestCallback>::LogDispatch(mCallback);
  }

  ~RequestCallbackEntry() = default;

  // Comparator operators to allow RemoveElementSorted with an
  // integer argument on arrays of RequestCallback
  bool operator==(uint32_t aHandle) const { return mHandle == aHandle; }
  bool operator<(uint32_t aHandle) const { return mHandle < aHandle; }

  RefPtr<RequestCallback> mCallback;
  const uint32_t mHandle;
  bool mCancelled = false;
};

template <typename RequestCallback>
class RequestCallbackManager {
 public:
  RequestCallbackManager() = default;
  ~RequestCallbackManager() = default;

  using CallbackList = nsTArray<RequestCallbackEntry<RequestCallback>>;

  nsresult Schedule(RequestCallback& aCallback, uint32_t* aHandle) {
    if (mCallbackCounter == std::numeric_limits<uint32_t>::max()) {
      // Can't increment without overflowing; bail out
      return NS_ERROR_NOT_AVAILABLE;
    }
    uint32_t newHandle = ++mCallbackCounter;

    mCallbacks.AppendElement(RequestCallbackEntry(aCallback, newHandle));

    *aHandle = newHandle;
    return NS_OK;
  }

  bool Cancel(uint32_t aHandle) {
    // mCallbacks is stored sorted by handle
    if (mCallbacks.RemoveElementSorted(aHandle)) {
      return true;
    }
    for (auto* callbacks : mFiringCallbacksOnStack) {
      auto index = callbacks->mList.BinaryIndexOf(aHandle);
      if (index != CallbackList::NoIndex) {
        callbacks->mList.ElementAt(index).mCancelled = true;
      }
    }
    return false;
  }

  bool IsEmpty() const { return mCallbacks.IsEmpty(); }

  // FiringCallbacks takes care of:
  //  * Stealing (and thus "freezing") the current callback list, in preparation
  //    for firing them.
  //  * Registering and unregistering in mFiringCallbacksOnStack, to deal with
  //    cancellation of in-flight callbacks in cases like the first callback on
  //    the list calling cancelAnimationFrame(secondCallbackId) or so.
  // mList is guaranteed not to reallocate once stolen. Instead if a callback is
  // cancelled mid-firing, the mCancelled bit is set, see Cancel().
  struct MOZ_NON_MEMMOVABLE MOZ_STACK_CLASS FiringCallbacks {
    explicit FiringCallbacks(RequestCallbackManager& aManager)
        : mManager(aManager) {
      mList = std::move(aManager.mCallbacks);
      aManager.mFiringCallbacksOnStack.AppendElement(this);
    }

    ~FiringCallbacks() {
      MOZ_ASSERT(mManager.mFiringCallbacksOnStack.LastElement() == this);
      mManager.mFiringCallbacksOnStack.RemoveLastElement();
    }

    RequestCallbackManager& mManager;
    CallbackList mList;
  };

  void Unlink() { mCallbacks.Clear(); }

  void Traverse(nsCycleCollectionTraversalCallback& aCB) {
    for (auto& i : mCallbacks) {
      NS_CYCLE_COLLECTION_NOTE_EDGE_NAME(
          aCB, "RequestCallbackManager::mCallbacks[i]");
      aCB.NoteXPCOMChild(i.mCallback);
    }
  }

 private:
  CallbackList mCallbacks;

  // The current lists of callbacks that are executing. Used to deal with
  // cancellation within the same frame. Note this is a list to deal reasonably
  // with event loop spinning.
  AutoTArray<FiringCallbacks*, 1> mFiringCallbacksOnStack;

  // The current frame request callback handle.
  uint32_t mCallbackCounter = 0;
};

template <class RequestCallback>
inline void ImplCycleCollectionUnlink(
    RequestCallbackManager<RequestCallback>& aField) {
  aField.Unlink();
}

template <class RequestCallback>
inline void ImplCycleCollectionTraverse(
    nsCycleCollectionTraversalCallback& aCallback,
    RequestCallbackManager<RequestCallback>& aField, const char* aName,
    uint32_t aFlags) {
  aField.Traverse(aCallback);
}

}  // namespace mozilla::dom

#endif