gpt4 book ai didi

python - 为崩溃的子进程 : no out and err after a call to communicate() 捕获 "Segmentation fault"消息

转载 作者:IT老高 更新时间:2023-10-28 21:14:03 25 4
gpt4 key购买 nike

我在使用 subprocess 模块获取崩溃程序的输出时遇到问题。我正在使用 python2.7 和 subprocess 调用带有奇怪参数的程序以获得一些段错误为了调用程序,我使用以下代码:

proc = (subprocess.Popen(called,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE))
out,err=proc.communicate()
print out,err

被调用的是一个包含程序名称和参数的列表(一个包含随机字节的字符串,除了子进程根本不喜欢的 NULL 字节)

当程序没有崩溃时,代码会运行并向我显示 stdout 和 stderr,但是当它确实崩溃时,out 和 err 是空的,而不是显示著名的“段错误”。

我希望找到一种方法,即使程序崩溃也会出错。

我也试过 check_output/call/check_call 方法

一些附加信息:

  • 我在 python 虚拟环境中的 Archlinux 64 位上运行这个脚本(这里不应该是重要的东西,但你永远不知道:p)

  • 段错误发生在我尝试运行的 C 程序中,是缓冲区溢出的结果

  • 问题是当segfault发生时,我无法得到子进程发生了什么的输出

  • 我得到正确的返回码:-11 (SIGSEGV)

  • 使用python我得到:

      ./dumb2 AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
    ('Exit code was:', -11)
    ('Output was:', '')
    ('Errors were:', '')
  • 在 python 之外我得到:

     ./dumb2 $(perl -e "print 'A'x50")  
    BEGINNING OF PROGRAM
    AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
    END OF THE PROGRAM
    Segmentation fault (core dumped)
  • shell的返回值一样:echo $?返回 139 所以 -11 ($? & 128)

最佳答案

“Segmentation fault” 消息可能由 shell 生成。要查明进程是否被 SIGSEGV 杀死,请检查 proc.returncode == -signal.SIGSEGV

如果您想查看消息,可以在 shell 中运行命令:

#!/usr/bin/env python
from subprocess import Popen, PIPE

proc = Popen(shell_command, shell=True, stdout=PIPE, stderr=PIPE)
out, err = proc.communicate()
print out, err, proc.returncode

我已经用 shell_command="python -c 'from ctypes import *; memset(0,1,1)'" 对其进行了测试,这会导致段错误并且消息被捕获在 错误

如果消息直接打印到终端,那么您可以使用 pexpect 模块来捕获它:

#!/usr/bin/env python
from pipes import quote
from pexpect import run # $ pip install pexpect

out, returncode = run("sh -c " + quote(shell_command), withexitstatus=1)
signal = returncode - 128 # 128+n
print out, signal

或者直接使用pty stdlib 模块:

#!/usr/bin/env python
import os
import pty
from select import select
from subprocess import Popen, STDOUT

# use pseudo-tty to capture output printed directly to the terminal
master_fd, slave_fd = pty.openpty()
p = Popen(shell_command, shell=True, stdin=slave_fd, stdout=slave_fd,
stderr=STDOUT, close_fds=True)
buf = []
while True:
if select([master_fd], [], [], 0.04)[0]: # has something to read
data = os.read(master_fd, 1 << 20)
if data:
buf.append(data)
else: # EOF
break
elif p.poll() is not None: # process is done
assert not select([master_fd], [], [], 0)[0] # nothing to read
break
os.close(slave_fd)
os.close(master_fd)
print "".join(buf), p.returncode-128

关于python - 为崩溃的子进程 : no out and err after a call to communicate() 捕获 "Segmentation fault"消息,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22250893/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com