python通过ctypes调用C/C++ SDK,当SDK异常时,同时打印C/C++/Python的栈信息

python通过ctypes调用C/C++ SDK,当SDK异常时,同时打印C/C++/Python的栈信息

  • 一.复现步骤
  • 二.输出

本文演示了python通过ctypes调用C/C++ SDK,当SDK异常时,同时打印C/C++/Python的栈信息.基于traceback、addr2line、PyErr_SetString、backtrace_symbols

一.复现步骤

cat > print_backtrace.h <<-'EOF'
#ifndef __PRINT_BACKTRACE_H__
#define __PRINT_BACKTRACE_H__
#ifdef __cplusplus
extern "C" {
#endif
void print_backtrace();
#ifdef __cplusplus
}
#endif
#endif // __PRINT_BACKTRACE_H__
EOFcat > print_backtrace.cpp <<-'EOF'
#include "print_backtrace.h"
#include <stdio.h>
#include <execinfo.h>
#include <stdlib.h>
#include <string.h>
#include <stdbool.h>
#include <assert.h>
#include <stdexcept>
#include <Python.h>#define MAX_FRAMES   128
#define BUFFER_SIZE  512void print_backtrace()
{// 1.打印调用栈void *buff[MAX_FRAMES] = { 0 };unsigned int depth = backtrace(buff, MAX_FRAMES);char **strings = backtrace_symbols(buff, depth);printf("-----------------------print_backtrace(c/c++)--------------------\n");for (int i = 0; i < MAX_FRAMES; i++){if (i >= depth) {break;}// 1.1 将地址转换为代码路径和行号{char cmdline[BUFFER_SIZE] = {0};snprintf(cmdline, BUFFER_SIZE, "bash ./parser.sh \"%s\"", strings[i]);FILE *pFd = popen(cmdline, "r");if ( pFd == nullptr ){printf("popen failed\n");return ;}char buf[BUFFER_SIZE] = {0};fread(buf, sizeof(buf)/sizeof (buf[0]), 1, pFd);printf("%s", buf);pclose(pFd); }        }free(strings);// 2.触发python异常if (!Py_IsInitialized()) {Py_Initialize();}        PyGILState_STATE gstate;gstate = PyGILState_Ensure();PyErr_SetString(PyExc_RuntimeError, "print_backtrace trigger_error");PyGILState_Release(gstate);
}
EOFcat > api.c <<-'EOF'
#include <stdio.h>
#include <stdlib.h>
#include "print_backtrace.h"int function_a() {printf("%s %d\n",__FILE__,__LINE__);print_backtrace();return 0;
}int function_b() {printf("%s %d\n",__FILE__,__LINE__);function_a();return 0;
}int function_c() {printf("%s %d\n",__FILE__,__LINE__);function_b();return 0;
}int function_d() {printf("%s %d\n",__FILE__,__LINE__);function_c();return 0;
}int function_e() {printf("%s %d\n",__FILE__,__LINE__);function_d();return 0;
}
EOFcat > parser.sh <<-'EOF'
#!/bin/bashline=`echo $1 | awk '{print $1}' | sed 's/[()]/ /g'`
so_path=`echo $line | awk '{print $1}'`if [ ! -f $so_path ];thenexit 0
fiaddr_path=`echo $line | awk '{print $2}'`
function_name=`echo $addr_path | awk -F+ '{print $1}'`
offset=`echo $addr_path | awk -F+ '{print $2}'`if [ $function_name ];thenfunction_base=`nm $so_path | grep -w "$function_name" | awk '{print $1}'`abs_addr=`echo "$function_base $offset" | gawk --non-decimal-data '{A="0x"$1;B=$2;printf "0x%x\n",A+B}'`debug_info=`addr2line -f $abs_addr -e $so_path`
elsedebug_info=`addr2line -f $offset -e $so_path`
fiecho $debug_info
EOFcat > main.py <<-'EOF'
import ctypes
import tracebackdef function_a():print("function_a")lib = ctypes.CDLL('./libapi.so')function_e = lib.function_efunction_e()def function_b():print("function_b")function_a()def function_c():print("function_c")function_b()def function_d():print("function_d")function_c()try:function_d()
except:print("-----------------------print_backtrace(python)--------------------")traceback.print_exc()
EOFg++ -fPIC -g -shared -o libprint_backtrace.so print_backtrace.cpp \-I /home/anaconda3/envs/demo/include/python3.10 \/home/anaconda3/envs/demo/lib/libpython3.10.so \-Wl,-rpath=/home/anaconda3/envs/demo/lib/gcc -fPIC -g -shared -o libapi.so api.c  -L . -lprint_backtrace -Wl,-rpath=.
python3 main.py

