IT技术博客大学习 共学习 共进步
首页 / 博学无忧
IT 2014-03-19 23:04:03 / 累计浏览 3,780

linux下cp,mv进行动态库覆盖问题分析

这篇讲的是一个生产环境中典型的“隐形地雷”:明明只是想更新一个动态库文件(.so),为什么用 cp 命令覆盖后,正在运行的程序就莫名崩溃了?作者从一次周会讨论的问题出发,抽丝剥茧地分析了这个现象的深层原因。 文章先厘清了 Linux 文件系统的两个关键概念:inode 存储文件元数据,dentry 建立文件名到 inode 的映射。核心分析由此展开:cp 和 mv 操作对正在被进程使用的文件影响截然不同。cp 命令本质是“打开目标文件并截断,然后写入新数据”,这个截断操作会通知内核释放该文件在内存中的所有页面映射。当程序再次执行到该库的代码时,会触发缺页中断,但因为原文件数据块已失效,内核无法正确填充内存,于是导致总线错误(bus error)或段错误(segment fault)。相比之下,mv 命令仅仅是重命名操作,只改变了目录项(dentry)的指向,并未影响文件本身(inode)及其内存映射,因此是安全的。 文章通过 strace 跟踪和进程内存映射的视角,清晰展示了故障现场。结论很明确:对于已被进程动态链接的库文件,在线更新的正确姿势是 mv(将新文件重命名为原名),而不是 cp。

IT 2014-03-19 22:44:47 / 累计浏览 3,940

怎样用core文件调试你的linux程序?

这篇讲的是如何配置Linux系统,让它能在程序异常崩溃时自动生成核心转储(core dump)文件,从而方便你找出程序崩溃的具体原因。 作者从默认Linux禁止生成core文件这个常见限制出发,一步步演示了解锁方法。核心是使用`ulimit -c unlimited`命令,但文章也特别指出了它的临时性——设置仅对当前会话有效,重启或重登就会失效。如果想要更持久的配置,可以修改`/etc/profile`,不过作者也留下了思考:为什么不推荐这样做呢? 更深入的配置在于控制core文件“生在哪里”和“叫什么名字”。文章详细讲解了通过编辑`/etc/sysctl.conf`文件,设置`kernel.core_pattern`参数来实现。例如,将核心文件统一生成到`/tmp`目录,并使用包含程序名、进程ID、信号值等信息的规则来命名,这对于同时调试多个程序非常方便。 最后,文章点明了核心文件的归宿:使用强大的gdb调试器载入这个文件,就能回溯程序崩溃现场,定位问题代码。整个流程非常实用,是每个Linux开发者都应该掌握的调试技巧。

IT 2013-07-28 15:46:01 / 累计浏览 3,500

php扩展中如何定义线程安全的全局对象

作者在开发一个PHP图片裁剪扩展(tclip)时,为了提升效率,希望将OpenCV的人脸识别分类器(CascadeClassifier)作为全局对象,在扩展初始化时加载一次供后续重复使用。这引出了一个具体的技术问题:如何在PHP扩展中定义线程安全的全局对象。 文章直击踩坑点:最初尝试直接在模块全局变量中声明CascadeClassifier对象,却遭遇了“‘CascadeClassifier’ does not name a type”的编译错误,因为该类并未在对应的头文件中引入。作者调整方案,改用`void *`指针来声明全局变量,并在C源文件中定义一个静态的CascadeClassifier实例。在模块初始化函数(`PHP_MINIT_FUNCTION`)中完成配置文件的加载与实例化后,将实例的地址赋给那个`void *`类型的全局指针。 核心的解决思路在于,使用`void *`绕开了头文件依赖问题,并将对象实例的实际生命周期管理在静态作用域。而在扩展的实际函数中使用时,则需要通过`TCLIP_G()`宏访问该指针并强制类型转换回`CascadeClassifier *`。文章特别指出,要确保扩展的线程安全,就必须通过这类模块全局变量宏来访问数据。整个过程为在PHP C扩展中安全复用大型配置对象提供了一个可行的实践路径。