草庐IT

python - 如何在没有 GDB Python API 的情况下在 C 或 Python 代码中控制 gdb?

coder 2023-08-14 原文

我正在尝试用 python 或 c 编写一个可以使用 gdb 调试 c 代码的程序。

我读过 solution of TomInvoke and control GDB from Python .但它们或多或少是在 python 中编写 gdb 脚本的解决方案。因为我要使用 arm-gdb 来调试嵌入式程序,所以我无法在我的 gdb 中启用 python 脚本。

我的目标是创建 gdb 的高级抽象。例如,启动 gdb,设置一些断点并在我的代码中继续。我还阅读了一些 Material gdb/mi 接口(interface)。但是谁能告诉我如何使用 gdb/mi 接口(interface)创建 gdb 进程并从 c/python 代码与 gdb 通信? (幸好我的arm-gdb支持gdb/mi接口(interface))。

最佳答案

正如上面评论中所 promise 的,我已经将我的(早期的、不完整的、几乎肯定有问题的)ruby 作品发布到 http://github.com/mcarpenter/rubug .

这是一个示例(您可以在 examples/breakpoint ).函数 check_for_crash 是一个可以调用的回调函数 在名为 factorial 的程序设置为运行后。断点采用函数名称 (fac;前面的冒号只是表示这是一个 ruby 符号,在这里的所有意图和目的都是轻量级的 字符串)。

EXE = 'factorial'

def check_for_crash(gdb, event)
  case event.type
  when :command_response
    raise RuntimeError, 'oops' unless
  [ :done, :running ].include? event.response.result
  when :breakpoint
    puts 'Breakpoint reached'
    pp event
    gdb.continue
  when :exit
    puts 'Exit'
    gdb.stop_event_loop
    exit
  end
end

gdb = Rubug::Gdb.new
resp = gdb.file EXE
gdb.register_callback(method :check_for_crash)
gdb.break(:fac)
gdb.run '5 > /dev/null'
gdb.start_event_loop

很公平地警告您代码可能...很粗糙。目前(这是我停止的地方)没什么用(在我工作中途更新 gdb 之后,请参阅 语法如下)。

同一目录下有一堆例子 但是,这个名称可能会有所帮助。要(尝试!)运行它们,您需要执行以下操作:

rake clean
rake grammar
rake make 
cd examples/simple_fuzzer
ruby -I ../../lib -r rubygems simple_fuzzer.rb

考虑到撰写本文的时间,您可能应该使用 ruby​​1.8 如果你有选择的话(我当时不喜欢 1.9,可能有 1.9 下的字符串编码问题)。

响应的解析由树顶执行 http://treetop.rubyforge.org , 一个 PEG 解析器。看语法 新鲜的眼睛我相信它可以简化。你需要安装这个 (以及任何其他所需的 gem)使用 gem install ...

如果您进行 Pythonize,还有一些提示:

文档

《Debugging with GDB》之外几乎没有什么 (第 22 章)。我已经扔了这个 PDF 和 ch。 22作为一个单独的文件放入 docs存储库的一部分。

异步

协议(protocol)是异步的(起初我以为这是 命令/响应类型协议(protocol),这是一个错误)。如果我要 重新实现这个我可能会使用事件机器之类的东西或者 libevent 而不是滚动我自己的 select() 循环。

语法

语法有点……令人困惑。虽然文档 (27.2.2) 指出响应“由零个或多个带外 记录后跟一个结果记录(可选)”:

`output -> ( out-of-band-record )* [ result-record ] "(gdb)" nl`

你应该知道,因为任何东西都可以随时到达,所以 read() 在套接字上显然可以返回 async/result/more 异步/终止符(!)。例如,我在当前的 gdb 中看到了这一点:

=thread-group-started,id="i1",pid="1086"
=thread-created,id="1",group-id="i1"
^running
*running,thread-id="all"
(gdb)

^ 开头的行是结果记录,其他都是异步的(然后 终结者)。这似乎是 规范。

