Change log:95336cb92b..191d55580eFull diff:95336cb92b..191d55580eRoll chromium third_party 4e16929f46..3a8f2a9e1e Change log:4e16929f46..3a8f2a9e1eChanged dependencies: * src/tools:c44a3f5eca..f524a53b81DEPS diff:95336cb92b..191d55580e/DEPS No update to Clang. TBR=titovartem@google.com, BUG=None CQ_INCLUDE_TRYBOTS=master.internal.tryserver.corp.webrtc:linux_internal Change-Id: Ic9c4a62b050383646e9fcf5cc07a5653c14ac06e Reviewed-on: https://webrtc-review.googlesource.com/76120 Reviewed-by: Patrik Höglund <phoglund@webrtc.org> Reviewed-by: Karl Wiberg <kwiberg@webrtc.org> Reviewed-by: Artem Titov <titovartem@webrtc.org> Commit-Queue: Artem Titov <titovartem@webrtc.org> Cr-Commit-Position: refs/heads/master@{#23205}
152 lines
4.4 KiB
Objective-C
152 lines
4.4 KiB
Objective-C
/*
|
|
* Copyright (c) 2014-2015 Erik Doernenburg and contributors
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
* not use these files 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 <objc/runtime.h>
|
|
#import <OCMock/OCMArg.h>
|
|
#import <OCMock/OCMConstraint.h>
|
|
#import "OCMPassByRefSetter.h"
|
|
#import "NSInvocation+OCMAdditions.h"
|
|
#import "OCMInvocationMatcher.h"
|
|
#import "OCClassMockObject.h"
|
|
#import "OCMFunctions.h"
|
|
|
|
|
|
@interface NSObject(HCMatcherDummy)
|
|
- (BOOL)matches:(id)item;
|
|
@end
|
|
|
|
|
|
@implementation OCMInvocationMatcher
|
|
|
|
- (void)dealloc
|
|
{
|
|
[recordedInvocation release];
|
|
[super dealloc];
|
|
}
|
|
|
|
- (void)setInvocation:(NSInvocation *)anInvocation
|
|
{
|
|
[recordedInvocation release];
|
|
// When the method has a char* argument we do not retain the arguments. This makes it possible
|
|
// to match char* args literally and with anyPointer. Not retaining the argument means that
|
|
// in these cases tests that use their own autorelease pools may fail unexpectedly.
|
|
if(![anInvocation hasCharPointerArgument])
|
|
[anInvocation retainArguments];
|
|
recordedInvocation = [anInvocation retain];
|
|
}
|
|
|
|
- (void)setRecordedAsClassMethod:(BOOL)flag
|
|
{
|
|
recordedAsClassMethod = flag;
|
|
}
|
|
|
|
- (BOOL)recordedAsClassMethod
|
|
{
|
|
return recordedAsClassMethod;
|
|
}
|
|
|
|
- (void)setIgnoreNonObjectArgs:(BOOL)flag
|
|
{
|
|
ignoreNonObjectArgs = flag;
|
|
}
|
|
|
|
- (NSString *)description
|
|
{
|
|
return [recordedInvocation invocationDescription];
|
|
}
|
|
|
|
- (NSInvocation *)recordedInvocation
|
|
{
|
|
return recordedInvocation;
|
|
}
|
|
|
|
- (BOOL)matchesSelector:(SEL)sel
|
|
{
|
|
if(sel == [recordedInvocation selector])
|
|
return YES;
|
|
if(OCMIsAliasSelector(sel) &&
|
|
OCMOriginalSelectorForAlias(sel) == [recordedInvocation selector])
|
|
return YES;
|
|
|
|
return NO;
|
|
}
|
|
|
|
- (BOOL)matchesInvocation:(NSInvocation *)anInvocation
|
|
{
|
|
id target = [anInvocation target];
|
|
BOOL isClassMethodInvocation = (target != nil) && (target == [target class]);
|
|
if(isClassMethodInvocation != recordedAsClassMethod)
|
|
return NO;
|
|
|
|
if(![self matchesSelector:[anInvocation selector]])
|
|
return NO;
|
|
|
|
NSMethodSignature *signature = [recordedInvocation methodSignature];
|
|
NSUInteger n = [signature numberOfArguments];
|
|
for(NSUInteger i = 2; i < n; i++)
|
|
{
|
|
if(ignoreNonObjectArgs && strcmp([signature getArgumentTypeAtIndex:i], @encode(id)))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
id recordedArg = [recordedInvocation getArgumentAtIndexAsObject:i];
|
|
id passedArg = [anInvocation getArgumentAtIndexAsObject:i];
|
|
|
|
if([recordedArg isProxy])
|
|
{
|
|
if(![recordedArg isEqual:passedArg])
|
|
return NO;
|
|
continue;
|
|
}
|
|
|
|
if([recordedArg isKindOfClass:[NSValue class]])
|
|
recordedArg = [OCMArg resolveSpecialValues:recordedArg];
|
|
|
|
if([recordedArg isKindOfClass:[OCMConstraint class]])
|
|
{
|
|
if([recordedArg evaluate:passedArg] == NO)
|
|
return NO;
|
|
}
|
|
else if([recordedArg isKindOfClass:[OCMPassByRefSetter class]])
|
|
{
|
|
id valueToSet = [(OCMPassByRefSetter *)recordedArg value];
|
|
// side effect but easier to do here than in handleInvocation
|
|
if(![valueToSet isKindOfClass:[NSValue class]])
|
|
*(id *)[passedArg pointerValue] = valueToSet;
|
|
else
|
|
[(NSValue *)valueToSet getValue:[passedArg pointerValue]];
|
|
}
|
|
else if([recordedArg conformsToProtocol:objc_getProtocol("HCMatcher")])
|
|
{
|
|
if([recordedArg matches:passedArg] == NO)
|
|
return NO;
|
|
}
|
|
else
|
|
{
|
|
if(([recordedArg class] == [NSNumber class]) &&
|
|
([(NSNumber*)recordedArg compare:(NSNumber*)passedArg] != NSOrderedSame))
|
|
return NO;
|
|
if(([recordedArg isEqual:passedArg] == NO) &&
|
|
!((recordedArg == nil) && (passedArg == nil)))
|
|
return NO;
|
|
}
|
|
}
|
|
return YES;
|
|
}
|
|
|
|
@end
|