blob: 9d155a333ba73cc8587fb2f47618db00cdc722fc [file] [log] [blame]
//
// Copyright 2019 Google LLC.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
#import "Service/Sources/EDODeallocationTracker.h"
#include <objc/runtime.h>
#import "Channel/Sources/EDOHostPort.h"
#import "Service/Sources/EDOClientService+Private.h"
#import "Service/Sources/EDOClientService.h"
#import "Service/Sources/EDOObject+Private.h"
#import "Service/Sources/EDOObjectReleaseMessage.h"
#import "Service/Sources/EDOWeakObject.h"
@interface EDODeallocationTracker ()
/** The tracked object address (EDOWeakObject) that is stored in the weak object dictionary. */
@property(readonly, nonatomic) EDOPointerType remoteObjectAddress;
/** The host port where weak object dictionary holds the remote object. */
@property(readonly, nonatomic) EDOHostPort *hostPort;
@end
@implementation EDODeallocationTracker
+ (void)enableTrackingForObject:(EDOWeakObject *)trackedObject hostPort:(EDOHostPort *)hostPort {
// This does not support multiple weak objects (e.g. from different services) that point to the
// same underlying object, as only one single deallocation tracker is associated with the
// underlying object. The host port is merely used to decide where the release request should be
// sent and only the first port associated with the underlying object is used.
EDODeallocationTracker *tracker = objc_getAssociatedObject(trackedObject.weakObject, &_cmd);
if (!tracker) {
tracker = [[self alloc] initWithTrackedObject:trackedObject hostPort:hostPort];
objc_setAssociatedObject(trackedObject.weakObject, &_cmd, tracker, OBJC_ASSOCIATION_RETAIN);
} else {
NSAssert(
[tracker.hostPort isEqual:hostPort],
@"Deallocation tracker does not support tracking the same object from multiple host ports."
@"Existing port: %@\nNew port: %@",
tracker.hostPort, hostPort);
}
}
- (instancetype)initWithTrackedObject:(EDOWeakObject *)trackedObject
hostPort:(EDOHostPort *)hostPort {
self = [super init];
if (self) {
_remoteObjectAddress = (EDOPointerType)trackedObject;
_hostPort = hostPort;
}
return self;
}
- (void)dealloc {
@try {
EDOObjectReleaseRequest *request =
[EDOObjectReleaseRequest requestWithWeakRemoteAddress:self.remoteObjectAddress];
[EDOClientService sendSynchronousRequest:request onPort:self.hostPort];
} @catch (NSException *e) {
// Safely ignore the exception because we don't care about the errors when we send the release
// message. The service could be terminated, or the message can't be processed, but either way,
// those can be ignored.
}
}
@end