/** * 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 #include @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; }