当前位置: 首页 > article >正文

GlibC 在线程里引发use-after-free退出时才崩溃原因与分析

背景最近在做开发时遇到了一个错误tcache_thread_shutdown(): unaligned tcache chunk detected这个错误是发生在多线程情况下没有做好互斥时发生一个指针被free之后再次使用也就是比较经典的use-after-free的错误触发了Abort虽然不是什么大问题很容易修复但是这引发了我一个比较好奇的事情就是为什么会这样为什么free之后继续使用没报错但是线程退出时导致崩溃了原理我使用的是GCC编译器标准库是Glibc对于Glibc而言每次malloc的内存都要比预期申请的要大一点因为里面要放入一些malloc的元数据基本上malloc的内存布局是这样的[malloc chunk header][用户区64bytes]^|pp是返回给我们使用的user地址每次当我们释放时Glibc并不会把它归还给系统而是自己维护一个内存chunk把这次的Malloc内存放入一个tcache链表维护起来这样后续在使用可以从这里拿内存提高分配效率这里就有一个点就是如果没有归还给系统的话页表的映射还在当前进程上面也就是说你free之后依然可以随意访问这块地址并不会发生什么非法地址访问的情况下面是一个示例#includestdio.h#includestdlib.hintmain(void){char*pmalloc(64);if(!p){return1;}free(p);p[0]0x01;p[1]0x02;printf(hello world!\n);return0;}此时hello world会正常打印并且也不会出现任何问题这种做法与我们平时接触到的C语言教学背道而驰平常教科书上的教学是针对C语言这门语言特性而Glibc是C语言的标准实现库它有自己的行为并且在不同系统上也有不同区别但是这里我们换一种方式用线程来执行#includepthread.h#includestdio.h#includestdlib.h#includestring.h#includeunistd.hstaticvoid*worker(void*arg){char*amalloc(64);printf([worker] a%p\n,a);free(a);memset(a,0x55,64);printf([worker] corrupted freed chunk, now return\n);returnNULL;}intmain(void){pthread_ttid;if(pthread_create(tid,NULL,worker,NULL)!0){perror(pthread_create);return1;}pthread_join(tid,NULL);printf([main] worker joined\n);return0;}运行结果[worker]a0x79d9f0000b70[worker]corrupted freed chunk, nowreturntcache_thread_shutdown(): unaligned tcache chunk detected[1]29787IOT instruction(core dumped)./a.out可以看到被异常结束了是Glibc主动调用了Abort函数发送SIGABRT异常信号结束当前进程并且打印了tcache_thread_shutdown()这个原因是C语言的线程检查行为对于Main而言它虽然也是一个线程但是Glibc对它行为是它是主线程结束时Glibc调用exit来回收它与普通线程行为不同Main的链路是_start- __libc_start_main(…)- 调用main(argc, argv, envp)- main return- exit(main_return_value)- 执行 atexit/on_exit 回调- flush/close stdio- glibc 内部清理- _exit(status)- exit_group 系统调用- 内核销毁整个进程普通线程行为调用的是pthread_create()- glibc clone创建线程- 新线程从glibc 的 start_thread入口开始- start_thread 调用你的 thread_func(arg)- thread_func return- glibc 执行 pthread_exit 收尾逻辑- 清理当前线程资源- 当前线程结束注意这里调用的是pthread_exit而不是系统的exit这里它会去调用tcache_thread_shutdown来检查chunk有没有被破坏这点你可以在它的源码里面找到这里我找的是2.xmalloc/malloc.c的源码staticvoidtcache_thread_shutdown(void){……for(i0;iTCACHE_MAX_BINS;i){while(tcache_tmp-entries[i]){tcache_entry*etcache_tmp-entries[i];if(__glibc_unlikely(!aligned_OK(e)))malloc_printerr(tcache_thread_shutdown(): unaligned tcache chunk detected);tcache_tmp-entries[i]REVEAL_PTR(e-next);__libc_free(e);}}……}可以看到它在循环调用aligned_OK来检查tcache里面的指针是否被破坏的也就是判断是不是Malloc合法边界这个是看当前CPU位数64位是16字节32位是8字节用MALLOC_ALIGNMENT宏表示在不同平台位数上它的内存颗粒不同它会多分配一些字节一部分用于维护Malloc元数据一部分给用户使用在64位上即使你分配没有16个字节它仍然会将你的分配按16字节对齐如果tcache记录的地址里面有一块不是按16字节对齐的那么就会立马出错它认为当前线程破坏了tcache导致内存模型出现了问题可能引发异常错误就会直接abort在malloc_printerr里面会调用abort。因此当我们平时调试的时候不要总是依靠C语言的标准答案而是去参考具体平台和标准实现细节例如对着Glibc库源码分析原因。Main之所以没有做这层检查是因为Main结束整个应用都要结束了因此没必要在做这些事情了直接调用exit让系统一并将这个task全部清理掉因为它所有的内存数据这些都在task_struct里被记录着内核一并释放掉就可以了。

