背景

项目中使用protobuf作为网络传输协议,最开始在项目中直接使用源代码编译,在真机上测试一直是正常的,直到某天开始在 CPU是64 bit的设备上发现protobuf导致crash了,于是就开始尝试使用.a静态库看看是否能解决问题(失败了)。bug解决方案直接看最后。

开发环境

  • OS X 10.9 Mavericks
  • Xcode 5.0.2
  • protobuf 2.5.0
  • IOS7

Crash现象

protobuf在静态初始化的时候,crash在一个delete[] XXXptr; 这个指针是一个NULL。非常奇怪在32bit的ios设备上同样也是执行到这里也是delete[] nullptr就没有问题。简单搜索了一下,没有什么收获,遂暂时放弃直接解决这个问题。

  1. template <typename TypeHandler>
  2. void RepeatedPtrFieldBase::Destroy() {
  3. for (int i = 0; i < allocated_size_; i++) {
  4. TypeHandler::Delete(cast<TypeHandler>(elements_[i]));
  5. }
  6. delete [] elements_;  // elements_ == NULL
  7. }

尝试编译.a

略过大堆因为Xcode升级而无效/需修改的方法,直接上目前找到最靠谱的一个全自动脚本:https://gist.github.com/BennettSmith/7150245

这个脚本实现了:

  • 下载protobuf 2.5.0源代码
  • 生产5个平台的静态库(mac 64, iPhone simulator, armv7, armv7s, arm 64)
  • 合并成一个完整的Universal静态库。

编译是成功的,可能对于大部分人来说,到这里就结束了。

但是。。。在将生成的.a加入工程中依然出现了std库找不到的link错误。

静态库编辑成功

调整了除mac64平台外的CXX编译参数,和项目中使用的参数匹配之后,终于成功了。

最终使用的参数如下,(mac64的保持不变)

    • CXX=clang++
    • CXXFLAGS_ARM="${CFLAGS} -stdlib=libstdc++"
    • LDFLAGS_ARM="-stdlib=libstdc++"

但是。。。在64bit设备上依然Crash,现象同上。可以宣布这个尝试方向失败了!

问题解决