二.输出

function_d
function_c
function_b
function_a
api.c 30
api.c 24
api.c 18
api.c 12
api.c 6
-----------------------print_backtrace(c/c++)--------------------
print_backtrace /home/print_backtrace.cpp:18
function_a /home/api.c:8
function_b /home/api.c:14
function_c /home/api.c:20
function_d /home/api.c:26
function_e /home/api.c:32
ffi_call_unix64 :?
ffi_call_int ffi64.c:?
ffi_call ??:?
addr2line: DWARF error: can't find .debug_ranges section.
_ctypes_callproc.cold :?
addr2line: DWARF error: can't find .debug_ranges section.
PyCFuncPtr_call.cold :?
nm: /lib/x86_64-linux-gnu/libc.so.6: no symbols
?? ??:0
-----------------------print_backtrace(python)--------------------
RuntimeError: print_backtrace trigger_errorThe above exception was the direct cause of the following exception:Traceback (most recent call last):File "/home/main.py", line 39, in <module>function_d()File "/home/main.py", line 36, in function_dfunction_c()File "/home/main.py", line 32, in function_cfunction_b()File "/home/main.py", line 28, in function_bfunction_a()File "/home/main.py", line 24, in function_afunction_e()
SystemError: <_FuncPtr object at 0x7f10c6d4cdc0> returned a result with an exception set

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/12418.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

自媒体的发展趋势:从个人表达到全球话语权

一、引言随着数字技术的快速发展&#xff0c;信息传播的方式和格局也在不断变化。自媒体&#xff0c;作为其中的一股重要力量&#xff0c;正在以它的独特方式改变着全球的信息传播和社会发展。本文将从自媒体的定义及发展历程入手&#xff0c;深入探讨自媒体未来的发展趋势&…

感知局部规划--似然场局部规划

系列文章目录 提示&#xff1a;这里可以添加系列文章的所有文章的目录&#xff0c;目录需要自己手动添加 TODO:写完再整理 文章目录 系列文章目录前言感知导航感知似然场局部规划&#xff08;很像DWA但是不依赖地图&#xff0c;完全依赖感知&#xff09; 前言 认知有限&#x…

Uniapp开发入门:构建跨平台应用的全面指南

引言 什么是Uniapp Uniapp是一款由DCloud公司推出的基于Vue.js的跨平台应用开发框架。它的核心理念是“一套代码&#xff0c;多端运行”&#xff0c;开发者只需编写一份代码&#xff0c;即可生成包括iOS、Android、H5、微信小程序、支付宝小程序、百度小程序等多平台的应用。…

初识C++ · string的使用(2)

目录 1 Modifiers部分 1.1 assign的使用 1.2 insert的使用 1.3 erase的使用 1.4 replace的使用 2 capacity部分 2.1 max_size的使用 2.2 capacity的使用 2.3 reserve的使用 2.4 shrink_to_fit简介 2.5 resize的使用 2.6 clear的使用 3 String operations部分 3.1 …

[数据结构1.0]快速排序

最近学习了快速排序&#xff0c;鼠鼠俺来做笔记了&#xff01; 本篇博客用排升序为例介绍快速排序&#xff01; 1.快速排序 快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法&#xff0c;其基本思想为&#xff1a;任取待排序元素序列中的某元素作为基准值&#x…

202103青少年软件编程(Python)等级考试试卷(一级)

一、单选题&#xff08;共25题&#xff0c;每题2分&#xff0c;共50分&#xff09; 下列哪个操作不能退出IDLE环境&#xff1f;&#xff08; &#xff09; A、AltF4 B、CtrlQ C、按ESC键 D、exit() 试题编号&#xff1a;20210124-yfj-003 题型&#xff1a;单选题 答案&#xf…

Java面试八股之一个char类型变量能不能存储一个中文字符

Java中一个char类型变量能不能存储一个中文字符&#xff1f;为什么&#xff1f; Java中一个char类型变量可以存储一个中文字符。原因如下&#xff1a; Unicode编码支持&#xff1a;Java语言采用Unicode字符集作为其内建字符编码方式。Unicode是一种广泛接受的字符编码标准&am…

两小时看完花书(深度学习入门篇)

1.深度学习花书前言 机器学习早期的时候十分依赖于已有的知识库和人为的逻辑规则&#xff0c;需要人们花大量的时间去制定合理的逻辑判定&#xff0c;可以说是有多少人工&#xff0c;就有多少智能。后来逐渐发展出一些简单的机器学习方法例如logistic regression、naive bayes等…

