在 vs 中 调试 lua脚本

调试lua现有的几种方案:

 1)luaedit  : 编辑断点和断点命中查看那些都让习惯用vs的人感到亲切,但是它只是在纯lua环境下的编辑器,也就是说和期望程序运行中能借用它来调试脚本可能性不大(至少我没有试验出来),能用上的需要给lua脚本做个main脚本函数,并提供一系列伪的c api(由lua实现的函数,只提供制定值的返回)。由luaedit启动脚本main函数进行调试

 2)利用 lua debug 库中的 hook ,然后记录一张断点位置表,设置行模式的 hook ,每次进入 hook 都检查是否是断点处,若是就停下来等待交互调试。这个方法有效,但是很消耗 cpu 。因为每进入一个新的代码行,都需要回调一个函数。当这个函数本身又是用 lua 写的时候,效率更低。

 3)利用lua脚本函数实现,在lua脚本中封装一个断点函数,这种方式结合debug库实现断点和栈信息,然后利用lua环境下的表对函数调用信息进行记录和打印,需要脚本使用者显式提供断点函数的调用。这个办法的优点自己可以组合出适合自己项目的断点调试方案。但缺点是相对繁琐,而且断点命中以后难以和现有方式融合

 

下面提供一个相对简洁且和vs调试程序结合紧密的lua脚本调试方案做参考:

 

 

首先提供一个c api 函数,该函数提供基础的断点功能 ,可以用 asm int 3 或是  winapi DebugBreak() 来实现,将该api函数 注册到lua中,调用该函数看看,可以产生断点,且断点产生时自动切回vs调试器。

 

这种函数在单独执行exe的时候会出现运行错误,需要在断点函数前加个判断IsDebuggerPresent() 作用是判断该exe是否通过调试进程启动。

 

有了断点命中能力以后接下来就是需要做显示断点命中时的脚本函数信息和变量信息。对于这些信息同样还是利用lua提供的debug库来实现。主要利用lua_getstack 获取栈中信息,lua_getinfo获取当前行,函数名等信息,lua_getlocal获取函数中参数相关信息。

 

 static void call_stack_info(lua_State* L,int n)

{

    lua_Debug ar;

    if(lua_getstack(L,n,&ar) == 1)

    {

        lua_getinfo(L,"nSlu",&ar);

        const char* indent;

        if(n == 0)

        {

            indent = "->";

            lua_tinker::output_info("[LUA_DEBUG] |/t[[call stack]]/n");

        }

        else

        {

            indent = "  ";

        }

        if(ar.name)

            lua_tinker::output_info("[LUA_DEBUG] |%s [%s : line %d] : %s()/n",indent,ar.short_src,ar.currentline,ar.name);

        else

            lua_tinker::output_info("[LUA_DEBUG] |%s [%s : line %d] : unkown/n",ar.currentline);

        int index = 0;

        do 

        {

            indent = lua_getlocal (L,&ar,++index);

            if (indent)

            {

                switch(lua_type(L,-1))

                {

                case LUA_TNIL:

                    lua_tinker::output_info( "[LUA_DEBUG] |/t/t/t/t/t|-%s:%s",lua_typename(L,lua_type(L,-1)),indent);

                    break;

                case LUA_TBOOLEAN:

                    lua_tinker::output_info( "[LUA_DEBUG] |/t/t/t/t/t|-%s:%s    %s/n",lua_toboolean(L,-1)?"true":"false");

                    break;

                case LUA_TLIGHTUSERDATA:

                    lua_tinker::output_info( "[LUA_DEBUG] |/t/t/t/t/t|-%s:%s    0x%08p/n",lua_topointer(L,-1));

                    break;

                case LUA_TNUMBER:

                    lua_tinker::output_info( "[LUA_DEBUG] |/t/t/t/t/t|-%s:%s    %f/n",lua_tonumber(L,-1));

                    break;

                case LUA_TSTRING:

                    lua_tinker::output_info( "[LUA_DEBUG] |/t/t/t/t/t|-%s:%s    %s/n",lua_tostring(L,-1));                    

                    break;

                case LUA_TTABLE:

                    lua_tinker::output_info( "[LUA_DEBUG] |/t/t/t/t/t|-%s:%s    0x%08p/n",-1));

                    break;

                case LUA_TFUNCTION:

                    lua_tinker::output_info( "[LUA_DEBUG] |/t/t/t/t/t|-%s:%s()    0x%08p/n",-1));

                    break;

                case LUA_TUSERDATA:

                    lua_tinker::output_info( "[LUA_DEBUG] |/t/t/t/t/t|-%s:%s    0x%08p/n",-1));

                    break;

                case LUA_TTHREAD:

                    lua_tinker::output_info( "[LUA_DEBUG] |/t/t/t/t/t|-%s:%s/n",indent);

                    break;

                }

                lua_settop(L,-2);

            }            

        } while (indent);    

        call_stack_info(L,n+1);

    }

}


 此函数在断点命中后,递归遍历当前的lua_state将栈中函数和其参数都遍历输出

 

