1 /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* vim: set ts=8 sts=2 et sw=2 tw=80: */
3 /* This Source Code Form is subject to the terms of the Mozilla Public
4 * License, v. 2.0. If a copy of the MPL was not distributed with this
5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
7 #include "mozilla/dom/cache/CacheStreamControlParent.h"
9 #include "mozilla/UniquePtr.h"
10 #include "mozilla/Unused.h"
11 #include "mozilla/dom/cache/CacheTypes.h"
12 #include "mozilla/dom/cache/ReadStream.h"
13 #include "mozilla/dom/cache/StreamList.h"
14 #include "mozilla/ipc/IPCStreamUtils.h"
15 #include "mozilla/ipc/PBackgroundParent.h"
16 #include "nsISupportsImpl.h"
19 namespace mozilla::dom::cache
{
21 using mozilla::ipc::FileDescriptor
;
23 // declared in ActorUtils.h
24 void DeallocPCacheStreamControlParent(PCacheStreamControlParent
* aActor
) {
28 CacheStreamControlParent::CacheStreamControlParent() {
29 MOZ_COUNT_CTOR(cache::CacheStreamControlParent
);
32 CacheStreamControlParent::~CacheStreamControlParent() {
33 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
34 MOZ_DIAGNOSTIC_ASSERT(!mStreamList
);
35 MOZ_COUNT_DTOR(cache::CacheStreamControlParent
);
38 void CacheStreamControlParent::SerializeControl(
39 CacheReadStream
* aReadStreamOut
) {
40 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
41 MOZ_DIAGNOSTIC_ASSERT(aReadStreamOut
);
42 aReadStreamOut
->control() = this;
45 void CacheStreamControlParent::SerializeStream(CacheReadStream
* aReadStreamOut
,
46 nsIInputStream
* aStream
) {
47 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
48 MOZ_DIAGNOSTIC_ASSERT(aReadStreamOut
);
50 DebugOnly
<bool> ok
= mozilla::ipc::SerializeIPCStream(
51 do_AddRef(aStream
), aReadStreamOut
->stream(), /* aAllowLazy */ false);
55 void CacheStreamControlParent::OpenStream(const nsID
& aId
,
56 InputStreamResolver
&& aResolver
) {
57 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
58 MOZ_DIAGNOSTIC_ASSERT(aResolver
);
60 if (!mStreamList
|| !mStreamList
->ShouldOpenStreamFor(aId
)) {
65 // Make sure to add ourself as a Listener even thought we are using
66 // a separate resolver function to signal the completion of the
67 // operation. The Manager uses the existence of the Listener to ensure
68 // that its safe to complete the operation.
69 mStreamList
->GetManager().ExecuteOpenStream(this, std::move(aResolver
), aId
);
72 void CacheStreamControlParent::NoteClosedAfterForget(const nsID
& aId
) {
73 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
78 void CacheStreamControlParent::AssertOwningThread() {
79 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
83 void CacheStreamControlParent::LostIPCCleanup(
84 SafeRefPtr
<StreamList
> aStreamList
) {
85 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
86 CloseAllReadStreamsWithoutReporting();
87 // If the initial SendPStreamControlConstructor() fails we will
88 // be called before mStreamList is set.
92 aStreamList
->GetManager().RemoveListener(this);
93 #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
94 aStreamList
->GetManager().RecordHaveDeletedCSCP(Id());
96 aStreamList
->RemoveStreamControl(this);
97 aStreamList
->NoteClosedAll();
98 mStreamList
= nullptr;
101 void CacheStreamControlParent::ActorDestroy(ActorDestroyReason aReason
) {
102 LostIPCCleanup(std::move(mStreamList
));
105 void CacheStreamControlParent::AssertWillDelete() {
106 #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
107 // If we cannot send, Send__delete__ will not do any cleanup.
108 // But if we are still unfreed, that might be wrong.
109 if (mStreamList
&& !CanSend()) {
110 MOZ_ASSERT(false, "Attempt to delete blocking CSCP that cannot send.");
111 mStreamList
->GetManager().RecordMayNotDeleteCSCP(Id());
116 mozilla::ipc::IPCResult
CacheStreamControlParent::RecvOpenStream(
117 const nsID
& aStreamId
, OpenStreamResolver
&& aResolver
) {
118 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
120 OpenStream(aStreamId
, [aResolver
, self
= RefPtr
{this}](
121 nsCOMPtr
<nsIInputStream
>&& aStream
) {
122 Maybe
<IPCStream
> stream
;
123 if (self
->CanSend() &&
124 mozilla::ipc::SerializeIPCStream(aStream
.forget(), stream
,
125 /* aAllowLazy */ false)) {
128 aResolver(Nothing());
135 mozilla::ipc::IPCResult
CacheStreamControlParent::RecvNoteClosed(
137 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
138 MOZ_DIAGNOSTIC_ASSERT(mStreamList
);
139 mStreamList
->NoteClosed(aId
);
143 void CacheStreamControlParent::SetStreamList(
144 SafeRefPtr
<StreamList
> aStreamList
) {
145 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
146 MOZ_DIAGNOSTIC_ASSERT(!mStreamList
);
147 mStreamList
= std::move(aStreamList
);
150 void CacheStreamControlParent::CloseAll() {
151 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
154 QM_WARNONLY_TRY(OkIf(SendCloseAll()));
157 void CacheStreamControlParent::Shutdown() {
158 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
161 // If child process is gone, warn and allow actor to clean up normally
162 QM_WARNONLY_TRY(OkIf(Send__delete__(this)));
165 void CacheStreamControlParent::NotifyCloseAll() {
166 NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent
);
167 CloseAllReadStreams();
170 } // namespace mozilla::dom::cache