mybatisplus查询练习代码

mybatisplus查询练习代码 package com.yase;import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.yase.entity.Student; import com.yase.entity.Teacher; import com.yase…

什么是CCRC?做什么用的?

CCRC&#xff08;中国网络安全审查认证和市场监管大数据中心&#xff09;原名为中国网络安全审查技术与认证中心&#xff0c;也被称为中国信息安全认证中心&#xff08;ISCCC&#xff09;。 该中心是经中央机构编制委员会办公室批准成立的&#xff0c;其主要职责是依据国家法律…

kafka集群传统部署(raft模式)—— 筑梦之路

kafka二进制包&#xff1a;https://dlcdn.apache.org/kafka/3.7.0/kafka_2.13-3.7.0.tgz 集群规划 主机名IP地址节点ID角色分配kafka1192.168.100.1001broker,controllerkafka2192.168.100.1012broker,controllerkafka3192.168.100.1023broker,controller 编辑配置文件 con…

代码随想录算法训练营第36天|● 738.单调递增的数字 ● 968.监控二叉树

738. 单调递增的数字 发现第一位变小了其他的迅速变9 class Solution:def monotoneIncreasingDigits(self, n: int) -> int:strnlist(str(n))for i in range(len(strn)-1,0,-1):if strn[i-1]>strn[i]:strn[i-1]str(int(strn[i-1])-1)for j in range(i,len(strn)):strn[…

超级简单的地图操作工具开发可疑应急,地图画点,画线,画区域,获取地图经纬度等

使用echars的地图画点,画线,画区域,获取地图经纬度等 解压密码:10086007 地图也是用临时的bmap.js和china.js纯离线二选一 一共就这么多文件 画点,画线,画区域 点击地图获取经纬度-打印到控制台,这样就能渲染航迹,多变形,结合其他算法算圆等等操作 下载资源:https://download…

JSON-server 服务的搭建

1、全局安装&#xff1a; pnpm i -g json-server2、创建db.json文件 {"posts": [{"id": 1,"title": "json-server","author": "typicode"}],"comments":[{"id": 1,"body": "…

什么情况下会造成索引失效?

2.3.4. 索引失效 对索引使用左或者左右模糊匹配 使用左或者左右模糊匹配的时候&#xff0c;也就是 like %xx 或者 like %xx% 这两种方式都会造成索引失效。但是如果前缀是确定的那么就可以使用到索引&#xff0c;例如 name like 许%。 因为索引 B 树是按照「索引值」有序排列…

SpringBoot 中 zip 文件解压工具类

SpringBoot 中 zip 文件解压工具类 zip 文件解压&#xff08;不支持密码&#xff09; 相关 Maven 依赖 <dependency> <groupId>org.apache.commons</groupId> <artifactId>commons-lang3</artifactId> <version>3.6</version>…

练习题(2024/5/14)

1四数相加 II 给你四个整数数组 nums1、nums2、nums3 和 nums4 &#xff0c;数组长度都是 n &#xff0c;请你计算有多少个元组 (i, j, k, l) 能满足&#xff1a; 0 < i, j, k, l < nnums1[i] nums2[j] nums3[k] nums4[l] 0 示例 1&#xff1a; 输入&#xff1a;n…

代码随想录训练营Day28:贪心算法06

1.738单调递增的数字 贪心策略&#xff1a;如果strNum[i]<strNum[i-1]那么strNum[i] 9,strNum[i-1]--;//比如87对应的最大的单调递增的就是79. 具体实现&#xff1a; 对于遇到小于的情况&#xff1a;如果strNum[i]<strNum[i-1]那么strNum[i] 9,strNum[i-1]--;遍历顺…

linux phpstudy 重启命令

[rootLinuxWeb phpstudy]# ./system/phpstudyctl restart 查看命令 1) phpstudy -start 启动小皮面板 2) phpstudy -stop 停止小皮面板 3) phpstudy -restart 重启小皮面板 4) phpstudy -status 查询面板状态 5) phpstudy -in…

OFDM802.11a的FPGA实现(十五)短训练序列:STS(含Matlab和verilog代码)

原文链接&#xff08;相关文章合集&#xff09;&#xff1a;OFDM 802.11a的xilinx FPGA实现 1.前言 在之前已经完成了data域数据的处理&#xff0c;在构建整个802.11a OFDM数据帧的时候&#xff0c;还剩下前导码和signal域的数据帧&#xff0c;这两部分的内容。 PLCP的前导部分…