一次linux中定位c++程序运行异常的经历

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 一次linux中定位c++程序运行异常的经历

今天下午我遇到了一些棘手的问题,因为在mips64上编译程序,经常出现程序编译不出来,或者运行不正常,花了很长的时间定位,最后和同事一些解决了,下面分享出来我提炼出来的一些核心定位问题的步骤。


子线程创建不出来


猜测:go的程序都能创建出子线程,但是c++的创建不出来,但是在 x86 可以,是不是什么 linux 系统限制?

aHR0cHM6Ly9jb2RpbmczbWluLm9zcy1hY2NlbGVyYXRlLmFsaXl1bmNzLmNvbS8yMDIwLzA1LzI1L05qVUZtZTIyNDgucG5n.png


aHR0cHM6Ly9jb2RpbmczbWluLm9zcy1hY2NlbGVyYXRlLmFsaXl1bmNzLmNvbS8yMDIwLzA1LzI1L2xiRDBDYzIyNTAucG5n.png


解决方法:加错误日志进行调试(最 lower 办法)

找到报错点:


aHR0cHM6Ly9jb2RpbmczbWluLm9zcy1hY2NlbGVyYXRlLmFsaXl1bmNzLmNvbS8yMDIwLzA1LzI1LzBIZGNsWjIyNTEucG5n.png


查询 man 手册,看看是不是能找到有帮助的东西

man pthread_attr_setstacksize


打印出解释


ERRORS
       pthread_attr_setstacksize() can fail with the following error:
       EINVAL The stack size is less than PTHREAD_STACK_MIN (16384) bytes.
       On some systems, pthread_attr_setstacksize() can fail with the error EINVAL if stacksize is not a multiple of the system page size.


翻译上面的话,可能会产生这个错误


EINVAL 堆栈大小小于 PTHREAD_STACK_MIN(16384) 字节。


在某些系统上,如果 stacksize 不是系统页面大小的倍数, pthread_attr_setstacksize() 可能会失败,并显示错误 EINVAL


查询 linux 报错码含义, 得知错误码 22 果然和 man 手册中说的一致,是参数有问题。


aHR0cHM6Ly9jb2RpbmczbWluLm9zcy1hY2NlbGVyYXRlLmFsaXl1bmNzLmNvbS8yMDIwLzA1LzI1L3V5TXJvMTIyNTUucG5n.png


第一次尝试:扩大线程栈到上面说的 16384 ,但还是报错


aHR0cHM6Ly9jb2RpbmczbWluLm9zcy1hY2NlbGVyYXRlLmFsaXl1bmNzLmNvbS8yMDIwLzA1LzI1L2pVUWlUNzIzMDEucG5n.png

根据经验,查看最小页大小,发现是 16k , 而 x86 架构是 4K ,原来用的 20K 是不对齐的,怪不得创建不出来线程。

aHR0cHM6Ly9jb2RpbmczbWluLm9zcy1hY2NlbGVyYXRlLmFsaXl1bmNzLmNvbS8yMDIwLzA1LzI1L1ZDbVhVUDIzMDMucG5n.png

依次尝试, 最终发现 10*16K 子线程成功创建。


但是不准确,按其说法,应该是设置 PAGESIZE 的整数倍都行,怀疑其对最小值有要求。 c++ 的头文件在 /usr/include 目录下面, PTHREAD_STACK_MIN 是一个常量,估计里面会有定义,尝试查找

$ grep -rl PTHREAD_STACK_MIN *
bits/local_lim.h
pthread.h

还真让我找到了, 根据英文注释,至少要两个 64K 作为线程栈,才能跑起一个线程。

aHR0cHM6Ly9jb2RpbmczbWluLm9zcy1hY2NlbGVyYXRlLmFsaXl1bmNzLmNvbS8yMDIwLzA1LzI1LzFBWkZPVjIzMTEucG5n.png


至此问题解决。


部分线程卡住


我发现程序虽然正常运行,但是部分功能不正常,经过查看日志发现,有一个线程只执行了一半就卡住了。


经过查看日志可以定位出是哪个线程卡住,如果从日志看不出来也没关系。可以使用 pstack 进程号 看一些进程堆栈。


查看进程 pid


ps -ef |grep 进程名


使用 gdb 查看是否出现问题,两个重要命令


gdb attach {pid} #查看正在运行程序的栈
info thread #进入以后使用,查看线程信息


找到错误位置,出现了 fgets() 和 read() 函数,怀疑是此处出现问题。


image.png



aHR0cHM6Ly9jb2RpbmczbWluLm9zcy1hY2NlbGVyYXRlLmFsaXl1bmNzLmNvbS8yMDIwLzA1LzI1LyVFNCVCQyU4MSVFNCVCOCU5QSVFNSVCRSVBRSVFNCVCRiVBMSVFNiU4OCVBQSVFNSU5QiVCRV9mNmZmMGUyMi0xYzUwLTQ1OGMtODZmYi0zY2VkYjAwYTYyOTUyMzI4LnBuZw.png