有了函数信息和参数信息,接下来就是输出了,如果用printf如果程序不是控制台就有点尴尬,还是借助vs的OutputDebugString来做。

搞定后当断点命中时,点击break,vs中就能出现如下信息了

 

 

接下来封装整个断点函数并把它注册到lua环境中

 

 

int lua_tinker::break_point(lua_State *L)

{

output_info("/n[LUA_DEBUG] |/t>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>   /n");


call_stack_info(L,0);

 

output_info("[LUA_ERROR] |/t<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<   /n/n");


#ifdef _WIN32_WINNT 

if (IsDebuggerPresent())

{

DebugBreak();

}

#endif

return 0;

}

 

 

 

ok 一切就绪只欠调用了 ,将函数注册到脚本中,暂叫 dbg()吧

 

lua_pushstring(L,"dbg");

lua_pushcfunction(L,lua_tinker::break_point);

lua_settable(L,LUA_GLOBALSINDEX);

 

 

调试的时候在脚本中需要断点的地方插入dbg(),然后保存重新加载一下脚本,下一次执行到刚修改的位置就会听到清脆的断点命中声了

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


1.github代码实践源代码是lua脚本语言,下载th之后运行thmain.lua-netTypevgg-savevgg_cifar10/-S0.0001,报错: 试看看安装lua:报错了,参考这篇文章:ubuntu18.04安装lua的步骤以及出现的问题_weixin_41355132的博客-CSDN博客问题解决,安装成功:情况并没有好转,出现相
此文为搬运帖,原帖地址https://www.cnblogs.com/zwywilliam/p/5999924.html前言在看了uwa之前发布的《Unity项目常见Lua解决方案性能比较》,决定动手写一篇关于lua+unity方案的性能优化文。整合lua是目前最强大的unity热更新方案,毕竟这是唯一可以支持ios热更新的办法。然而作
Rime输入法通过定义lua文件,可以实现获取当前时间日期的功能。1.TIMERime是一款可以高度自定义的输入法,相关教程可以查看往期文章,关于时间获取是指输入一个指定关键字,输出当前时间,效果如下(我定义了time关键字):实现如下:①在用户文件夹中新建一个rime.lua文件加入如下代码 ti
localfunctiongenerate_action(params)localscale_action=cc.ScaleTo:create(params.time,params.scale_x,params.scale_y)localfade_action=cc.FadeIn:create(params.time)returncc.Spawn:create(scale_action,fade_action)end
2022年1月11日13:57:45 官方:https://opm.openresty.org/官方文档:https://opm.openresty.org/docs#table-of-contents为什么建议使用opm不建议使用luarocks?http://openresty.org/cn/using-luarocks.html官方解释:请注意!LuaRocks并不是OpenResty官方推荐的装包方式。LuaRoc
在Lua中的table(表),就像c#中的HashMap(哈希表),key和value一一对应。元表:table的一个操作的拓展,里面包含关联了对应的方法,元方法就是其中一个。元方法:当你通过键来访问table的时候,如果这个键没有值,那么Lua就会寻找该table的metatable(假定有metatable)中的__index键。如果__inde
表排序:table.sort(list[,comp])参数list:指定表,可选参数comp:排序函数,无参数时通常按升序排序。排序函数针对表中连续的序列,其间不可以存在空洞或nil,排序函数需要两个形参(对应表中每次参加比较的两个数据),需要一个比较两个形参表达式的返回值,不能含有等于关系,例如>=,<=,==。do
一、安装lua环境1.1安装依赖包[root@centos7~]#yuminstallgccreadline-devel1.2下线lua源码包并解压[root@centos7~]#wgethttp://www.lua.org/ftp/lua-5.3.5.tar.gz[root@centos7~]#tarxvflua-5.3.5.tar.gz-C/usr/local/src1.3进行编译[root@centos7~]
官网OpenResty® 是一个基于 Nginx 与Lua的高性能Web平台,其内部集成了大量精良的Lua库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态Web应用、Web服务和动态网关。OpenResty® 通过汇聚各种设计精良的 Nginx 模块(主要由
表参考《lua程序设计》可以认为,表是一种动态分配的对象,程序只能操作指向表的引用(或指针)。除此以外,Lua语言不会进行隐藏的拷贝(hiddencopies)或创建新的表--创建表a={}--创建空表k="x"a[k]=10--键“x”值10a[20]="great"--键20值“great”print(a["x"])-->10
https://github.com/galenho/crossover.git一个跨平台的lua游戏服务器开发框架,该框架采用多线程并发来处理消息,开发者只需要调用相应的接口函数并绑定相应的回调函数即可,在逻辑层表现为单线程的开发模式,使开发者易用,易调试,易维护,易扩展,同时拥有快速的响应能力。   框架使用面
参考链接:https://www.runoob.com/lua/lua-metatables.htmlhttps://www.jianshu.com/p/cb945e7073a3 元表是一个table,可以让我们改变table的行为,每个行为有对应的元方法例如,对table进行设置键值,查找键值,运算等,就会触发对应的元方法1--__index:table被访问时,如果找不到这
https://github.com/yuin/gopher-luahttps://github.com/yuin/gopher-lua Lua5.1ReferenceManual-contentshttp://www.lua.org/manual/5.1/ go中使用luapackagemainimport( lua"github.com/yuin/gopher-lua")funcmain(){ l:=lua.NewState() d
编译问题不要留到运行时才跑出来啊。早上9:00-中午3:00,6个小时,服了自己了。 写了一个测试,springboot+redis+lua执行到redisTemplate.execute(redisScript,idList)的时候一直报错,integer无法转换为string。我一直以为是lua脚本写错了,翻文档翻过来又翻过去,写法变了又变,还是解
        。。是字符串连接符,字典用=号连接,  注意fordoend都是连一起,  注意ifthen,  如果local在函数里,是可以访问,非local出了函数一样能用,  doend代码块也是一样,    注意点号表示,只能key是字符串,  注意括号不是必须
C语言与Lua之间的相互调用详解写一个C调用Lua的Demo编译运行C语言调用Lua编译问题总结正确的编译命令问题1:缺少-lm参数问题2:缺少-ldl参数​1、为什么会出现undefinedreferenceto‘xxxxx’错误?​2、-l参数和-L参数写一个C调用Lua的Demo编译运行add.c内容//你需要
1、动态输出打开E:\study\openresty\openresty-1.19.9.1-win64目录下的confginx.conf文件在server中增加一下代码 location/hello{ default_typetext/html; content_by_lua'ngx.say("<p>hello,world</p>")'; }运行后,效果如下图localhost
参见:lipp/lua-websockets:WebsocketsforLua.(github.com)github网址可能需手动转换lipp.github.com/lua-websockets/>github.com/lipp/lua-websocketswebsockets为底层的类似于TCP、UDP的socket(实现上基于更底层的socket),不同于上层的webserver服务端(Service)需并行地支持多
lua发送消息到rabbitmq,我们选择类库lua-resty-rabbitmqstomp 来完成这个任务。类库安装:进入nginx.conf中 lua_package_path 中对应的目录下的resty目录(没有则创建),执行:wget-chttps:/aw.githubusercontent.com/wingify/lua-resty-rabbitmqstomp/master/libes
1Lua介绍Lua是一门以其性能著称的脚本语言,被广泛应用在很多方面。Lua一般用于嵌入式应用,现在越来越多应用于游戏当中,魔兽世界,愤怒的小鸟都有用到。优势Lua极易嵌入到其他程序,可当做一种配置语言。提升应用性能,比如:游戏脚本,nginx,wireshark的脚本兼容性强,可以直接使用C