美文网首页
代码生成

代码生成

作者: 小小青蛙不怕风吹雨打 | 来源:发表于2017-03-10 17:31 被阅读0次

语法解析流程简述

lua之类的脚本语言先把源代码翻译成字节码,而后再使用虚拟机执行字节码。
而从源代码到字节码要经过多个过程:

词法解析,语法解析,语义解析,代码生成。
源代码->单词流->语法树->带语义的语法树->字节码。

语义解析主要是完善语法树,添加语义信息:

  1. 设置变量作用域。
  2. 设置break和continue对应的循环语句。

lua的官方实现做了优化,将4步流程压缩成一个。词法解析提供接口供语法解析调用,语法解析、语义解析压缩成一个流程,代码生成提供接口供语法解析使用。这也造成解析代码难以看懂修改。

luna的实现:词法解析还是提供接口,语法解析,语义解析,代码生成是分开的,就好懂很多了。(其中的观察者模式用的不适合)

oms的实现中语义解析合并代码生成中。

词法解析,语法解析,代码生成。
源代码->单词流->语法树->字节码。

指令生成

遍历语法树生成代码的过程比较无脑,对每个语法树节点类型写一个解析函数。
一个例子。

void HandleBlock(Block tree)
{
    foreach(var stmt in tree.statements)
    {
        if (stmt is DoStatement)
            HandleDoStatement(stmt as DoStatement);
        else if (stmt is WhileStatement)
            HandleWhileStatement(stmt as WhileStatement);
        else if (stmt is IfStatement)
            HandleIfStatement(stmt as IfStatement);
        else if (stmt is ForStatement)
            HandleForStatement(stmt as ForStatement);
        else if (stmt is ForEachStatement)
            HandleForEachStatement(stmt as ForEachStatement);
        else if (stmt is ForInStatement)
            HandleForInStatement(stmt as ForInStatement);
        else if (stmt is FunctionStatement)
            HandleFunctionStatement(stmt as FunctionStatement);
        else if (stmt is LocalFunctionStatement)
            HandleLocalFunctionStatement(stmt as LocalFunctionStatement);
        else if (stmt is LocalNameListStatement)
            HandleLocalNameListStatement(stmt as LocalNameListStatement);
        else if (stmt is ReturnStatement)
            HandleReturnStatement(stmt as ReturnStatement);
        else if (stmt is BreakStatement)
            HandleBreakStatement(stmt as BreakStatement);
        else if (stmt is ContinueStatement)
            HandleContinueStatement(stmt as ContinueStatement);
        else if (stmt is AssignStatement)
            HandleAssignStatement(stmt as AssignStatement);
        else
            HandleExpRead(stmt);
    }
}

指令系统和局部变量分配

lua的指令是带寄存器地址的,最多3地址,一个地址一个字节,也就限制了局部变量的总数不可能超过256个。
自己实现时,指令也是带寄存器地址的,最直接的好处是,指令能少很多。然后局部变量分配就是基于栈的了,简单方便。
简化的指令bit分配。

code: int32_t
A   : uint8_t
B   : uint8_t
C   : uint8_t
Bx  : int16_t (B+C)

局部变量每个函数单独分配,从0开始,定义一个局部变量就+1,退出block时回收当前作用域内的。
临时分配的也要及时回收,如

void HandleAssignStatement(AssignStatement tree)
{
    HandleExpList(tree.exp_list, tree.var_list.Count);
    // var list
    int register = GetNextRegisterId();
    ResetRegisterId(register + tree.var_list.Count);
    for(int i = 0; i < tree.var_list.Count; ++i)
    {
        HandleVarWrite(tree.var_list[i], register + i);
    }
    ResetRegisterId(register);// 回收临时分配的局部变量。
}

语义分析部分

在遍历语法树时,要维护作用域树循环结构链表

//"while" exp "do" block "end"
void HandleWhileStatement(WhileStatement tree)
{
    EnterLoop();// 进入循环
    EnterBlock();// 进入新的作用域
    HandleExpRead(tree.exp);
    // jump to loop tail when expression return false
    var f = GetCurrentFunction();
    var code = Instruction.ABx(OpType.OpType_JmpFalse, GetNextRegisterId(), 0);
    int index = f.AddInstruction(code, -1);
    AddLoopJumpInfo(JumpType.JumpTail, index);
    HandleBlock(tree.block);// 变量循环block
    LeaveBlock();// 退出作用域
    // jump to loop head
    code = Instruction.Bx(OpType.OpType_Jmp, 0);
    index = f.AddInstruction(code, -1);
    AddLoopJumpInfo(JumpType.JumpHead, index);
    LeaveLoop();// 退出循环
}

相关文章

  • [Swift]语法树

    生成语法书 生成最简洁的SIL代码 生成LLVM IR代码 生成汇编代码

  • 代码生成器,目标语言,目标代码中的地址

    代码生成器,目标语言,目标代码中的地址 代码生成器 代码生成器的是生成正确的代码,代码生成器的设计依赖于中间表示形...

  • MybatisPlus学习笔记(九)------MybatisP

    代码生成器 MybatisPlus与Mybatis的代码生成器对比 Mybatis是基于xml生成代码; Myba...

  • Android 打造编译时注解解析框架,了解一下

    (一)了解(二)分析,实战(三)结合项目源码(四)类、代码生成(五)代码生成的几种方式(代码生成路径)

  • 编译器前端和后端

    编译器粗略分为词法分析,语法分析,类型检查,中间代码生成,代码优化,目标代码生成,目标代码优化。把中间代码生成及之...

  • 代码生成

    这次的小课堂主要是讲公司的代码生成框架。 内容很多,主要内容都在视频了,所以慢慢看吧。 大部分内容都涉及配置文件什...

  • 代码生成

    语法解析流程简述 lua之类的脚本语言先把源代码翻译成字节码,而后再使用虚拟机执行字节码。而从源代码到字节码要经过...

  • Android组件化实现方案(二)

    APT(Annotation Processing Tool),根据注解自动给生成代码。JavaPoet,代码生成...

  • Mybatis-plus自定义模板生成

    下方代码引用自mybatis-plus的代码生成模块,点击运行即可生成代码。本篇第一部分将介绍代码生成的使用,第二...

  • 2019-04-28

    前端 生成工程代码时选择的前端组件是Angular,所以前端资源会按如下流程生成:a, 生成工程代码或者实体代码的...

网友评论

      本文标题:代码生成

      本文链接:https://www.haomeiwen.com/subject/bmlogttx.html