相关文章:

GlibC 在线程里引发use-after-free退出时才崩溃原因与分析

背景 最近在做开发时遇到了一个错误tcache_thread_shutdown(): unaligned tcache chunk detected,这个错误是发生在多线程情况下没有做好互斥时发生一个指针被free之后再次使用也就是比较经典的use-after-free的错误,触发了Abort,虽然不是什么…...

软件测试核心知识点梳理(包括黑盒测试,白盒测试,抽卡,通行证测试用例等)

一、黑盒测试与白盒测试核心区别黑盒测试:关注软件 “做什么”(外部行为),不关心内部结构,如同测试一个密封的黑盒子。白盒测试:关注软件 “怎么做”(内部逻辑),需要查看…...

5分钟快速上手:wxappUnpacker微信小程序逆向工程完整指南 [特殊字符]

5分钟快速上手:wxappUnpacker微信小程序逆向工程完整指南 🚀 【免费下载链接】wxappUnpacker forked from https://github.com/qwerty472123/wxappUnpacker 项目地址: https://gitcode.com/gh_mirrors/wxappu/wxappUnpacker 想要深入了解微信小程…...

ThinkPad风扇控制终极指南:用TPFanCtrl2实现智能散热与静音平衡

ThinkPad风扇控制终极指南:用TPFanCtrl2实现智能散热与静音平衡 【免费下载链接】TPFanCtrl2 ThinkPad Fan Control 2 (Dual Fan) for Windows 10 and 11 项目地址: https://gitcode.com/gh_mirrors/tp/TPFanCtrl2 你是否厌倦了ThinkPad风扇在轻负载时的频繁…...

RAG 系列(九):效果不好怎么定位——用 RAGAS 做根因诊断

"感觉效果不太好"不是诊断 你部署了一个 RAG 系统,用户反馈说"答案有时候不准"。 然后呢?你改了 Prompt,感觉好一点。再换了个 Embedding 模型,又好了一点。几轮下来,你也不知道是哪一步起了效果…...

CANoe/CAPL实战:模拟ECU端,完整响应UDS刷写(34/36/37服务)的保姆级脚本教程

CANoe/CAPL实战:构建高仿真ECU响应UDS刷写全流程 在汽车电子开发与测试领域,诊断刷写仿真是验证ECU软件更新可靠性的关键环节。当我们需要在实验室环境中完整模拟一个支持UDS协议的ECU时,如何通过CAPL脚本精准响应34/36/37服务链,…...

从五角星到雷达图:用CocosCreator Graphics组件封装一个可复用的自定义图表库

从五角星到雷达图:用CocosCreator Graphics组件封装可复用的自定义图表库 在游戏UI和数据可视化领域,自定义图表的需求无处不在——从角色属性面板的六边形能力图,到商城系统的星级评价展示,再到运营数据的折线趋势分析。传统做法…...

连续变量量子密钥分发技术及其距离自适应策略

1. 连续变量量子密钥分发技术概述 量子密钥分发(QKD)技术作为量子安全通信的核心手段,近年来在理论和实验层面都取得了显著进展。其中,连续变量量子密钥分发(CV-QKD)因其独特的优势正受到越来越多的关注。与…...

别再死磕ImageNet了!用CLIP的Zero-Shot能力,5分钟搞定你的自定义图像分类任务