怀疑 1: _LINE_LENGTH 1024 长度太短,接受命令返回值后超过了数组本身的长度,覆盖了位置的内存。


这种情况我以前遇到过,表现应该是程序直接就崩溃了。


怀疑 2:执行命令的时候卡了,导致后面的程序没有执行。


根据 gdb 打印出来的参数,执行 linux 命令进行测试,果然是卡在这了!


再次使用 pstree -p {pid} 查看,确实主线程,调用了 linux 命令卡住。


aHR0cHM6Ly9jb2RpbmczbWluLm9zcy1hY2NlbGVyYXRlLmFsaXl1bmNzLmNvbS8yMDIwLzA1LzI1LyVFNCVCQyU4MSVFNCVCOCU5QSVFNSVCRSVBRSVFNCVCRiVBMSVFNiU4OCVBQSVFNSU5QiVCRV9lZThiZTVhNC02MzEzLTQ5YTktOTY4Yy04NTYzZDljM2U5Y2QyMzM0LnBuZw.png


接下来解决卡命令的问题

解决 1:加 timeout 处理空返回。下面是示例命令,并不是我使用的命令。


timeout 5 ls -al


代表超过 5 秒返回。

解决 2:定位为什么这个 linux 命令会卡住。


strace ls -al



直到解决为止。这就是今晚我加班到 10 点 30 解决的问题,我又用了一个小时总结下整个过程备忘,希望对你也有帮助。

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
3月前
|
网络协议 安全 Linux
Linux C/C++之IO多路复用(select)
这篇文章主要介绍了TCP的三次握手和四次挥手过程,TCP与UDP的区别,以及如何使用select函数实现IO多路复用,包括服务器监听多个客户端连接和简单聊天室场景的应用示例。
105 0
|
1月前
|
机器学习/深度学习 人工智能 API
如何在c++侧编译运行一个aclnn(AOL)算子?
CANN的AOL库提供了一系列高性能算子API,优化了昇腾AI处理器的调用流程。通过两段式接口设计,开发者可以高效地调用算子库API,实现模型创新与应用,提升开发效率和模型性能。示例中展示了如何使用`aclnnAdd`算子,包括环境初始化、算子调用及结果处理等步骤。
|
3月前
|
存储 Linux C语言
Linux C/C++之IO多路复用(aio)
这篇文章介绍了Linux中IO多路复用技术epoll和异步IO技术aio的区别、执行过程、编程模型以及具体的编程实现方式。
135 1
Linux C/C++之IO多路复用(aio)
|
3月前
|
资源调度 Linux 调度
Linux c/c++之进程基础
这篇文章主要介绍了Linux下C/C++进程的基本概念、组成、模式、运行和状态,以及如何使用系统调用创建和管理进程。
54 0
|
3月前
|
Ubuntu Linux 编译器
Linux/Ubuntu下使用VS Code配置C/C++项目环境调用OpenCV
通过以上步骤,您已经成功在Ubuntu系统下的VS Code中配置了C/C++项目环境,并能够调用OpenCV库进行开发。请确保每一步都按照您的系统实际情况进行适当调整。
736 3
|
3月前
|
运维 Java Linux
【运维基础知识】Linux服务器下手写启停Java程序脚本start.sh stop.sh及详细说明
### 启动Java程序脚本 `start.sh` 此脚本用于启动一个Java程序,设置JVM字符集为GBK,最大堆内存为3000M,并将程序的日志输出到`output.log`文件中,同时在后台运行。 ### 停止Java程序脚本 `stop.sh` 此脚本用于停止指定名称的服务(如`QuoteServer`),通过查找并终止该服务的Java进程,输出操作结果以确认是否成功。
93 1
|
3月前
|
资源调度 Linux 调度
Linux C/C++之线程基础
这篇文章详细介绍了Linux下C/C++线程的基本概念、创建和管理线程的方法,以及线程同步的各种机制,并通过实例代码展示了线程同步技术的应用。
40 0
Linux C/C++之线程基础
|
3月前
|
Linux C++
Linux C/C++之IO多路复用(poll,epoll)
这篇文章详细介绍了Linux下C/C++编程中IO多路复用的两种机制:poll和epoll,包括它们的比较、编程模型、函数原型以及如何使用这些机制实现服务器端和客户端之间的多个连接。
49 0
Linux C/C++之IO多路复用(poll,epoll)
|
3月前
|
网络协议 Linux 网络性能优化
Linux C/C++之TCP / UDP通信
这篇文章详细介绍了Linux下C/C++语言实现TCP和UDP通信的方法,包括网络基础、通信模型、编程示例以及TCP和UDP的优缺点比较。
66 0
Linux C/C++之TCP / UDP通信
|
3月前
|
消息中间件 Linux API
Linux c/c++之IPC进程间通信
这篇文章详细介绍了Linux下C/C++进程间通信(IPC)的三种主要技术:共享内存、消息队列和信号量,包括它们的编程模型、API函数原型、优势与缺点,并通过示例代码展示了它们的创建、使用和管理方法。
46 0
Linux c/c++之IPC进程间通信