很意外的发现,crash的地方是Libproto.dylib,这个库是Apple自带的,而项目中是没有使用到这个库的。Google了一 下:“ImageLoader on the newer devices (iPhone 5S and apparently iPad air) has its own copy of Protocol Buffers which causes symbol collisions.",也就是说在最新的3个设备上有个库自带了一个Protobuf,和项目中的protobuf的namespace是一样的,这 样就就产生了冲突。
办法就很简单了:
  1. 使用protobuf源代码(正好这项目原来就使用源代码)
  2. 修改google\protobuf\stubs\common.h,加入一行宏定义,修改本地的命名空间
  1. namespace std {}
  2. #define google google_private
  3. namespace google {
  4. namespace protobuf {

相关信息

Xcode 5.0.2的一些默认参数

  • C Language Dialect:GUN99(-std=gnu99)
  • C++ Language Dialect:GNU++98(-std=gnu++98)
  • C++ Standard Library:libstdc++(-stdlib=libstdc++)

StackOverflow上的回答:http://stackoverflow.com/questions/19848118/weird-ios-libprotobuf-dylib-cause-crash

【转】protobuf2.5.0在<delete [] elements_;>crash的问题。的更多相关文章

  1. [转]protobuf-2.5.0.tar.gz的下载与安装

    protobuf-2.5.0.tar.gz的下载与安装 原文地址:http://blog.csdn.net/tdmyl/article/details/31811317 版权声明:本文为博主原创文章, ...

  2. 编译protobuf-2.5.0中的错误处理

    最近在编译protobuf-2.5.0源码的时候发现的错误已经应对方法 1. 在源码目录执行 ./configure 命令的时候,发生如下错误 error: C++ preprocessor &quo ...

  3. protobuf-2.5.0.tar.gz的下载与安装

    1.下载 hadoop使用protocol buffer进行通信,须要下载和安装protobuf-2.5.0.tar.gz.因为如今protobuf-2.5.0.tar.gz已经无法在官网https: ...

  4. protobuf-2.5.0的下载与安装

    1.下载 Hadoop使用protocol buffer进行通信,需要下载和安装protobuf-2.5.0.tar.gz.由于现在protobuf-2.5.0.tar.gz已经无法在官网https: ...

  5. linux编译安装protobuf2.5.0

    1.下载安装包 https://github.com/google/protobuf/releases?after=v3.0.0-alpha-4.1 找到相应的版本下载 2.解压安装包 #.tar.g ...

  6. 深入解读MySQL8.0 新特性 :Crash Safe DDL

    前言 在MySQL8.0之前的版本中,由于架构的原因,mysql在server层使用统一的frm文件来存储表元数据信息,这个信息能够被不同的存储引擎识别.而实际上innodb本身也存储有元数据信息.这 ...

  7. 低于0.01%的极致Crash率是怎么做到的?

    WeTest 导读 看似系统Bug的Crash 99%都不是系统问题!本文将与你一起探索Crash分析的科学方法. 在移动互联网闯荡多年的iOS手机管家,经过不断迭代创新,已经涵盖了隐私(加密相册). ...

  8. Hadoop3.2.0使用详解

    1.概述 Hadoop3已经发布很久了,迭代集成的一些新特性也是很有用的.截止本篇博客书写为止,Hadoop发布了3.2.0.接下来,笔者就为大家分享一下在使用Hadoop3中遇到到一些问题,以及解决 ...

  9. 2690036 - SAP HANA 2.0 SPS 03 Database Revision 034

    Symptom This is the SAP Release Note for SAP HANA 2.0 Database Revision 034 (2.00.034.00) of the SAP ...

随机推荐

  1. Oracle分析函数 — rank, dense_rank, row_number用法

    本文通过例子演示了Oracle分析函数 —— rank, dense_rank, row_number的用法. //首先建score表 create table score( course   nva ...

  2. 2、[转]WPF与WinForm的比较

    http://www.cnblogs.com/KnightsWarrior/archive/2010/07/09/1774059.htmlhttp://www.cnblogs.com/zenghong ...

  3. Oracle数据库间的数据复制 - SQLPlus中的COPY命令

    Copy命令可以实现不同Oracle数据库间的数据的复制,也是可以实现同一数据库的数据复制,其性能表现和导入/导出相同. 根据9i文档,说Copy命令未来会不支持,但实际上Oracle 11g仍然支持 ...

  4. WebApi2 jsonp跨域问题

    一:故事背景     以前在写WebApi2的时候,一直是用作前后端分离(WebApi2 +angularjs),可是最近自己在给WebApp写接口的时候遇到了很多坑,总结一下就是跨域问题.而跨域问题 ...

  5. First Groovy

    class Sample { def names = ["anna", "annie", "tommy", "bobby" ...

  6. shell复习笔记----命令与参数

    shell最基本的工作就是执行命令. 每键入一道命令, shell 就会执行. $cd work;ls -l whizprog.c 首先:格式很简单,以空白(Space 键或者 Tab键)隔开命令行中 ...

  7. HTTP POST GET 区别

    一 原理区别 一般在浏览器中输入网址访问资源都是通过GET方式:在FORM提交中,可以通过Method指定提交方式为GET或者POST,默认为GET提交 Http定义了与服务器交互的不同方法,最基本的 ...

  8. mysql可以运行在不同sql mode模式下面,sql mode模式定义了mysql应该支持的sql语法,数据校验等

    查看默认的sql mode模式:select @@sql_mode;我的数据库是:STRICT_TRANS_TABLES,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUT ...

  9. AAC ADTS AAC LATM 格式分析

    http://blog.csdn.net/tx3344/article/details/7414543# 目录(?)[-] ADTS是个啥 ADTS内容及结构 将AAC打包成ADTS格式 1.ADTS ...

  10. Servlet课程0425(七) 到数据库中去验证用户,同时防止SQL注入漏洞

    Login.java //登录界面 package com.tsinghua; import javax.servlet.http.*; import java.io.*; public class ...