- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我们一直在构建一个大型开源 software在各种平台(Linux、Windows、Mac OS X、32 位和 64 位)上运行数年都没有问题。然而最近,Mac OS X 版本(64 位)停止正常工作并开始随机崩溃。它或多或少与我们构建机器上的 Mac OS X 从 10.7 更新到 10.8.2 相吻合(但编译器工具链没有改变,它仍然是 llvm-gcc 4.2.1)。
我们的应用程序由几个动态(共享)库和许多使用它们的可执行文件组成。共享库之一覆盖了 new
和 delete
运营商出于各种原因。在 Mac OS X(和 Linux)上,默认导出所有符号,包括我们重载的 new
和 delete
运营商。 Mac OS X 上的崩溃似乎与一些内存分配了一个内存子系统(不是我们的)然后通过我们自己的(和不兼容的)释放了 delete
实现。
最明智的解决方案似乎是防止重载运算符对共享库的用户可见。这可以通过两种方式完成:用 __attribute__((visibility("hidden")))
标记运算符。 ,或使用 -unexported_symbols_list
链接器命令行选项,以防止某些符号被导出。不幸的是,第一个解决方案不起作用:gcc 发出警告说运算符的声明不同(在 <new>
中),因此属性将被忽略。从我在不同地方的阅读来看,第二种解决方案似乎是解决这个问题的正确方法。但是由于某些原因我们无法让它工作。
链接共享库时,我们传递了 -Wl,-unexported_symbols_list unexported_symbols_list.txt
g++ 的选项,它又应该传递给 ld。 unexported_symbols_list.txt
文件包含以下符号列表:
__ZdaPv
__ZdaPvRKSt9nothrow_t
__ZdlPv
__ZdlPvRKSt9nothrow_t
__ZdlPvS_
__Znam
__ZnamRKSt9nothrow_t
__Znwm
__ZnwmPv
__ZnwmRKSt9nothrow_t
这些都是 new
的变体和 delete
我们覆盖并希望隐藏。我们通过执行 nm libappleseed.dylib
找到了这些符号然后使用 c++filt
解开符号名称.
这是 CMake 生成的用于链接 libappeseed.dylib
的命令行:
/usr/bin/g++ -g -Werror -dynamiclib -Wl,-headerpad_max_install_names -framework Cocoa -lcurl -Werror -Wl,-unexported_symbols_list -Wl,unexported_symbols_list.txt -o ../mac-gcc4/appleseed/libappleseed.dylib [...]
不幸的是,尽管我们做出了所有努力,但符号似乎仍然存在(如 nm 所示)。
知道我们做错了什么吗?我们可以尝试另一种方法吗?
2012 年 12 月 19 日更新:
我们的问题和假定的解决方案在 Apple 的这份技术说明中得到了很好的涵盖:http://developer.apple.com/library/mac/#technotes/tn2185/_index.html (“覆盖新建/删除”部分)。
指向相关源代码的指针:
operator new
和 operator delete
覆盖:allocator.cpp nm
的片段使用 -fvisibility=hidden
构建 libappleseed.dylib 后的输出正在运行 strip -x libappleseed.dylib
:
...
00000000002a41b0 T __ZdaPv
00000000002a41f0 T __ZdaPvRKSt9nothrow_t
00000000002a4190 T __ZdlPv
00000000002a41d0 T __ZdlPvRKSt9nothrow_t
00000000002a4060 T __Znam
00000000002a4130 T __ZnamRKSt9nothrow_t
00000000002a3ff0 T __Znwm
00000000002a40d0 T __ZnwmRKSt9nothrow_t
...
最佳答案
您应该使用 -fvisibility=hidden
进行构建,然后仅导出您想要的内容。在这里阅读:
http://gcc.gnu.org/wiki/Visibility
它还解释了 -fvisibility-inlines-hidden
。许多大型库(例如 Qt)都使用它。好处是相当可观的。
关于c++ - 在 Mac OS X 上的共享库中隐藏符号,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13793325/
在几个 SO 的问题中,有这些行可以访问代码的父目录,例如os.path.join(os.path.dirname(__file__)) returns nothing和 os.path.join(o
我想用 Python 更改文件模式。 os 模块具有三个功能上看似相同的功能: os.chmod os.fchmod os.lchmod 这三个版本有什么区别? 最佳答案 chmod 用于更改路径指定
考虑: pipe_read, pipe_write = os.pipe() 现在,我想知道两件事: (1) 我有两个线程。如果我保证只有一个正在读取 os.read(pipe_read,n) 而另一个
这个问题不太可能帮助任何 future 的访问者;它只与一个小的地理区域、一个特定的时间点或一个非常狭窄的情况有关,这些情况并不普遍适用于互联网的全局受众。为了帮助使这个问题更广泛地适用,visit
让我们以硬盘驱动器/网络接口(interface)为例。它由多个进程共享。现在多个进程可能会向硬盘驱动器发出并发命令来读取数据。当数据可用时,内核如何知道哪个进程的数据已准备好?操作系统和硬盘驱动器之
嗨,我正在尝试编写像这样的原子写入函数...... with tempfile.NamedTemporaryFile(mode= "w", dir= target_directory) as f:
net.Conn接口(interface)提供了 SetTimeout 方法,我应该用 os.Timeout 检查返回的错误.但是我看不到在返回的 os.Error 上调用 os.Timeout 的方
我正在使用 os 模块在我的 Django 项目 settings.py 文件中具有相对路径。变量 SITE_ROOT 设置为 settings.py 文件的当前工作目录,然后用于引用同样位于同一目录
正如我们所知,Windows 接受 "\" 和 "/" 作为分隔符。但是在python中,使用的是"\"。例如,调用 os.path.join("foo","bar"),将返回 'foo\\bar'。
我有以下工作目录:/Users/jordan/Coding/Employer/code_base ,我想要获取绝对路径的文件位于 /Users/jordan/Coding/Employer/code_
在 Python 中,如果路径中包含“~”,我能否确定扩展的用户调用将是绝对路径? 例如,这个表达式是否总是为真? path = '~/.my_app' os.path.expanduser(path
我是 Django 项目的初学者。Django 项目的 settings.py 文件包含这两行: BASE_DIR = os.path.dirname(os.path.dirname(os.path.
我有一个旧 MAC OS 文件存储中的文件集合。我知道集合存在文件名/路径名问题。问题源于我认为在原始操作系统中呈现为破折号的路径中包含一个代码点,但 Windows 与代码点斗争,并且其中一个包含
Ubuntu怎么安装mac os x主题呢?下文小编将为大家分享ubuntu14.04安装mac os x主题教程,安装MAC OS X&
我有一个 Firefox OS 应用程序,我希望在该应用程序之外打开一个链接(该链接指向不同的站点,在应用程序中打开它会使应用程序在没有强制的情况下无法使用)。我怎么做? Related bug re
我想为 Firefox OS 编写我的应用程序.使用什么样的语言(如 Android 的 Java 和 iOS 的 Objective C++)和工具(如 Eclipse、Xcode)? 最佳答案 适
我正在尝试创建一个 Palm OS 应用程序,以每 X 分钟或几小时检查一次网站,并在有数据可用时提供通知。我知道这种事情可以在新的 Palm 上完成——例如,当应用程序不在顶部时,我的 Centro
我需要在 Firefox OS 中显示全屏图像。我有一个具有 qHD 分辨率(960x540 像素)的“峰值”开发预览手机。 如何确保我的应用程序在其他具有不同屏幕分辨率的 firefox-os 设备
我正在尝试在 Firefox OS 中安装一个新的语言环境,但我不确定我是否正确地按照这些步骤操作。 首先,我尝试使用 Mercurial 下载所需的语言环境:它对我不起作用,Mercurial 说访
我有这个shell脚本Test.sh: #! /bin/bash FILE_TO_CHECK="/Users/test/start.txt" EXIT=0 while [ $EXIT -eq 0 ];
我是一名优秀的程序员,十分优秀!