xref: /aosp_15_r20/external/grpc-grpc/src/objective-c/RxLibrary/GRXConcurrentWriteable.m (revision cc02d7e222339f7a4f6ba5f422e6413f4bd931f2)
1*cc02d7e2SAndroid Build Coastguard Worker/*
2*cc02d7e2SAndroid Build Coastguard Worker *
3*cc02d7e2SAndroid Build Coastguard Worker * Copyright 2015 gRPC authors.
4*cc02d7e2SAndroid Build Coastguard Worker *
5*cc02d7e2SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
6*cc02d7e2SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
7*cc02d7e2SAndroid Build Coastguard Worker * You may obtain a copy of the License at
8*cc02d7e2SAndroid Build Coastguard Worker *
9*cc02d7e2SAndroid Build Coastguard Worker *     http://www.apache.org/licenses/LICENSE-2.0
10*cc02d7e2SAndroid Build Coastguard Worker *
11*cc02d7e2SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
12*cc02d7e2SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
13*cc02d7e2SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*cc02d7e2SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
15*cc02d7e2SAndroid Build Coastguard Worker * limitations under the License.
16*cc02d7e2SAndroid Build Coastguard Worker *
17*cc02d7e2SAndroid Build Coastguard Worker */
18*cc02d7e2SAndroid Build Coastguard Worker
19*cc02d7e2SAndroid Build Coastguard Worker#import "GRXConcurrentWriteable.h"
20*cc02d7e2SAndroid Build Coastguard Worker
21*cc02d7e2SAndroid Build Coastguard Worker#import "GRXWriteable.h"
22*cc02d7e2SAndroid Build Coastguard Worker
23*cc02d7e2SAndroid Build Coastguard Worker@interface GRXConcurrentWriteable ()
24*cc02d7e2SAndroid Build Coastguard Worker// This is atomic so that cancellation can nillify it from any thread.
25*cc02d7e2SAndroid Build Coastguard Worker@property(atomic, strong) id<GRXWriteable> writeable;
26*cc02d7e2SAndroid Build Coastguard Worker@end
27*cc02d7e2SAndroid Build Coastguard Worker
28*cc02d7e2SAndroid Build Coastguard Worker@implementation GRXConcurrentWriteable {
29*cc02d7e2SAndroid Build Coastguard Worker  dispatch_queue_t _writeableQueue;
30*cc02d7e2SAndroid Build Coastguard Worker
31*cc02d7e2SAndroid Build Coastguard Worker  // This ivar ensures that writesFinishedWithError: is only sent once to the writeable. Protected
32*cc02d7e2SAndroid Build Coastguard Worker  // by _writeableQueue.
33*cc02d7e2SAndroid Build Coastguard Worker  BOOL _alreadyFinished;
34*cc02d7e2SAndroid Build Coastguard Worker
35*cc02d7e2SAndroid Build Coastguard Worker  // This ivar ensures that a cancelWithError: call prevents further values to be sent to
36*cc02d7e2SAndroid Build Coastguard Worker  // self.writeable. It must support manipulation outside of _writeableQueue and thus needs to be
37*cc02d7e2SAndroid Build Coastguard Worker  // protected by self lock.
38*cc02d7e2SAndroid Build Coastguard Worker  BOOL _cancelled;
39*cc02d7e2SAndroid Build Coastguard Worker}
40*cc02d7e2SAndroid Build Coastguard Worker
41*cc02d7e2SAndroid Build Coastguard Worker- (instancetype)init {
42*cc02d7e2SAndroid Build Coastguard Worker  return [self initWithWriteable:nil];
43*cc02d7e2SAndroid Build Coastguard Worker}
44*cc02d7e2SAndroid Build Coastguard Worker
45*cc02d7e2SAndroid Build Coastguard Worker// Designated initializer
46*cc02d7e2SAndroid Build Coastguard Worker- (instancetype)initWithWriteable:(id<GRXWriteable>)writeable
47*cc02d7e2SAndroid Build Coastguard Worker                    dispatchQueue:(dispatch_queue_t)queue {
48*cc02d7e2SAndroid Build Coastguard Worker  if (self = [super init]) {
49*cc02d7e2SAndroid Build Coastguard Worker    _writeableQueue = queue;
50*cc02d7e2SAndroid Build Coastguard Worker    _writeable = writeable;
51*cc02d7e2SAndroid Build Coastguard Worker    _alreadyFinished = NO;
52*cc02d7e2SAndroid Build Coastguard Worker    _cancelled = NO;
53*cc02d7e2SAndroid Build Coastguard Worker  }
54*cc02d7e2SAndroid Build Coastguard Worker  return self;
55*cc02d7e2SAndroid Build Coastguard Worker}
56*cc02d7e2SAndroid Build Coastguard Worker
57*cc02d7e2SAndroid Build Coastguard Worker- (instancetype)initWithWriteable:(id<GRXWriteable>)writeable {
58*cc02d7e2SAndroid Build Coastguard Worker  return [self initWithWriteable:writeable dispatchQueue:dispatch_get_main_queue()];
59*cc02d7e2SAndroid Build Coastguard Worker}
60*cc02d7e2SAndroid Build Coastguard Worker
61*cc02d7e2SAndroid Build Coastguard Worker- (void)enqueueValue:(id)value completionHandler:(void (^)(void))handler {
62*cc02d7e2SAndroid Build Coastguard Worker  dispatch_async(_writeableQueue, ^{
63*cc02d7e2SAndroid Build Coastguard Worker    if (self->_alreadyFinished) {
64*cc02d7e2SAndroid Build Coastguard Worker      return;
65*cc02d7e2SAndroid Build Coastguard Worker    }
66*cc02d7e2SAndroid Build Coastguard Worker
67*cc02d7e2SAndroid Build Coastguard Worker    @synchronized(self) {
68*cc02d7e2SAndroid Build Coastguard Worker      if (self->_cancelled) {
69*cc02d7e2SAndroid Build Coastguard Worker        return;
70*cc02d7e2SAndroid Build Coastguard Worker      }
71*cc02d7e2SAndroid Build Coastguard Worker    }
72*cc02d7e2SAndroid Build Coastguard Worker
73*cc02d7e2SAndroid Build Coastguard Worker    [self.writeable writeValue:value];
74*cc02d7e2SAndroid Build Coastguard Worker    handler();
75*cc02d7e2SAndroid Build Coastguard Worker  });
76*cc02d7e2SAndroid Build Coastguard Worker}
77*cc02d7e2SAndroid Build Coastguard Worker
78*cc02d7e2SAndroid Build Coastguard Worker- (void)enqueueSuccessfulCompletion {
79*cc02d7e2SAndroid Build Coastguard Worker  dispatch_async(_writeableQueue, ^{
80*cc02d7e2SAndroid Build Coastguard Worker    if (self->_alreadyFinished) {
81*cc02d7e2SAndroid Build Coastguard Worker      return;
82*cc02d7e2SAndroid Build Coastguard Worker    }
83*cc02d7e2SAndroid Build Coastguard Worker    @synchronized(self) {
84*cc02d7e2SAndroid Build Coastguard Worker      if (self->_cancelled) {
85*cc02d7e2SAndroid Build Coastguard Worker        return;
86*cc02d7e2SAndroid Build Coastguard Worker      }
87*cc02d7e2SAndroid Build Coastguard Worker    }
88*cc02d7e2SAndroid Build Coastguard Worker    [self.writeable writesFinishedWithError:nil];
89*cc02d7e2SAndroid Build Coastguard Worker
90*cc02d7e2SAndroid Build Coastguard Worker    // Skip any possible message to the wrapped writeable enqueued after this one.
91*cc02d7e2SAndroid Build Coastguard Worker    self->_alreadyFinished = YES;
92*cc02d7e2SAndroid Build Coastguard Worker    self.writeable = nil;
93*cc02d7e2SAndroid Build Coastguard Worker  });
94*cc02d7e2SAndroid Build Coastguard Worker}
95*cc02d7e2SAndroid Build Coastguard Worker
96*cc02d7e2SAndroid Build Coastguard Worker- (void)cancelWithError:(NSError *)error {
97*cc02d7e2SAndroid Build Coastguard Worker  NSAssert(error != nil, @"For a successful completion, use enqueueSuccessfulCompletion.");
98*cc02d7e2SAndroid Build Coastguard Worker  @synchronized(self) {
99*cc02d7e2SAndroid Build Coastguard Worker    self->_cancelled = YES;
100*cc02d7e2SAndroid Build Coastguard Worker  }
101*cc02d7e2SAndroid Build Coastguard Worker  dispatch_async(_writeableQueue, ^{
102*cc02d7e2SAndroid Build Coastguard Worker    if (self->_alreadyFinished) {
103*cc02d7e2SAndroid Build Coastguard Worker      // a cancel or a successful completion is already issued
104*cc02d7e2SAndroid Build Coastguard Worker      return;
105*cc02d7e2SAndroid Build Coastguard Worker    }
106*cc02d7e2SAndroid Build Coastguard Worker    [self.writeable writesFinishedWithError:error];
107*cc02d7e2SAndroid Build Coastguard Worker
108*cc02d7e2SAndroid Build Coastguard Worker    // Skip any possible message to the wrapped writeable enqueued after this one.
109*cc02d7e2SAndroid Build Coastguard Worker    self->_alreadyFinished = YES;
110*cc02d7e2SAndroid Build Coastguard Worker    self.writeable = nil;
111*cc02d7e2SAndroid Build Coastguard Worker  });
112*cc02d7e2SAndroid Build Coastguard Worker}
113*cc02d7e2SAndroid Build Coastguard Worker
114*cc02d7e2SAndroid Build Coastguard Worker- (void)cancelSilently {
115*cc02d7e2SAndroid Build Coastguard Worker  dispatch_async(_writeableQueue, ^{
116*cc02d7e2SAndroid Build Coastguard Worker    if (self->_alreadyFinished) {
117*cc02d7e2SAndroid Build Coastguard Worker      return;
118*cc02d7e2SAndroid Build Coastguard Worker    }
119*cc02d7e2SAndroid Build Coastguard Worker    self.writeable = nil;
120*cc02d7e2SAndroid Build Coastguard Worker  });
121*cc02d7e2SAndroid Build Coastguard Worker}
122*cc02d7e2SAndroid Build Coastguard Worker@end
123