速度

我的主要关注点是安全性,我对 MI 很感兴趣 自动模糊测试、二进制检查等。为此,GDB/MI 也是 慢(在调试器中启动程序的成本)。 YMMV.

MI/CLI 映射

我在标准 gdb CLI 命令集中有一些东西 看不到如何使用 MI 命令来实现。我有骨架 类似这样的代码:

gdb = Gdb::MI.new
gdb.cli(:file, '/bin/ls')
gdb.cli(:set, :args, '> /dev/null')
gdb.cli(:run)
gdb.cli(:quit)

(我认为,对于我们这些非 MI 专家但了解 gdb 的用户来说,这很好很清楚)。 我现在不记得那些有问题的东西是什么(一年多了 自从我看了这个)但是如果那些神经元真的开火了我会回来的 更新这个。

备选方案

刚开始的时候 在这条路上我发现了 Jamis Buck 的一篇博文: http://weblog.jamisbuck.org/2006/9/25/gdb-wrapper-for-ruby这包裹了一个 popen() 中的 gdb 命令行 session 让我有点畏缩。在 特别的人可能会认为它很脆弱,因为 gdb 没有 保证 CLI 输出的稳定性。你可以(也可以不) 更喜欢这种方法。

如果您使用的是 Windows,那么 PyDbg/PeiMei 可能会对您感兴趣:http://code.google.com/p/paimei/

您可能也喜欢这本书 Grey Hat Python: Python Programming for Hackers (塞茨)。同样,主要基于 Windows,但可能会鼓舞人心。

关于python - 如何在没有 GDB Python API 的情况下在 C 或 Python 代码中控制 gdb?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6822584/

有关python - 如何在没有 GDB Python API 的情况下在 C 或 Python 代码中控制 gdb?的更多相关文章

  1. ruby - 如何在 Ruby 中顺序创建 PI - 2

    出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits

  2. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  3. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  4. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  5. ruby - 难道Lua没有和Ruby的method_missing相媲美的东西吗? - 2

    我好像记得Lua有类似Ruby的method_missing的东西。还是我记错了? 最佳答案 表的metatable的__index和__newindex可以用于与Ruby的method_missing相同的效果。 关于ruby-难道Lua没有和Ruby的method_missing相媲美的东西吗?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/7732154/

  6. ruby-on-rails - Rails 源代码 : initialize hash in a weird way? - 2

    在rails源中:https://github.com/rails/rails/blob/master/activesupport/lib/active_support/lazy_load_hooks.rb可以看到以下内容@load_hooks=Hash.new{|h,k|h[k]=[]}在IRB中,它只是初始化一个空哈希。和做有什么区别@load_hooks=Hash.new 最佳答案 查看rubydocumentationforHashnew→new_hashclicktotogglesourcenew(obj)→new_has

  7. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  8. ruby-on-rails - rails 目前在重启后没有安装 - 2

    我有一个奇怪的问题:我在rvm上安装了ruby​​onrails。一切正常,我可以创建项目。但是在我输入“railsnew”时重新启动后,我有“程序'rails'当前未安装。”。SystemUbuntu12.04ruby-v"1.9.3p194"gemlistactionmailer(3.2.5)actionpack(3.2.5)activemodel(3.2.5)activerecord(3.2.5)activeresource(3.2.5)activesupport(3.2.5)arel(3.0.2)builder(3.0.0)bundler(1.1.4)coffee-rails(

  9. ruby - 默认情况下使选项为 false - 2

    这是在Ruby中设置默认值的常用方法:classQuietByDefaultdefinitialize(opts={})@verbose=opts[:verbose]endend这是一个容易落入的陷阱:classVerboseNoMatterWhatdefinitialize(opts={})@verbose=opts[:verbose]||trueendend正确的做法是:classVerboseByDefaultdefinitialize(opts={})@verbose=opts.include?(:verbose)?opts[:verbose]:trueendend编写Verb

  10. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

随机推荐