用CLIP的零样本能力5分钟构建自定义图像分类器 在计算机视觉领域,ImageNet曾经是模型训练的黄金标准,但如今我们有了更高效的替代方案。CLIP(Contrastive Language-Image Pretraining)作为OpenAI推出的多模态模型,彻底…...

Arm Cortex-X2处理器错误分析与规避方案

1. Arm Cortex-X2处理器错误概述Arm Cortex-X2作为Armv9架构下的高性能处理器核心,广泛应用于移动计算和高性能嵌入式领域。在实际部署中,开发者可能会遇到各类硬件异常和功能错误,这些错误通常记录在Arm官方发布的勘误表(Errata Notice)中。…...

告别TypeError:用f-string和format()优雅地拼接字符串与变量(Python 3.6+实战)

Python字符串格式化实战:从TypeError到优雅拼接 在Python开发中,字符串拼接是最基础却又最容易出错的环节之一。当你在构建动态报告、记录日志或生成API响应时,突然遇到"TypeError: can only concatenate str (not int) to str"这样…...

华为OD机试真题 新系统 2026-04-29 C++ 实现【操作历史管理器的撤销/重做能力】

目录 题目 思路 Code 题目 实现一个操作历史管理器,使用双向链表存储执行过的操作。支持执行新操作、撤销和重做功能。 功能说明: 执行操作(execute {操作描述}):执行新操作,并清除当前操作之后的所有历史记录 撤销(undo):回退到上一个操作状态(上一个操作状态可以…...

手把手教你用CH340X给STM32做一键下载电路(附自动复位/BOOT控制原理详解)

CH340X与STM32一键下载电路实战指南 1. 为什么需要一键下载功能? 每次给STM32烧录程序都要手动按复位键、切换BOOT0跳线?这种重复性操作不仅降低开发效率,还容易因操作失误导致芯片锁死。传统下载方式存在三大痛点: 物理操作繁…...

手把手教你用BERT实战电信诈骗文本分类:从数据脱敏到模型融合的完整流程

电信诈骗文本分类实战指南:从数据清洗到模型部署的全链路解析 电信诈骗案件分类一直是公共安全领域的技术难点。去年接触到一个真实案例:某地反诈中心每天需要处理近千条报案记录,人工分类效率低下导致关键预警延迟。这正是我们需要自动化文本…...

import os

import os 的原理os 是 Python 内置标准库,提供了操作系统接口——让 Python 代码可以操作文件、目录、环境变量、进程等,而且同一套代码在 Windows、macOS、Linux 上都能用。核心原理:抽象层 os 模块的本质是一个跨平台抽象层: 你…...

毕业设计:Python+MySQL+Django学生信息管理系统(源码)

目录 一、项目背景 二、技术介绍 三、功能介绍 四、代码设计 五、系统实现 一、项目背景 随着我国高等教育事业的高速发展,高校招生规模持续扩大,在校学生数量呈现爆发式增长。据统计,2024年全国普通高校在校生规模已突破4500万人&…...

告别CAN总线焦虑:手把手教你用低成本LIN总线搞定车窗、雨刷等车身控制

低成本LIN总线实战指南:车窗雨刷控制的工程化解决方案 在汽车电子开发中,控制单元的成本压力始终是工程师面临的核心挑战。当我们需要为车窗升降、雨刷控制这类非核心功能设计通信系统时,动辄数百元的CAN总线方案往往显得过于昂贵。这时&…...

终极指南:如何用imewlconverter解决20+输入法词库迁移难题

终极指南:如何用imewlconverter解决20输入法词库迁移难题 【免费下载链接】imewlconverter ”深蓝词库转换“ 一款开源免费的输入法词库转换程序 项目地址: https://gitcode.com/gh_mirrors/im/imewlconverter 你是否曾经因为更换电脑或输入法而丢失了多年积…...

测试02测试02测试02测试02测4测试02测试02测试02测试02测4

测试02测试02测试02测试02测4测试02测试02测试02测试02测4...

YOLO+Pytorch基于深度学习的水果蔬菜检测系统(源码)

目录 一、项目背景 二、技术介绍 三、功能介绍 四、代码设计 五、系统实现 一、项目背景 我国是全球最大的水果蔬菜生产国和消费国,果蔬产业在国民经济中占据重要地位。然而,果蔬产后处理环节长期依赖人工分拣与品质检测,存在效率低、主…...

终极一站式游戏模组管理解决方案:XXMI启动器完全指南

终极一站式游戏模组管理解决方案:XXMI启动器完全指南 【免费下载链接】XXMI-Launcher Modding platform for GI, HSR, WW and ZZZ 项目地址: https://gitcode.com/gh_mirrors/xx/XXMI-Launcher 你是否厌倦了为每个游戏单独下载、安装和管理模组?是…...

避坑指南:在RT-Thread上玩转DHT11和MQTT时,我遇到的5个常见问题及解决方法

RT-Thread实战:DHT11与MQTT开发中的5个典型问题深度解析 在嵌入式物联网项目中,RT-Thread凭借其丰富的软件包生态和模块化设计,成为许多开发者的首选。但当我们将DHT11温湿度传感器与MQTT协议结合使用时,从环境搭建到功能实现的全…...

智慧树学习助手:3步实现自动化高效学习,节省90%课程时间

智慧树学习助手:3步实现自动化高效学习,节省90%课程时间 【免费下载链接】zhihuishu 智慧树刷课插件,自动播放下一集、1.5倍速度、无声 项目地址: https://gitcode.com/gh_mirrors/zh/zhihuishu 你是否曾为智慧树平台的繁琐操作而烦恼…...

GB28181语音对讲实战:从SIP信令到PCMA音频流的完整抓包分析(附C++代码示例)

GB28181语音对讲实战:从SIP信令到PCMA音频流的完整抓包分析(附C代码示例) 在视频监控系统的开发中,语音对讲功能往往是实现双向实时通信的关键环节。GB28181标准作为国内广泛应用的视频监控联网标准,其语音对讲功能基于…...

别让MPU6050数据坑了你:STM32H5上部署CNN人体活动识别的传感器校准与数据对齐实战

别让MPU6050数据坑了你:STM32H5上部署CNN人体活动识别的传感器校准与数据对齐实战 当你兴奋地将训练好的CNN模型部署到STM32H5开发板,准备实时识别人体活动时,却发现输出结果完全不符合预期——走路被识别为静坐,上下楼梯被误判为…...

别再只盯着准确率了!用sklearn的roc_curve函数,5分钟搞定模型好坏诊断

别再只盯着准确率了!用sklearn的roc_curve函数,5分钟搞定模型好坏诊断 刚入门的机器学习开发者常常陷入一个误区:把准确率(Accuracy)当作评估模型的唯一标准。但当你面对一个信用卡欺诈检测数据集时,99%的…...

3个微信聊天记录备份痛点与开源解决方案:WeChatExporter完全指南

3个微信聊天记录备份痛点与开源解决方案:WeChatExporter完全指南 【免费下载链接】WeChatExporter 一个可以快速导出、查看你的微信聊天记录的工具 项目地址: https://gitcode.com/gh_mirrors/wec/WeChatExporter 你是否曾因手机丢失或更换而痛失珍贵的微信聊…...

终极MP4视频修复指南:用Untrunc拯救你的损坏视频文件

终极MP4视频修复指南:用Untrunc拯救你的损坏视频文件 【免费下载链接】untrunc Restore a damaged (truncated) mp4, m4v, mov, 3gp video. Provided you have a similar not broken video. 项目地址: https://gitcode.com/gh_mirrors/unt/untrunc 你是否曾因…...

Selenium菜鸟教程学习笔记

Selenium菜鸟教程学习笔记 本博客仅为个人学习记录与理解分享,非商业用途,所有代码与文档版权归原项目及其贡献者所有。selenium菜鸟教程 一、Selenium环境搭建 1.安装Selenium库 使用Python编写自动化脚本来控制浏览器 pip install selenium2.测试…...

5分钟上手LosslessCut:零基础掌握无损视频剪辑神器

5分钟上手LosslessCut:零基础掌握无损视频剪辑神器 【免费下载链接】lossless-cut The swiss army knife of lossless video/audio editing 项目地址: https://gitcode.com/gh_mirrors/lo/lossless-cut 还在为剪辑视频等待漫长渲染而烦恼吗?每次剪…...