百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

服务器程序从 Windows 系统移植到 Linux/x86_64 平台时总是崩溃?

haoteby 2025-06-30 16:18 2 浏览

清华大学出版社的《高效 C/C++调试》一书给出了回答:
我们的服务器程序最初是为 Windows 系统开发的, 第一次将它移植到 Linux/x86_64 平台时,它在回归测试中十次会崩溃八次,且仅在多处理器机器上出现这种现象。这种症状强烈地暗示着存在多线程问题。核心转储文件显示,系统库 libc.so 的内存管理器中的函数 malloc_consolidate 内部出现了段错误。考虑到该函数正尝试合并相邻的空闲内存块,我们怀疑应用程序可能以某种方式破坏了堆。

我们打开了内存管理器的调试功能,方法是设置环境变量 MALLOC_CHECK_=2(参见第 10章)。内存调试器捕获到了一个问题—一个内存块被释放了两次,这导致程序崩溃。我们确定了这个内存块上的数据对象,发现它被多个线程共享。然而,对象的所有权是通过引用计数严格控制的,这是共享数据的标准实现方式。当引用计数减少到 0 时,对象将被删除。这个处理在其他平台上得到了良好的执行并且已经通过了全面的测试。那么,为什么在新的平台上会出现问题呢?为了验证引用计数是否按照设计工作,我们在修改对象引用计数的函数中插入了日志语句。

class AtomicLong { public: ... long operator++() throw() { return AtomicIncrement(&mnAtomicLongData); } long operator--() throw() {return AtomicDecrement(&mnAtomicLongData); } ... private: long mnAtomicLongData; }

virtual void AddRef() const throw(void* ipObject, AtomicLong& irRefCount) { RefCountLog("AddRef", ipObject, irRefCount); ++irRefCount; }

virtual void Release(void* ipObject, AtomicLong& irRefCount)const throw() { RefCountLog("Release", ipObject, irRefCount); if (--irRefCount == 0) {

// 这是最后一次引用free(ipObject);

} }

void RefCountLog(const char* ipFunc, void* ipObject, long iRefCount) { FILE* lpFile = GetLogFile(); fprintf(lpFile, "tid= %ld\tFunction=%s\tObjectID=0x%lx”

“\tRefCount=%ld\n",

pthread_self(), ipFunc, ipObject, iRefCount);

// 检索调用者的堆栈跟踪

const int max_stack_depth = 32; void* lTrace[max_stack_depth]; int lTraceSize = ::backtrace(lTrace, max_stack_depth); _ASSERT(lTraceSize > 0);

// backtrace_symbols 分配一块内存来存储符号

char** lpFunctionNames = ::backtrace_symbols(lTrace, lTraceSize); for(int i=0; i<lTraceSize; i++) { _ASSERT(lpFunctionNames[i] != NULL); fprintf(lpFile, "\t[%d] %s\n", i, lpFunctionNames[i]); } }

日志函数记录了发起调用的线程 ID、回溯、对象的地址、当前的引用计数以及对对象的操作。当问题再次出现时,我们在日志文件中得到了一系列的记录。通过脚本解析这些输出,发现 AddRef和 Release 函数被调用的次数与设计时预期的一致,但对象的引用计数并未反映出这一点。这提示我们引用计数器本身(即 AtomicLong 类中的长整数)的递增和递减操作可能存在问题。这个类的递增和递减运算符是通过调用 AtomicIncrement 和 AtomicDecrement 函数实现的,这两个函数都是用汇编语言编写的, 这些函数本应具有原子性和线程安全性。但是,对这些函数的进一步代码审查引发了我们对它们在新平台上执行的正确性的怀疑。经过更严格的单元测试,我们发现在高负载下,这些函数确实存在问题。修复这个错误后,测试就一直运行得很好。

因此,在这种情况下,日志记录的方法帮助我们将问题定位到了引用计数器的原子递增和递减代码上。

相关推荐

JAVA零基础入门:JDK的概述及安装(jdk完整安装教程)

一.什么是jdkJDK(JavaDevelopmentToolKit)是Java开发工具包,JDK是整个JAVA的核心,包括了Java运行环境(JavaRuntimeEnvirnment),一...

