简体   繁体   English

-[NSString(NSURLUtilities)stringByAddingPercentEncodingWithAllowedCharacters:]中的堆栈溢出

[英]Stack overflow in -[NSString(NSURLUtilities) stringByAddingPercentEncodingWithAllowedCharacters:]

From http://www.openradar.me/20404230 : http://www.openradar.me/20404230

Method -[NSString(NSURLUtilities) stringByAddingPercentEncodingWithAllowedCharacters:] has a stack overflow issue, which can be reproduced with some strings containing hieroglyphs. 方法-[NSString(NSURLUtilities)stringByAddingPercentEncodingWithAllowedCharacters:]有一个堆栈溢出问题,可以用一些包含象形文字的字符串重现。 In these cases __stack_chk_fail will abort the application when building for arm64 architecture, and stack will be corrupted when building for armv7. 在这些情况下,__ arm_chk_fail会在为arm64体系结构构建时中止应用程序,而在为armv7进行构建时堆栈将会损坏。

Example from https://github.com/PavelTretyakov/nsstring-crash will crash on iOS 8.2: 来自https://github.com/PavelTretyakov/nsstring-crash的示例将在iOS 8.2上崩溃:

NSString *str = @"/Users/zaryanov/Movies/rootfolder/시티 오브 히어로 (City of Heroes)/로니 리 가드너 (1961년부터 2010년까지)는 1985 년에 살인죄로 사형을받은 유타 주에서 총살형 된 미국의 악당이었다. 1984 년에 그는 솔트 레이크 시티에서 강도 동안 바텐더를 살해.m4v";
str = [str stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLPathAllowedCharacterSet]];

Example from https://gist.github.com/clowwindy/0d800f07a5e95e5c4dd0 will crash on iOS 8.1: 来自https://gist.github.com/clowwindy/0d800f07a5e95e5c4dd0的示例将在iOS 8.1上崩溃:

NSString *base64String = @"5a+55LqOTGF1bmNoZXLov5nnsbvkuqflk4HmnaXor7TvvIzlroPlvojlrrnmmJPorqnkurrku6zpmbflhaXov5nmmK/lt6Xlhbfov5jmmK/lubPlj7DnmoTkuonmiafkuK3jgILkuI3ov4flnKjmnY7mtpvnnIvmnaXvvIzov5nnp43kuonmiaflrozlhajmmK/kuIDkuKrkvKrlkb3popjvvIzlm6DkuLrkuIDmrL7kuqflk4HnlKjnmoTkurrlpJrkuoblroPoh6rnhLblsLHmmK/lubPlj7DvvIznlKjnmoTkurrlsJHkuoblroPku4DkuYjpg73kuI3mmK/jgILln7rkuo7mraTvvIzmnY7mtpvlhbblrp7lubbmsqHmnInov4flpJrnmoTljrvogIPomZFBUFVTIExhdW5jaGVy6KaB5YGa5bmz5Y+w6L+Y5piv5bel5YW377yM5LuW5oOz55qE5pu05aSa55qE5piv5aaC5L2V6Kej5Yaz55So5oi355qE6Zeu6aKY44CC5L2c5Li65LiA5Liq5Y2z55SoaU9T5Y+I55SoQW5kcm9pZOeahOeUqOaIt++8jOaIkeacrOS6uueahOS4gOS4quS9k+S8muWwseaYr2lQaG9uZeS8mue7meS6uuS4gOenjeS9oOi2iueUqOi2iuinieW+l+Wug+WlveeUqOeahOaEn+inie+8jOS9hkFuZHJvaWTlsLHkuI3kvJrjgILmiYDku6VBUFVTIExhdW5jaGVy546w5Zyo5bCx6KaB6Kej5Yaz6L+Z5Liq6Zq+6aKY77yM6K6pQW5kcm9pZOWPmOW+l+WlveeUqOOAgui/meS5n+aYr+S4uuS9leadjua2m+S8muivtOiHquW3seWBmueahOS4jeaYr+S4gOS4qkxhdW5jaGVy6ICM5piv5LiA5aWX4oCc55So5oi357O757uf4oCd44CC";
NSData *decodedData = [[NSData alloc] initWithBase64EncodedString:base64String options:0];
NSString *str = [[NSString alloc] initWithData:decodedData encoding:NSUTF8StringEncoding];
str = [str stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLFragmentAllowedCharacterSet]];

Example from https://github.com/Alamofire/Alamofire/issues/206 will crash on iOS 7 to iOS 8.2: 来自https://github.com/Alamofire/Alamofire/issues/206的示例将在iOS 7到iOS 8.2崩溃:

let str = String(repeating: "一二三四五六七八九十", count: 2_000)
var allowedCharacterSet = CharacterSet.urlQueryAllowed
allowedCharacterSet.remove(charactersIn: ":#[]@!$&'()*+,;=")
_ = str.addingPercentEncoding(withAllowedCharacters: allowedCharacterSet)

The identification of a workaround for the memory issue was provided by PrideChung in https://github.com/Alamofire/Alamofire/issues/206 . PrideChung在https://github.com/Alamofire/Alamofire/issues/206中提供了针对内存问题的解决方法的标识。 The following description was subsequently given by cnoon: cnoon随后给出了以下描述:

After much debugging, I was able to track this issue down to only occurring in Alamofire on iOS 8.1 and 8.2 using the iPhone 4S and iPhone 5 simulators. 经过大量调试后,我能够使用iPhone 4S和iPhone 5模拟器跟踪到仅在iOS 8.1和8.2的Alamofire中发生的问题。 It is 100% reproducible, but is crashing in different ways depending on the size of the Chinese string that is passed in. It's always some form of a malloc error. 它是100%可重现的,但是根据传入的中文字符串的大小,崩溃的方式也不同。它总是某种形式的malloc错误。

[...] [...]

Batching is required for escaping due to an internal bug in iOS 8.1 and 8.2. 由于iOS 8.1和8.2中存在内部错误,转义必须进行批处理。 Encoding more than a few hundred Chinese characters causes various malloc error crashes. 编码数百个汉字会导致各种malloc错误崩溃。 To avoid this issue batching MUST be used for encoding. 为避免此问题,必须将批处理用于编码。

And my actual workaround, inspired by AlamoFire, is: 受AlamoFire启发,我的实际解决方法是:

extension String {
    //  Due to an internal bug in iOS 7.x, 8.1 and 8.2, encoding more
    //  than a few hundred Unicode characters causes various malloc error crashes.
    //  To avoid this issue, batching MUST be used for encoding.
    //      - https://github.com/Alamofire/Alamofire/issues/206
    //      - https://stackoverflow.com/a/44309416/1033581
    func safeAddingPercentEncoding(withAllowedCharacters allowedCharacters: CharacterSet) -> String? {
        if #available(iOS 8.3, *) {
            return addingPercentEncoding(withAllowedCharacters: allowedCharacters)
        } else {
            let batchSize = 50
            var batchPosition = startIndex
            var escaped = ""
            while batchPosition != endIndex {
                let range = batchPosition ..< (index(batchPosition, offsetBy: batchSize, limitedBy: endIndex) ?? endIndex)
                guard let percentEncodedSubstring = substring(with: range).addingPercentEncoding(withAllowedCharacters: allowedCharacters) else {
                    return nil
                }
                escaped.append(percentEncodedSubstring)
                batchPosition = range.upperBound
            }
            return escaped
        }
    }
}

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM