mirror of
https://github.com/gnustep/libs-base.git
synced 2025-05-31 16:50:58 +00:00
Tidyup
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@3630 72102866-910b-0410-8b05-ffd578937521
This commit is contained in:
parent
723e76469e
commit
04ab713dac
4 changed files with 303 additions and 90 deletions
|
@ -58,15 +58,10 @@
|
||||||
|
|
||||||
@interface NSDistantObject(GNUstepExtensions)
|
@interface NSDistantObject(GNUstepExtensions)
|
||||||
|
|
||||||
+ newForRemoteTarget: (unsigned)target connection: (NSConnection*)conn;
|
|
||||||
|
|
||||||
- awakeAfterUsingCoder: aDecoder;
|
- awakeAfterUsingCoder: aDecoder;
|
||||||
- classForPortCoder;
|
- classForPortCoder;
|
||||||
+ newWithCoder: aRmc;
|
|
||||||
- (const char *) selectorTypeForProxy: (SEL)selector;
|
- (const char *) selectorTypeForProxy: (SEL)selector;
|
||||||
- forward: (SEL)aSel :(arglist_t)frame;
|
- forward: (SEL)aSel :(arglist_t)frame;
|
||||||
- (id) localForProxy;
|
|
||||||
- (unsigned) targetForProxy;
|
|
||||||
@end
|
@end
|
||||||
|
|
||||||
#endif /* __NSDistantObject_h_GNUSTEP_BASE_INCLUDE */
|
#endif /* __NSDistantObject_h_GNUSTEP_BASE_INCLUDE */
|
||||||
|
|
|
@ -277,7 +277,6 @@ static BOOL debug_connected_coder = NO;
|
||||||
if (is_decoding)
|
if (is_decoding)
|
||||||
{
|
{
|
||||||
NSAssert([anObj isProxy], NSInternalInconsistencyException);
|
NSAssert([anObj isProxy], NSInternalInconsistencyException);
|
||||||
NSAssert([anObj targetForProxy] == xref, NSInternalInconsistencyException);
|
|
||||||
/* This gets done in Proxy +newForRemote:connection:
|
/* This gets done in Proxy +newForRemote:connection:
|
||||||
[connection addProxy:anObj]; */
|
[connection addProxy:anObj]; */
|
||||||
}
|
}
|
||||||
|
|
|
@ -64,71 +64,65 @@ NSString *NSConnectionProxyCount = @"NSConnectionProxyCount";
|
||||||
|
|
||||||
@interface NSDistantObject (NSConnection)
|
@interface NSDistantObject (NSConnection)
|
||||||
- (BOOL) isVended;
|
- (BOOL) isVended;
|
||||||
|
- (id) localForProxy;
|
||||||
- (void) setProxyTarget: (unsigned)target;
|
- (void) setProxyTarget: (unsigned)target;
|
||||||
- (void) setVended;
|
- (void) setVended;
|
||||||
|
- (unsigned) targetForProxy;
|
||||||
@end
|
@end
|
||||||
|
|
||||||
@implementation NSDistantObject (NSConnection)
|
@implementation NSDistantObject (NSConnection)
|
||||||
- (BOOL) isVended
|
- (BOOL) isVended
|
||||||
{
|
{
|
||||||
return _isVended;
|
return _isVended;
|
||||||
|
}
|
||||||
|
- (id) localForProxy
|
||||||
|
{
|
||||||
|
return _object;
|
||||||
}
|
}
|
||||||
- (void) setProxyTarget: (unsigned)target
|
- (void) setProxyTarget: (unsigned)target
|
||||||
{
|
{
|
||||||
_handle = target;
|
_handle = target;
|
||||||
}
|
}
|
||||||
- (void) setVended
|
- (void) setVended
|
||||||
{
|
{
|
||||||
_isVended = YES;
|
_isVended = YES;
|
||||||
|
}
|
||||||
|
- (unsigned) targetForProxy
|
||||||
|
{
|
||||||
|
return _handle;
|
||||||
}
|
}
|
||||||
@end
|
@end
|
||||||
|
|
||||||
static unsigned local_object_counter = 0;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* ConnectionLocalCounter is a trivial class to keep track of how
|
* GSLocalCounter is a trivial class to keep track of how
|
||||||
* many different connections a particular local object is vended
|
* many different connections a particular local object is vended
|
||||||
* over. This is required so that we know when to remove an object
|
* over. This is required so that we know when to remove an object
|
||||||
* from the global list when it is removed from the list of objects
|
* from the global list when it is removed from the list of objects
|
||||||
* vended on a particular connection.
|
* vended on a particular connection.
|
||||||
*/
|
*/
|
||||||
@interface ConnectionLocalCounter : NSObject
|
@interface GSLocalCounter : NSObject
|
||||||
{
|
{
|
||||||
@public
|
@public
|
||||||
unsigned ref;
|
unsigned ref;
|
||||||
unsigned target;
|
unsigned target;
|
||||||
id object;
|
id object;
|
||||||
}
|
}
|
||||||
- (void)decrement;
|
+ (GSLocalCounter*) newWithObject: (id)ob;
|
||||||
- (void)increment;
|
|
||||||
- (unsigned int) value;
|
|
||||||
@end
|
@end
|
||||||
|
|
||||||
@implementation ConnectionLocalCounter
|
@implementation GSLocalCounter
|
||||||
|
|
||||||
- (void) decrement
|
static unsigned local_object_counter = 0;
|
||||||
{
|
|
||||||
ref--;
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void) increment
|
+ (GSLocalCounter*) newWithObject: (id)obj
|
||||||
{
|
{
|
||||||
ref++;
|
GSLocalCounter *counter;
|
||||||
}
|
|
||||||
|
|
||||||
- init
|
counter = (GSLocalCounter*)NSAllocateObject(self, 0, NSDefaultMallocZone());
|
||||||
{
|
counter->ref = 1;
|
||||||
self = [super init];
|
counter->object = obj;
|
||||||
if (self)
|
counter->target = ++local_object_counter;
|
||||||
{
|
return counter;
|
||||||
ref = 1;
|
|
||||||
}
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
- (unsigned int) value
|
|
||||||
{
|
|
||||||
return ref;
|
|
||||||
}
|
}
|
||||||
@end
|
@end
|
||||||
|
|
||||||
|
@ -346,7 +340,7 @@ static int messages_received_count;
|
||||||
NSCreateMapTable (NSIntMapKeyCallBacks,
|
NSCreateMapTable (NSIntMapKeyCallBacks,
|
||||||
NSNonOwnedPointerMapValueCallBacks, 0);
|
NSNonOwnedPointerMapValueCallBacks, 0);
|
||||||
all_connections_local_cached =
|
all_connections_local_cached =
|
||||||
NSCreateMapTable (NSNonOwnedPointerMapKeyCallBacks,
|
NSCreateMapTable (NSIntMapKeyCallBacks,
|
||||||
NSObjectMapValueCallBacks, 0);
|
NSObjectMapValueCallBacks, 0);
|
||||||
received_request_rmc_queue = [[NSMutableArray alloc] initWithCapacity:32];
|
received_request_rmc_queue = [[NSMutableArray alloc] initWithCapacity:32];
|
||||||
received_request_rmc_queue_gate = [NSLock new];
|
received_request_rmc_queue_gate = [NSLock new];
|
||||||
|
@ -402,7 +396,8 @@ static int messages_received_count;
|
||||||
CachedLocalObject *item = [cached_locals objectAtIndex: i-1];
|
CachedLocalObject *item = [cached_locals objectAtIndex: i-1];
|
||||||
|
|
||||||
if ([item countdown] == NO) {
|
if ([item countdown] == NO) {
|
||||||
NSMapRemove(all_connections_local_cached, [item obj]);
|
GSLocalCounter *counter = [item obj];
|
||||||
|
NSMapRemove(all_connections_local_cached, (void*)counter->target);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ([cached_locals count] == 0) {
|
if ([cached_locals count] == 0) {
|
||||||
|
@ -1504,9 +1499,9 @@ static int messages_received_count;
|
||||||
|
|
||||||
if ([self includesLocalTarget: target] == nil)
|
if ([self includesLocalTarget: target] == nil)
|
||||||
{
|
{
|
||||||
ConnectionLocalCounter *counter;
|
GSLocalCounter *counter;
|
||||||
|
|
||||||
counter = (ConnectionLocalCounter*)[[self class] includesLocalTarget: target];
|
counter = (GSLocalCounter*)[[self class] includesLocalTarget: target];
|
||||||
if (counter != nil)
|
if (counter != nil)
|
||||||
[NSDistantObject proxyWithLocal: counter->object connection: self];
|
[NSDistantObject proxyWithLocal: counter->object connection: self];
|
||||||
}
|
}
|
||||||
|
@ -1798,9 +1793,9 @@ static int messages_received_count;
|
||||||
/* Managing objects and proxies. */
|
/* Managing objects and proxies. */
|
||||||
- (void) addLocalObject: anObj
|
- (void) addLocalObject: anObj
|
||||||
{
|
{
|
||||||
id object = [anObj localForProxy];
|
id object = [anObj localForProxy];
|
||||||
unsigned target;
|
unsigned target;
|
||||||
ConnectionLocalCounter *counter;
|
GSLocalCounter *counter;
|
||||||
|
|
||||||
NSParameterAssert (is_valid);
|
NSParameterAssert (is_valid);
|
||||||
[proxiesHashGate lock];
|
[proxiesHashGate lock];
|
||||||
|
@ -1814,15 +1809,13 @@ static int messages_received_count;
|
||||||
counter = NSMapGet(all_connections_local_targets, (void*)target);
|
counter = NSMapGet(all_connections_local_targets, (void*)target);
|
||||||
if (counter)
|
if (counter)
|
||||||
{
|
{
|
||||||
[counter increment];
|
counter->ref++;
|
||||||
target = counter->target;
|
target = counter->target;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
counter = [ConnectionLocalCounter new];
|
counter = [GSLocalCounter newWithObject: object];
|
||||||
target = ++local_object_counter;
|
target = counter->target;
|
||||||
counter->target = target;
|
|
||||||
counter->object = object;
|
|
||||||
NSMapInsert(all_connections_local_objects, (void*)object, counter);
|
NSMapInsert(all_connections_local_objects, (void*)object, counter);
|
||||||
NSMapInsert(all_connections_local_targets, (void*)target, counter);
|
NSMapInsert(all_connections_local_targets, (void*)target, counter);
|
||||||
[counter release];
|
[counter release];
|
||||||
|
@ -1831,7 +1824,7 @@ static int messages_received_count;
|
||||||
NSMapInsert(local_targets, (void*)target, anObj);
|
NSMapInsert(local_targets, (void*)target, anObj);
|
||||||
if (debug_connection > 2)
|
if (debug_connection > 2)
|
||||||
NSLog(@"add local object (0x%x) to connection (0x%x) (ref %d)\n",
|
NSLog(@"add local object (0x%x) to connection (0x%x) (ref %d)\n",
|
||||||
(unsigned)object, (unsigned) self, [counter value]);
|
(unsigned)object, (unsigned) self, counter->ref);
|
||||||
[proxiesHashGate unlock];
|
[proxiesHashGate unlock];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1866,7 +1859,7 @@ static int messages_received_count;
|
||||||
{
|
{
|
||||||
NSDistantObject *prox;
|
NSDistantObject *prox;
|
||||||
unsigned target;
|
unsigned target;
|
||||||
id counter;
|
GSLocalCounter *counter;
|
||||||
unsigned val = 0;
|
unsigned val = 0;
|
||||||
|
|
||||||
[proxiesHashGate lock];
|
[proxiesHashGate lock];
|
||||||
|
@ -1881,8 +1874,8 @@ static int messages_received_count;
|
||||||
counter = NSMapGet(all_connections_local_objects, (void*)anObj);
|
counter = NSMapGet(all_connections_local_objects, (void*)anObj);
|
||||||
if (counter)
|
if (counter)
|
||||||
{
|
{
|
||||||
[counter decrement];
|
counter->ref--;
|
||||||
if ((val = [counter value]) == 0)
|
if ((val = counter->ref) == 0)
|
||||||
{
|
{
|
||||||
NSMapRemove(all_connections_local_objects, (void*)anObj);
|
NSMapRemove(all_connections_local_objects, (void*)anObj);
|
||||||
NSMapRemove(all_connections_local_targets, (void*)target);
|
NSMapRemove(all_connections_local_targets, (void*)target);
|
||||||
|
@ -1902,8 +1895,8 @@ static int messages_received_count;
|
||||||
userInfo: nil
|
userInfo: nil
|
||||||
repeats: YES];
|
repeats: YES];
|
||||||
}
|
}
|
||||||
item = [CachedLocalObject itemWithObject: anObj time: 30];
|
item = [CachedLocalObject itemWithObject: counter time: 30];
|
||||||
NSMapInsert(all_connections_local_cached, anObj, item);
|
NSMapInsert(all_connections_local_cached, (void*)target, item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,7 +28,90 @@
|
||||||
#include <Foundation/NSMethodSignature.h>
|
#include <Foundation/NSMethodSignature.h>
|
||||||
#include <Foundation/NSException.h>
|
#include <Foundation/NSException.h>
|
||||||
|
|
||||||
static int debug_proxy;
|
static int debug_proxy = 0;
|
||||||
|
static Class placeHolder = 0;
|
||||||
|
static Class distantObjectClass = 0;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The GSDistantObjectPlaceHolder class is simply used as a placeholder
|
||||||
|
* for an NSDistantObject so we can manage efficient allocation and
|
||||||
|
* initialisation - in most cases when we ask for an NSDistantObject
|
||||||
|
* instance, we will get a pre-existing one, so we don't want to go
|
||||||
|
* allocating the memory for a new instance unless absolutely necessary.
|
||||||
|
*/
|
||||||
|
@interface GSDistantObjectPlaceHolder
|
||||||
|
+ (id) initWithLocal: (id)anObject connection: (NSConnection*)aConnection;
|
||||||
|
+ (id) initWithTarget: (unsigned)target connection: (NSConnection*)aConnection;
|
||||||
|
+ (void) autorelease;
|
||||||
|
+ (void) release;
|
||||||
|
+ (id) retain;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@implementation GSDistantObjectPlaceHolder
|
||||||
|
|
||||||
|
+ (void) autorelease
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
+ (void) release
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
+ (id) retain
|
||||||
|
{
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
+ (void) initialize
|
||||||
|
{
|
||||||
|
if (self == [GSDistantObjectPlaceHolder class])
|
||||||
|
{
|
||||||
|
distantObjectClass = [NSDistantObject class];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
+ (id) initWithLocal: (id)anObject connection: (NSConnection*)aConnection
|
||||||
|
{
|
||||||
|
NSDistantObject *proxy;
|
||||||
|
|
||||||
|
NSAssert([aConnection isValid], NSInternalInconsistencyException);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If there already is a local proxy for this target/connection
|
||||||
|
* combination, don't create a new one, just return the old one.
|
||||||
|
*/
|
||||||
|
if ((proxy = [aConnection localForObject: anObject]))
|
||||||
|
{
|
||||||
|
return [proxy retain];
|
||||||
|
}
|
||||||
|
|
||||||
|
proxy = (NSDistantObject*)NSAllocateObject(distantObjectClass,
|
||||||
|
0, NSDefaultMallocZone());
|
||||||
|
return [proxy initWithLocal: anObject connection: aConnection];
|
||||||
|
}
|
||||||
|
|
||||||
|
+ (id) initWithTarget: (unsigned)target connection: (NSConnection*)aConnection
|
||||||
|
{
|
||||||
|
NSDistantObject *proxy;
|
||||||
|
|
||||||
|
NSAssert([aConnection isValid], NSInternalInconsistencyException);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If there already is a local proxy for this target/connection
|
||||||
|
* combination, don't create a new one, just return the old one.
|
||||||
|
*/
|
||||||
|
if ((proxy = [aConnection proxyForTarget: target]))
|
||||||
|
{
|
||||||
|
return [proxy retain];
|
||||||
|
}
|
||||||
|
|
||||||
|
proxy = (NSDistantObject*)NSAllocateObject(distantObjectClass,
|
||||||
|
0, NSDefaultMallocZone());
|
||||||
|
return [proxy initWithTarget: target connection: aConnection];
|
||||||
|
}
|
||||||
|
@end
|
||||||
|
|
||||||
@interface NSDistantObject (Debug)
|
@interface NSDistantObject (Debug)
|
||||||
+ (void) setDebug: (int)val;
|
+ (void) setDebug: (int)val;
|
||||||
|
@ -53,32 +136,31 @@ enum
|
||||||
PROXY_REMOTE_FOR_BOTH
|
PROXY_REMOTE_FOR_BOTH
|
||||||
};
|
};
|
||||||
|
|
||||||
|
+ (void) initialize
|
||||||
|
{
|
||||||
|
if (self == [NSDistantObject class])
|
||||||
|
{
|
||||||
|
placeHolder = [GSDistantObjectPlaceHolder class];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
+ (id) allocWithZone: (NSZone*)z
|
||||||
|
{
|
||||||
|
return placeHolder;
|
||||||
|
}
|
||||||
|
|
||||||
+ (NSDistantObject*) proxyWithLocal: (id)anObject
|
+ (NSDistantObject*) proxyWithLocal: (id)anObject
|
||||||
connection: (NSConnection*)aConnection
|
connection: (NSConnection*)aConnection
|
||||||
{
|
{
|
||||||
NSDistantObject *new_proxy;
|
return [[placeHolder initWithLocal: anObject
|
||||||
|
connection: aConnection] autorelease];
|
||||||
NSAssert([aConnection isValid], NSInternalInconsistencyException);
|
|
||||||
if ((new_proxy = [aConnection localForObject: anObject]))
|
|
||||||
{
|
|
||||||
return new_proxy;
|
|
||||||
}
|
|
||||||
return [[[NSDistantObject alloc] initWithLocal: anObject
|
|
||||||
connection: aConnection] autorelease];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
+ (NSDistantObject*) proxyWithTarget: (unsigned)anObject
|
+ (NSDistantObject*) proxyWithTarget: (unsigned)anObject
|
||||||
connection: (NSConnection*)aConnection
|
connection: (NSConnection*)aConnection
|
||||||
{
|
{
|
||||||
NSDistantObject *new_proxy;
|
return [[placeHolder initWithTarget: anObject
|
||||||
|
connection: aConnection] autorelease];
|
||||||
NSAssert([aConnection isValid], NSInternalInconsistencyException);
|
|
||||||
if ((new_proxy = [aConnection proxyForTarget: anObject]))
|
|
||||||
{
|
|
||||||
return new_proxy;
|
|
||||||
}
|
|
||||||
return [[[NSDistantObject alloc] initWithTarget: anObject
|
|
||||||
connection: aConnection] autorelease];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSConnection*) connectionForProxy
|
- (NSConnection*) connectionForProxy
|
||||||
|
@ -111,7 +193,7 @@ enum
|
||||||
|
|
||||||
- (void) encodeWithCoder: (NSCoder*)aRmc
|
- (void) encodeWithCoder: (NSCoder*)aRmc
|
||||||
{
|
{
|
||||||
unsigned proxy_target;
|
unsigned proxy_target;
|
||||||
gsu8 proxy_tag;
|
gsu8 proxy_tag;
|
||||||
NSConnection *encoder_connection;
|
NSConnection *encoder_connection;
|
||||||
|
|
||||||
|
@ -222,10 +304,162 @@ format: @"NSDistantObject objects only encode with PortEncoder class"];
|
||||||
format: @"Not yet implemented '%s'", sel_get_name(_cmd)];
|
format: @"Not yet implemented '%s'", sel_get_name(_cmd)];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id) initWithCoder: (NSCoder*)coder
|
- (id) initWithCoder: (NSCoder*)aCoder
|
||||||
{
|
{
|
||||||
[NSException raise: NSInvalidArgumentException
|
gsu8 proxy_tag;
|
||||||
format: @"Not yet implemented '%s'", sel_get_name(_cmd)];
|
unsigned target;
|
||||||
|
id decoder_connection;
|
||||||
|
|
||||||
|
if ([aCoder class] != [PortDecoder class])
|
||||||
|
{
|
||||||
|
[self release];
|
||||||
|
[NSException raise: NSGenericException
|
||||||
|
format: @"NSDistantObject objects only decode with "
|
||||||
|
@"PortDecoder class"];
|
||||||
|
}
|
||||||
|
|
||||||
|
decoder_connection = [(NSPortCoder*)aCoder connection];
|
||||||
|
NSAssert(decoder_connection, NSInternalInconsistencyException);
|
||||||
|
|
||||||
|
/* First get the tag, so we know what values need to be decoded. */
|
||||||
|
[aCoder decodeValueOfCType: @encode(typeof(proxy_tag))
|
||||||
|
at: &proxy_tag
|
||||||
|
withName: NULL];
|
||||||
|
|
||||||
|
switch (proxy_tag)
|
||||||
|
{
|
||||||
|
case PROXY_LOCAL_FOR_RECEIVER:
|
||||||
|
/*
|
||||||
|
* This was a proxy on the other side of the connection, but
|
||||||
|
* here it's local.
|
||||||
|
* Lookup the target handle to ensure that it exists here.
|
||||||
|
* Return a retained copy of the local target object.
|
||||||
|
*/
|
||||||
|
[aCoder decodeValueOfCType: @encode(typeof(target))
|
||||||
|
at: &target
|
||||||
|
withName: NULL];
|
||||||
|
|
||||||
|
if (debug_proxy)
|
||||||
|
NSLog(@"Receiving a proxy for local object 0x%x "
|
||||||
|
@"connection 0x%x\n", target, (unsigned)decoder_connection);
|
||||||
|
|
||||||
|
if (![[decoder_connection class] includesLocalTarget: target])
|
||||||
|
{
|
||||||
|
[self release];
|
||||||
|
[NSException raise: @"ProxyDecodedBadTarget"
|
||||||
|
format: @"No local object with given address"];
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
NSDistantObject *o;
|
||||||
|
|
||||||
|
o = [decoder_connection includesLocalTarget: target];
|
||||||
|
if (debug_proxy)
|
||||||
|
{
|
||||||
|
NSLog(@"Local object is 0x%x (0x%x)\n",
|
||||||
|
(unsigned)o, (unsigned)o ? o->_object : 0);
|
||||||
|
}
|
||||||
|
[self release];
|
||||||
|
return o ? [o->_object retain] : nil;
|
||||||
|
}
|
||||||
|
|
||||||
|
case PROXY_LOCAL_FOR_SENDER:
|
||||||
|
/*
|
||||||
|
* This was a local object on the other side of the connection,
|
||||||
|
* but here it's a proxy object. Get the target address, and
|
||||||
|
* send [NSDistantObject +proxyWithTarget:connection:]; this will
|
||||||
|
* return the proxy object we already created for this target, or
|
||||||
|
* create a new proxy object if necessary.
|
||||||
|
*/
|
||||||
|
[aCoder decodeValueOfCType: @encode(typeof(target))
|
||||||
|
at: &target
|
||||||
|
withName: NULL];
|
||||||
|
if (debug_proxy)
|
||||||
|
NSLog(@"Receiving a proxy, was local 0x%x connection 0x%x\n",
|
||||||
|
(unsigned)target, (unsigned)decoder_connection);
|
||||||
|
[self release];
|
||||||
|
return [[NSDistantObject proxyWithTarget: target
|
||||||
|
connection: decoder_connection] retain];
|
||||||
|
|
||||||
|
case PROXY_REMOTE_FOR_BOTH:
|
||||||
|
/*
|
||||||
|
* This was a proxy on the other side of the connection, and it
|
||||||
|
* will be a proxy on this side too; that is, the local version
|
||||||
|
* of this object is not on this host, not on the host the
|
||||||
|
* NSPortCoder is connected to, but on a *third* host.
|
||||||
|
* This is why I call this a "triangle connection". In addition
|
||||||
|
* to decoding the target, we decode the OutPort object that we
|
||||||
|
* will use to talk directly to this third host. We send
|
||||||
|
* [NSConnection +newForInPort:outPort:ancestorConnection:]; this
|
||||||
|
* will either return the connection already created for this
|
||||||
|
* inPort/outPort pair, or create a new connection if necessary.
|
||||||
|
*/
|
||||||
|
{
|
||||||
|
NSDistantObject *result;
|
||||||
|
NSConnection *proxy_connection;
|
||||||
|
NSPort *proxy_connection_out_port = nil;
|
||||||
|
|
||||||
|
[aCoder decodeValueOfCType: @encode(typeof(target))
|
||||||
|
at: &target
|
||||||
|
withName: NULL];
|
||||||
|
|
||||||
|
[aCoder decodeObjectAt: &proxy_connection_out_port
|
||||||
|
withName: NULL];
|
||||||
|
|
||||||
|
NSAssert(proxy_connection_out_port, NSInternalInconsistencyException);
|
||||||
|
/* xxx - if there already exists a connection for talking to the
|
||||||
|
* out port, we use that one rather than creating a new one from
|
||||||
|
* our listening port.
|
||||||
|
*
|
||||||
|
* First we try for a connection from our receive port,
|
||||||
|
* Then we try any connection to the send port
|
||||||
|
* Finally we resort to creating a new connection - we don't
|
||||||
|
* release the newly created connection - it will get released
|
||||||
|
* automatically when no proxies are left on it.
|
||||||
|
*/
|
||||||
|
proxy_connection = [[decoder_connection class]
|
||||||
|
connectionByInPort:
|
||||||
|
[decoder_connection receivePort]
|
||||||
|
outPort:
|
||||||
|
proxy_connection_out_port];
|
||||||
|
if (proxy_connection == nil)
|
||||||
|
proxy_connection = [[decoder_connection class]
|
||||||
|
connectionByOutPort:
|
||||||
|
proxy_connection_out_port];
|
||||||
|
if (proxy_connection == nil)
|
||||||
|
proxy_connection = [[decoder_connection class]
|
||||||
|
newForInPort: [decoder_connection receivePort]
|
||||||
|
outPort: proxy_connection_out_port
|
||||||
|
ancestorConnection: decoder_connection];
|
||||||
|
|
||||||
|
if (debug_proxy)
|
||||||
|
NSLog(@"Receiving a triangle-connection proxy 0x%x "
|
||||||
|
@"connection 0x%x\n", target, (unsigned)proxy_connection);
|
||||||
|
|
||||||
|
NSAssert(proxy_connection != decoder_connection, NSInternalInconsistencyException);
|
||||||
|
NSAssert([proxy_connection isValid], NSInternalInconsistencyException);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If we don't already have a proxy for the object on the
|
||||||
|
* remote system, we must tell the other end to retain its
|
||||||
|
* local object for our use.
|
||||||
|
*/
|
||||||
|
if ([proxy_connection includesProxyForTarget: target] == NO)
|
||||||
|
[proxy_connection retainTarget: target];
|
||||||
|
|
||||||
|
[self release];
|
||||||
|
result = [[NSDistantObject proxyWithTarget: target
|
||||||
|
connection: proxy_connection] retain];
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
/* xxx This should be something different than NSGenericException. */
|
||||||
|
[self release];
|
||||||
|
[NSException raise: NSGenericException
|
||||||
|
format: @"Bad proxy tag"];
|
||||||
|
}
|
||||||
|
/* Not reached. */
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -395,6 +629,7 @@ static inline BOOL class_is_kind_of (Class self, Class aClassObject)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#if 0
|
||||||
+ newWithCoder: aRmc
|
+ newWithCoder: aRmc
|
||||||
{
|
{
|
||||||
gsu8 proxy_tag;
|
gsu8 proxy_tag;
|
||||||
|
@ -441,10 +676,10 @@ format: @"NSDistantObject objects only decode with PortDecoder class"];
|
||||||
if (debug_proxy)
|
if (debug_proxy)
|
||||||
{
|
{
|
||||||
NSLog(@"Local object is 0x%x (0x%x)\n",
|
NSLog(@"Local object is 0x%x (0x%x)\n",
|
||||||
(unsigned)o, (unsigned)[o localForProxy]);
|
(unsigned)o, (unsigned)o->_object);
|
||||||
}
|
}
|
||||||
[self release];
|
[self release];
|
||||||
return [[o localForProxy] retain];
|
return [o->_object retain];
|
||||||
}
|
}
|
||||||
|
|
||||||
case PROXY_LOCAL_FOR_SENDER:
|
case PROXY_LOCAL_FOR_SENDER:
|
||||||
|
@ -543,6 +778,7 @@ format: @"NSDistantObject objects only decode with PortDecoder class"];
|
||||||
/* Not reached. */
|
/* Not reached. */
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
- (const char *) selectorTypeForProxy: (SEL)selector
|
- (const char *) selectorTypeForProxy: (SEL)selector
|
||||||
{
|
{
|
||||||
|
@ -566,16 +802,6 @@ format: @"NSDistantObject objects only decode with PortDecoder class"];
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id) localForProxy
|
|
||||||
{
|
|
||||||
return _object;
|
|
||||||
}
|
|
||||||
|
|
||||||
- (unsigned) targetForProxy
|
|
||||||
{
|
|
||||||
return _handle;
|
|
||||||
}
|
|
||||||
|
|
||||||
- forward: (SEL)aSel :(arglist_t)frame
|
- forward: (SEL)aSel :(arglist_t)frame
|
||||||
{
|
{
|
||||||
if (debug_proxy)
|
if (debug_proxy)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue