You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

206 lines
7.8 KiB

/**
* Tencent is pleased to support the open source community by making QMUI_iOS available.
* Copyright (C) 2016-2020 THL A29 Limited, a Tencent company. All rights reserved.
* Licensed under the MIT License (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
* http://opensource.org/licenses/MIT
* 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.
*/
//
// QMUIRuntime.m
// QMUIKit
//
// Created by QMUI Team on 2018/9/5.
//
#import "QMUIRuntime.h"
#import "QMUICommonDefines.h"
#include <mach-o/getsect.h>
#include <mach-o/dyld.h>
@implementation QMUIPropertyDescriptor
+ (instancetype)descriptorWithProperty:(objc_property_t)property {
QMUIPropertyDescriptor *descriptor = [[QMUIPropertyDescriptor alloc] init];
NSString *propertyName = [NSString stringWithUTF8String:property_getName(property)];
descriptor.name = propertyName;
// getter
char *getterChar = property_copyAttributeValue(property, "G");
descriptor.getter = NSSelectorFromString(getterChar != NULL ? [NSString stringWithUTF8String:getterChar] : propertyName);
if (getterChar != NULL) {
free(getterChar);
}
// setter
char *setterChar = property_copyAttributeValue(property, "S");
NSString *setterString = setterChar != NULL ? [NSString stringWithUTF8String:setterChar] : NSStringFromSelector(setterWithGetter(NSSelectorFromString(propertyName)));
descriptor.setter = NSSelectorFromString(setterString);
if (setterChar != NULL) {
free(setterChar);
}
// atomic/nonatomic
char *attrValue_N = property_copyAttributeValue(property, "N");
BOOL isAtomic = (attrValue_N == NULL);
descriptor.isAtomic = isAtomic;
descriptor.isNonatomic = !isAtomic;
if (attrValue_N != NULL) {
free(attrValue_N);
}
// assign/weak/strong/copy
char *attrValue_isCopy = property_copyAttributeValue(property, "C");
char *attrValue_isStrong = property_copyAttributeValue(property, "&");
char *attrValue_isWeak = property_copyAttributeValue(property, "W");
BOOL isCopy = attrValue_isCopy != NULL;
BOOL isStrong = attrValue_isStrong != NULL;
BOOL isWeak = attrValue_isWeak != NULL;
if (attrValue_isCopy != NULL) {
free(attrValue_isCopy);
}
if (attrValue_isStrong != NULL) {
free(attrValue_isStrong);
}
if (attrValue_isWeak != NULL) {
free(attrValue_isWeak);
}
descriptor.isCopy = isCopy;
descriptor.isStrong = isStrong;
descriptor.isWeak = isWeak;
descriptor.isAssign = !isCopy && !isStrong && !isWeak;
// readonly/readwrite
char *attrValue_isReadonly = property_copyAttributeValue(property, "R");
BOOL isReadonly = (attrValue_isReadonly != NULL);
if (attrValue_isReadonly != NULL) {
free(attrValue_isReadonly);
}
descriptor.isReadonly = isReadonly;
descriptor.isReadwrite = !isReadonly;
// type
char *type = property_copyAttributeValue(property, "T");
descriptor.type = [QMUIPropertyDescriptor typeWithEncodeString:[NSString stringWithUTF8String:type]];
if (type != NULL) {
free(type);
}
return descriptor;
}
- (NSString *)description {
NSMutableString *result = [[NSMutableString alloc] init];
[result appendString:@"@property("];
if (self.isNonatomic) [result appendString:@"nonatomic, "];
[result appendString:self.isAssign ? @"assign" : (self.isWeak ? @"weak" : (self.isStrong ? @"strong" : @"copy"))];
if (self.isReadonly) [result appendString:@", readonly"];
if (![NSStringFromSelector(self.getter) isEqualToString:self.name]) [result appendFormat:@", getter=%@", NSStringFromSelector(self.getter)];
if (self.setter != setterWithGetter(NSSelectorFromString(self.name))) [result appendFormat:@", setter=%@", NSStringFromSelector(self.setter)];
[result appendString:@") "];
[result appendString:self.type];
[result appendString:@" "];
[result appendString:self.name];
[result appendString:@";"];
return result.copy;
}
#define _DetectTypeAndReturn(_type) if (strncmp(@encode(_type), typeEncoding, strlen(@encode(_type))) == 0) return @#_type;
+ (NSString *)typeWithEncodeString:(NSString *)encodeString {
if ([encodeString containsString:@"@\""]) {
NSString *result = [encodeString substringWithRange:NSMakeRange(2, encodeString.length - 2 - 1)];
if ([result containsString:@"<"] && [result containsString:@">"]) {
// protocol
if ([result hasPrefix:@"<"]) {
// id pointer
return [NSString stringWithFormat:@"id%@", result];
}
}
// class
return [NSString stringWithFormat:@"%@ *", result];
}
const char *typeEncoding = encodeString.UTF8String;
_DetectTypeAndReturn(NSInteger)
_DetectTypeAndReturn(NSUInteger)
_DetectTypeAndReturn(int)
_DetectTypeAndReturn(short)
_DetectTypeAndReturn(long)
_DetectTypeAndReturn(long long)
_DetectTypeAndReturn(char)
_DetectTypeAndReturn(unsigned char)
_DetectTypeAndReturn(unsigned int)
_DetectTypeAndReturn(unsigned short)
_DetectTypeAndReturn(unsigned long)
_DetectTypeAndReturn(unsigned long long)
_DetectTypeAndReturn(CGFloat)
_DetectTypeAndReturn(float)
_DetectTypeAndReturn(double)
_DetectTypeAndReturn(void)
_DetectTypeAndReturn(char *)
_DetectTypeAndReturn(id)
_DetectTypeAndReturn(Class)
_DetectTypeAndReturn(SEL)
_DetectTypeAndReturn(BOOL)
return encodeString;
}
@end
#ifndef __LP64__
typedef struct mach_header headerType;
#else
typedef struct mach_header_64 headerType;
#endif
static BOOL strendswith(const char *str, const char *suffix) {
if (!str || !suffix) return NO;
size_t lenstr = strlen(str);
size_t lensuffix = strlen(suffix);
if (lensuffix > lenstr) return NO;
return strncmp(str + lenstr - lensuffix, suffix, lensuffix) == 0;
}
static const headerType *getProjectImageHeader() {
const uint32_t imageCount = _dyld_image_count();
const char *target_image_name = ((NSString *)[[NSBundle mainBundle] objectForInfoDictionaryKey:(NSString *)kCFBundleExecutableKey]).UTF8String;
if (!target_image_name || strlen(target_image_name) <= 0) return nil;
const headerType *target_image_header = 0;
for (uint32_t i = 0; i < imageCount; i++) {
const char *image_name = _dyld_get_image_name(i);// name 是一串完整的文件路径,以 image 名结尾
if (strendswith(image_name, target_image_name)) {
target_image_header = (headerType *)_dyld_get_image_header(i);
break;
}
}
return target_image_header;
}
// from https://github.com/opensource-apple/objc4/blob/master/runtime/objc-file.mm
static classref_t *getDataSection(const headerType *machHeader, const char *sectname, size_t *outCount) {
if (!machHeader) return nil;
unsigned long byteCount = 0;
classref_t *data = (classref_t *)getsectiondata(machHeader, "__DATA", sectname, &byteCount);
if (!data) {
data = (classref_t *)getsectiondata(machHeader, "__DATA_CONST", sectname, &byteCount);
}
if (!data) {
data = (classref_t *)getsectiondata(machHeader, "__DATA_DIRTY", sectname, &byteCount);
}
if (outCount) *outCount = byteCount / sizeof(classref_t);
return data;
}
int qmui_getProjectClassList(classref_t **classes) {
size_t count = 0;
if (!!classes) {
*classes = getDataSection(getProjectImageHeader(), "__objc_classlist", &count);
} else {
getDataSection(getProjectImageHeader(), "__objc_classlist", &count);
}
return (int)count;
}