开源、强大的工作流引擎:camunda入门介绍

原创不易,请多多支持!对Java技术感兴趣的童鞋请关注我,后续技术分享更精彩。简介CamundaisaJava-basedframeworksupportingBPMNforwork...

Centos8搭建Java环境(JDK1.8+Nginx+Tomcat9+Redis+Mysql)

一、开篇1.1目的每次换新的服务器,都要找资料配下环境,所以我写这篇文章,重新梳理了一下,方便了自己,希望也能给大家带来一些帮助。安装的软件有:JDK1.8+Nginx+Tomcat9+...

记录一次tomcat的升级过程(tomcat6升级tomcat8)

原因:ApacheTomcat资源管理错误漏洞(CVE-2021-42340)版本:ApacheTomcat/9.0.46,tomcat解决方法:升级tomcat9到最新版本9.0.581.官...

Tomcat10安装与配置图文教程(tomcat安装及配置)

Tomcat10安装与配置图文教程1、百度搜索“tomcat下载”,进入官网下载https://tomcat.apache.org/index.html...

VS2022配置x86/x64调用32位和64位汇编语言动态库环境

配置X86MASM汇编环境1.创建项目打开VS2022创建新项目,新建asm文件(注意要手动修改cpp文件后缀名为asm文件后缀名)。2.设置入口点选择菜单栏中的“调试”-“demo调试属性”-...

ARM版Win10用户狂喜 微软全新补丁让应用不再不兼容

Windows10onARM仅支持模拟32位的X86应用程序,这意味着大多数的桌面应用是无法在这一平台上运行的,这在很大程度上限制该平台的发展。为了解决这一问题,微软在内部开发频道推出可用于AR...

分享收藏的 oracle 11.2.0.4各平台的下载地址

概述oracle11.2.0.4是目前生产环境用的比较多的版本,同时也是很稳定的一个版本。目前官网上已经找不到下载链接了,有粉丝在头条里要求分享一下下载地址。一、各平台下载地址...

Android-x86现已基于5.1.1 Lollipop:支持UEFI和64位内核

采用Linux内核的Android-x86,旨在为PC带来最新的Android移动操作系统体验。而近日,该操作系统已经发布了Android-x865.1的首个候选发布(RC)版本。发行说明中提到:A...

Linux Kernel源码阅读: x86-64 系统调用实现细节(二)

特别说明:该文章前两天发布过,但一直在审核中。看头条网友说字数太多可能一直处于审核中状态,我把该文章拆分成几个章节发布,如影响阅读体验还请见谅。五、系统调用编号...

树莓派4B安装win10后实测,CPU秒杀AMD Athlon64 3200+

在上一篇文章介绍了如何给树莓派4B安装win10系统,这篇就简单对系统进行测试,上一篇文章链接https://www.toutiao.com/i7015518822056886821/因为树莓派是a...

一键离线部署x86、arm64 RabbitMQ,花了2天去验证整理,直接拿去

最近有一个项目,客户是内网网络,只能离线部署,采用的麒麟ARM64服务器系统,不能远程部署,需要提前准备离线部署包让客户IT拷备上去再现场部署,部署时间就只有1天。自家系统采用的vue+springb...

Linux软件包管理(linux系统软件包的安装方法,并简要说明其特点)

Linux系统如果需要安装软件怎么办?如何安装,大概有以下几种方式1.二级制软件包管理(RPM、YUM)...

Tachyum要做全球最强64位处理器:性能比X86强,面积比ARM小

全球半导体芯片研发、生产最强的国家非美国莫属,如果有某家美国公司宣布要开发性能超强的芯片,大家不会意外,但要是一家斯洛伐克初创公司宣布要研发超级芯片呢?Tachyum公司就是这样一家公司,成立于201...

Android L 64位模拟器终于来了:x86独享

GoogleI/O2014大会已经过去了很久,64位的AndroidL依然停留在纸面上,但现在至少可以让开发者们先行品尝品尝了:64位的AndroidL模拟器已经发布。这次公布的模拟器镜像是专...