llvm数据流分析

news2025/3/12 16:43:22

llvm数据流分析

  • 1.数据流分析
  • 2.LLVM实现
    • 2.1.常量传播
    • 2.2.活跃性分析

相关参考文档:DataFlowAnalysisIntro、ustc编译原理课程、南大程序分析课程1、南大程序分析课程2。

1.数据流分析

数据流分析在编译优化等程序分析任务上都有重要应用。通常数据流分析可被抽象为一个IFDS-based worklist算法。核心是给每个CFG结点 s s s (结点表示一个语句或者基本块)计算两个集合 IN [ s ] \text{IN}[s] IN[s] OUT [ s ] \text{OUT}[s] OUT[s]

  • meet运算: IN [ s ] = meet s i ∈ prev ( s ) OUT [ s i ] \text{IN}[s] = \text{meet}_{s_i \in \text{prev}(s)} \text{OUT}[s_i] IN[s]=meetsiprev(s)OUT[si] meet [ s ] = ∪ s i ∈ next ( s ) IN [ s i ] \text{meet}[s] = \cup_{s_i \in \text{next}(s)} \text{IN}[s_i] meet[s]=sinext(s)IN[si] (反向数据流传播任务)。

  • update: OUT [ s ] = gen [ s ] ∪ ( IN [ s ] − kill [ s ] ) \text{OUT}[s] = \text{gen}[s] \cup (\text{IN}[s] - \text{kill}[s]) OUT[s]=gen[s](IN[s]kill[s]), IN [ s ] = gen [ s ] ∪ ( OUT [ s ] − kill [ s ] ) \text{IN}[s] = \text{gen}[s] \cup (\text{OUT}[s] - \text{kill}[s]) IN[s]=gen[s](OUT[s]kill[s]) (反向数据流传播任务)。通常具体实现中 gen \text{gen} gen kill \text{kill} kill 可由update操作完成。

数据流分析的worklist算法 (前向) 如下。worklist算法会不断循环直到 IN [ s ] \text{IN}[s] IN[s] OUT [ s ] \text{OUT}[s] OUT[s] 收敛(集合不再变大)为止。 IN [ s ] \text{IN}[s] IN[s] OUT [ s ] \text{OUT}[s] OUT[s] 的元素为fact,fact的具体类型视数据流分析任务而定。需要注意的是这个分析框架是flow-sensitive & path-insensitive的,path-sensitive的分析(符号执行)不会有 meet 操作,而是直接将当前分析状态 fork 多份然后将每个 fork 后的状态单独添加到 worklist中。

def DataFlowAnalysis(cfg: CFG):
    for stmt in CFG:
	    worklist.append(stmt)
	    Out[stmt].clear()
    while (!worklist.empty()):
        cur_stmt = worklist.pop()
        for prev_stmt in Prev(cur_stmt):
            meet(In[cur_stmt], Out[prev_stmt])
        changed = update(cur_stmt, Out[cur_stmt], In[cur_stmt])
        if (changed):
            worklist.append(cur_stmt)
数据流分析任务方向meet操作fact示例genkill
到达定值分析 (reaching-define analysis)前向 ∪ \cup 变量定义l: x = ax: lx: prevL, prevL 为上一个定义 x 的语句
常量传播 (constant propagation)前向 ∪ \cup 变量值x = ax: val(a), val(a) 表示 a 的值x: _, _ 表示之前的值
活跃变量分析 (live variabiles analysis)反向 ∪ \cup 变量x = nnx
可用表达式分析 (available expressions analysis)前向 ∩ \cap 表达式a = x op yx op q所有涉及 a 的表达式
  • 1.reaching-define的结果可以用来构造数据依赖图(参考Joern),比如 l: x = a,中 OUT [ l ] = { ( a , l 1 ) , ( x , l ) } \text{OUT}[l] = \{(a, l_1), (x, l)\} OUT[l]={(a,l1),(x,l)},那么可以知道引用的 al1 出定义,添加边 l 1 → a l l_1 \stackrel{a}{\rightarrow} l l1al。同时对于循环中的 x = y op z,如果 yz 的定义在循环外,那么可以把 y op z 移动到循环外。

  • 2.常量传播的结果 OUT [ s ] \text{OUT}[s] OUT[s] 表示 s s s 处为常量的变量情况。可以用来对变量进行常量替换。

  • 3.活跃变量的分析结果中 IN [ s ] \text{IN}[s] IN[s] 表示 s s s 处的活跃变量,可以用来优化寄存器分配。

  • 4.可用表达式的分析结果可以用来删除公共子表达式减少多余计算。

除了上述提到的经典问题外,flow-sensitive的指针分析抽象解释也是一个数据流分析问题。指针分析的fact是指向关系 (比如 x → o x \rightarrow o xo 表示 x x x 可能指向 o o o),meet操作是 ∪ \cup genkill 则需要考虑strong update和weak update。具体可参考blog。

  • 对于赋值语句 s : x = y s: x = y s:x=y gen [ s ] = { x → o ∣ ∀ o ∈ pts ( y ) } \text{gen}[s] = \{x \rightarrow o \mid \forall o \in \text{pts}(y)\} gen[s]={xoopts(y)} kill [ s ] = { x → _ } \text{kill}[s] = \{x \rightarrow \_ \} kill[s]={x_},进行strong update。

  • 对于 s : ∗ x = y s: *x = y s:x=y gen [ s ] = { z → o ∣ ∀ z ∈ pts ( x )    ∀ o ∈ pts ( y ) } \text{gen}[s] = \{z \rightarrow o \mid \forall z \in \text{pts}(x) \; \forall o \in \text{pts}(y) \} gen[s]={zozpts(x)opts(y)} kill \text{kill} kill 情况就比较复杂。

    • 如果 x x x 只指向内存对象 z z z,那么 kill [ s ] = { z → _ } \text{kill}[s] = \{z \rightarrow \_ \} kill[s]={z_},依旧可以进行strong update。

    • 如果 x x x 可能指向好几块内存对象,那么 kill [ s ] = { } \text{kill}[s] = \{\} kill[s]={},此时进行的是weak update。

抽象解释可参考blog,fact为变量值域,SVF的抽象解释模块对 p = c (c 为整型常量,p 为整型变量) 生成的fact为 p → ⟨ [ c , c ] , ⊤ ⟩ p \rightarrow \langle [c, c], \top \rangle p⟨[c,c],,包括值域 [ c , c ] [c, c] [c,c] 和指向集 ⊤ \top ,同时进行抽象解释和指针分析(也就是对应paper提到的Combined Abstract Domains)。和指针分析一样,抽象解释中 store 指令可能涉及到 kill 操作,也需要考虑strong update和weak update。

难点:C/C++中别名的存在是数据流分析的一大难点(参考blog),不同变量对同一内存的引用导致多个定义可能指向一个变量,通常数组指针结构体的使用可能带来别名关系,为了保证soundness,越保守的策略通常越不会 kill 不确定的fact。通常可以先进行一个指针分析来确定大致别名关系。或者参考Dr.Checker和Falcon将指针分析和到达定值一起分析。考虑到二者到达收敛的循环次数可能不一致,Dr.Checker预设了一个循环次数上限,不过这种策略可能会导致指针分析不够可靠(unsound)。对于Falcon,它用到了两个集合 E E E S S S 来分别保存top-level pointer的指向集和address-taken object的到达定值, ( π , l , q ) ∈ S ( o ) (\pi, l, q) \in S(o) (π,l,q)S(o) 表示 store 语句 l : ∗ p = q l: *p = q l:p=q o ∈ pts ( p ) o \in \text{pts}(p) opts(p) 因此 o o o l l l 处的到达定值为 ( π , q ) (\pi, q) (π,q) ( π \pi π 是路径条件,Falcon采用路径敏感的分析策略),Falcon中只有 store 语句会更新 S S S 集合,其它语句(load, gep 等) 会查询 S S S 并更新top-level variable对应 E E E 集合。

2.LLVM实现

2.1.常量传播

llvm提供了source code level和LLVM IR level的常量传播算法,source code level通过ConstantPropagationAnalysis类实现,不过由于只是简单实现没法适用到real-world环境下,因此只是放到unittests下。在这个实现中变量值有上界 ⊤ \top , 下届 ⊥ \bot , 常量值 c c c 3种类型。用 a a a 表示任意值,其中有 ⊥ ∪ a = a \bot \cup a = a a=a ⊤ ∪ a = ⊤ \top \cup a = \top a=, c ∪ c = c c \cup c = c cc=c, c 1 ∪ c 2 = ⊤ c_1 \cup c_2 = \top c1c2=。对于给定 Stmt* S, ConstantPropagationAnalysis 类通过AST分析识别其中是否存在: 1.变量声明并初始化 (int x = 42)、2.赋值语句 (x = 24)、3.复合赋值 (x += 4)。对于前两种语句 ConstantPropagationAnalysis 会尝试计算右表达式的常量值,如果不是常量则设置左表达式对应变量为 ⊤ \top ,反之设置为具体值。对于复合赋值直接将左变量设置为 ⊤ \top 。这里会调用 Expr::EvaluateAsInt 对表达式进行求值。x = a 中,如果 a 是常量那么返回常量值,如果 a 是变量似乎不会再递归求值直接返回 ⊤ \top

LLVM IR level的通过SCCPPass来进行函数内Sparse Conditional Constant Propagation (SCCP)。跨函数传播通过IPSCCPPass实现。SCCP相比普通常量传播的改进在于增加对分支条件的处理。普通常量传播不会考虑分支条件为常量 (truefalse),SCCP则会基于分支条件的常量值删除不可达分支。根据这个stackoverflow post,启用常量传播pass的前提条件应该是先使用mem2reg优化。

函数内常量传播的入口为runSCCP函数,lattice定义为ValueLatticeElement,变量值通常通常有以下状态。

状态含义
unknown该值尚无已知信息,可以转换为任何其他状态。通常作为起始状态
undef该值是 UndefValue 或产生 undef,可以与 constantconstantrange meet后转化为 constant。可以转换为 constantconstantrange_including_undefoverdefined
constant该值是一个特定的常量,不能是 undef
notconstant该值已知不是某个特定常量(适用于非整数类型)。
constantrange该值属于某个范围(仅适用于整数类型)。
constantrange_including_undef该值属于某个范围,但也可能是 undefundefconstantrange meet后为 constantrange_including_undef。与其他fact meet后仍为 constantrange_including_undef
overdefined该值无法精确建模,即可能具有多个动态值,不再进行任何状态转换。相当于 ⊥ \bot

涉及到merge (meet)操作时,状态转移关系定义在ValueLatticeElement::mergeIn,状态转移关系如下。对于 res = lhs op rhs,运算结果如下:

lhs\rhsunknownoverdefinedundefconstantconstantrangenotconstantconstantrange_including_undef
unknownunknownoverdefinedundefconstantconstantrangenotconstantconstantrange_including_undef
overdefinedoverdefinedoverdefinedoverdefinedoverdefinedoverdefinedoverdefinedoverdefined
undefoverdefinedundefconstantconstantrangeoverdefinedoverdefined
constantoverdefinedoverdefinedconstantconstant (lhs == rhs), overdefined (lhs != rhs)overdefinedoverdefinedoverdefined
notconstantoverdefinedoverdefinedoverdefinedoverdefinedoverdefinednotconstantoverdefined
constantrangeoverdefinedoverdefinedoverdefinedoverdefinedconstantrange (值域会进行union)overdefinedoverdefined

其中还有一个辅助类SCCPInstVisitor来收集中间结果。

成员变量类型作用
BBExecutableSmallPtrSet<BasicBlock *, 8>记录可执行的基本块
ValueStateDenseMap<Value *, ValueLatticeElement>记录 Value 的lattice状态, Value 通常对应1个llvm指令
StructValueStateDenseMap<std::pair<Value *, unsigned>, ValueLatticeElement>记录结构体字段的 lattice 状态
TrackedGlobalsDenseMap<GlobalVariable *, ValueLatticeElement>记录全局变量的lattice状态
TrackedRetValsMapVector<Function *, ValueLatticeElement>记录单返回值函数的返回值状态
TrackedMultipleRetValsMapVector<std::pair<Function *, unsigned>, ValueLatticeElement>记录多返回值函数的返回值状态
MRVFunctionsTrackedSmallPtrSet<Function *, 16>存储 TrackedMultipleRetVals 中的函数,方便查找
MustPreserveReturnsInFunctionsSmallPtrSet<Function *, 16>存储返回值不可修改的函数
TrackingIncomingArgumentsSmallPtrSet<Function *, 16>存储需要分析参数的函数
OverdefinedInstWorkListSmallVector<Value *, 64>记录即将 overdefined 的指令,加速SCCP收敛
InstWorkListSmallVector<Value *, 64>记录待 SCCP 处理的指令,加速SCCP收敛
BBWorkListSmallVector<BasicBlock *, 64>记录待 SCCP 处理的基本块 ,主worklist
KnownFeasibleEdgesDenseSet<Edge>记录已确认的CFG边,避免重复计算
AnalysisResultsDenseMap<Function *, AnalysisResultsForFn>存储函数SCCP分析结果

这里worklist算法大致如下,相比原版worklist算法再次做了些优化:

  • 1.visit(BB) 表示对basic block BB 下的所有指令 (Value) 进行 meetupdate

  • 2.markUsersAsChanged 会对 I 的所有 users 进行 visit,也就是优先处理 OverdefinedInstWorkList 中的元素和非 overdefined 的结构体元素。

  • 3.优先处理 OverdefinedInstWorkList 的原因是如果变量 (Value) I 的状态为 Overdefined,它的 user 状态多半为 Overdefined,没法再收敛了,减少这部分的迭代次数能加速SCCP收敛。

  • 4.InstWorkList 主要用于处理值从 undef 变成 constant 的情况。尽早传播 constant,让更多值变成 constant,提高优化效果。

def SCCP_Worklist():
    while !BBWorkList.empty() or !OverdefinedInstWorkList.empty() or !InstWorkList.empty():
        # 优先处理 OverdefinedInstWorkList,尽快传播 overdefined 状态
        while !OverdefinedInstWorkList.empty():
            I = OverdefinedInstWorkList.pop()
            markUsersAsChanged(I)

        # 处理普通指令工作列表
        while !InstWorkList.empty():
            I = InstWorkList.pop()
            # 只处理未 overdefined 的值
            if isStructType(I) or not isOverdefined(I):
                markUsersAsChanged(I)

        # 处理基本块工作列表
        while !BBWorkList.empty():
            BB = BBWorkList.pop()
            visit(BB)

具体的update规则可以参考 SCCP::visitXXInst 函数。这里有意思的是:

  • 1.SCCPInstVisitor::visitStoreInst,其中只对全局变量进行处理,也就是局部变量的 store 都不操作。

  • 2.SCCPInstVisitor::visitLoadInst中如果加载的是结构体变量直接为 overdefined ( ⊥ \bot ),SCCPInstVisitor::resolvedUndefsIn中对于其它 load 将值设为 undef。这可能是考虑到别名的一种保守处理。根据这个stackoverflow post,对于下面代码,llvm编译器压根没做优化。原因是 c = a + b 中存在 load aload bstore 1, a, store 2, b 操作。而 SCCPPass 没有详细地处理,因此,没能优化。也是因此开启该优化的一个前提是先用 mem2reg

int a,b,c;
a = 1;
b = 2;
c = a + b;
  • 3.SCCPInstVisitor::visitCmpInst中调用ValueLatticeElement::getCompare对 r = cmp op1, op2 计算 r 的值,最终结果有 overdefined, constant(1), constant(0) 3种。

  • 4.SCCPInstVisitor::visitSelectInst中会根据 cmp 的计算结果计算当前值。如果 cond 的常量值为 1,则选 true branch的值,反之亦然;如果为 overdefined,则合并两个分支的常量值。

runSCCP结尾会调用SCCPSolver::simplifyInstsInBlock对变量进行常量替换以及删除无用指令并调用removeNonFeasibleEdges删除infeasible CFG edge以及随后删除无用基本块。

2.2.活跃性分析

llvm提供两个level的活跃性(包括活跃变量和可用表达式)分析:source code和machine code,source code level可通过clang static analyzer的LiveVariablesDumper (CSA参数为 debug.DumpLiveVars) 以及LiveExpressionsDumper(CSA参数为 debug.DumpLiveExprs)打印活跃变量信息。在machine code level会通过LiveVariablesAnalysis进行活跃变量分析。

source code level的活跃性分析这位大佬的两篇blog(clang中的活跃性分析讲解源码,clang中的活跃性分析(续)给出示例)对clang的源代码进行了具体说明。负责活跃性分析的包括LiveVariables和RelaxedLiveVariables类,后者相比前者分析结果更不精确,不过source code level的活跃性分析主要是辅助clang static analyzer而不是编译优化,因此在analyzer中反而应用了 RelaxedLiveVariables,而 LiveVariables 纯粹只是用来dump source code level的活跃变量信息。transfer function的定义在TransferFunctions中,活跃性分析的dataflow fact分别用VarDecl*(变量分析)和Expr*(表达式分析)表示。

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

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

相关文章

HTML-网页介绍

一、网页 1.什么是网页&#xff1a; 网站是指在因特网上根据一定的规则&#xff0c;使用 HTML 等制作的用于展示特定内容相关的网页集合。 网页是网站中的一“页”&#xff0c;通常是 HTML 格式的文件&#xff0c;它要通过浏览器来阅读。 网页是构成网站的基本元素&#xf…

【C#学习笔记03】进制转换与反码、补码、原码

1. 进制转换 计算机中的数据通常以二进制形式存储&#xff0c;但在编程和调试过程中&#xff0c;我们经常需要与十进制、八进制和十六进制打交道。因此&#xff0c;掌握进制转换是C语言学习中的重要一环。 1.1 进制的基本概念 二进制&#xff08;Binary&#xff09;&#xff…

python学智能算法(七)|KNN邻近算法

【1】引言 前述学习进程中&#xff0c;已经了解了一些非常经典的智能算法&#xff0c;相关文章包括且不限于&#xff1a; python学智能算法&#xff08;三&#xff09;|模拟退火算法&#xff1a;深层分析_模拟退火 动画演示-CSDN博客 python学智能算法&#xff08;四&#x…

Java数据结构第二十二期:Map与Set的高效应用之道(一)

专栏&#xff1a;Java数据结构秘籍 个人主页&#xff1a;手握风云 目录 一、Map和Set 1.1. 概念 二、搜索树 2.1. 概念 2.2. 查找操作 2.2. 插入操作 2.3. 删除操作 2.4. 性能分析 三、搜索 3.1. 概念及场景 3.2. 模型 四、Map 4.1. Map的说明 3.2. Map的使用 五…

兴达易控modbusTCP转profinet接防撞雷达测试

modbusTCP转profinet接防撞雷达测试 随着工业自动化程度的不断提高&#xff0c;现场设备之间的通信需求日益增长。ModbusTCP作为一种广泛应用的工业通信协议&#xff0c;因其简单、可靠的特点&#xff0c;被广泛应用于各种自动化设备中。而Profinet作为工业以太网的一种&#…

flutter实践:断点调试踩坑

问题&#xff1a;使用VSCode开发flutter,最近突然开始打断点不生效&#xff0c;程序可以attach,修改有日志输出&#xff0c;但是断点处怎么都停不了&#xff0c;程序异常断点会停。 分析&#xff1a;开始误以为是flutterSDK出了问题折腾了一天&#xff0c;后来又怀疑是lauch.j…

STM32——GPIO介绍

GPIO(General-Purpose IO ports,通用输入/输出接口)模块是STM32的外设接口的核心部分,用于感知外界信号(输入模式)和控制外部设备(输出模式),支持多种工作模式和配置选项。 1、GPIO 基本结构 STM32F407 的每个 GPIO 引脚均可独立配置,主要特性包括: 9 组 GPIO 端口…

Photo Works在线图片编辑器:一键修复老照片,轻松焕新记忆

★【概况介绍】 今天突然收到我的朋友电脑出故障了,截图给我,我一看就知道这个是缺少必要的组件引起的故障。结合这个问题,我来谈谈自己的解决思路和方法,希望能够帮助到大家。帮助大家是我最开心的事情。以前只是帮朋友解决问题,没有记录下来,刚刚接触到这个平台,刚好可…

SQLiteStudio:一款免费开源跨平台的SQLite管理工具

目录 1.简介 2.下载与安装 3.实现分析 4.总结 1.简介 SQLiteStudio 是一款专门用于管理 SQLite 数据库的图形化工具&#xff0c;由波兰开发者开发并维护。由于 SQLite 以其轻量级、零配置、嵌入式等特性被广泛应用于各种小型项目、移动应用和桌面应用中&#xff0c;而 SQLi…

Markdown 语法入门指南(VSCode 版)

此博客为一份详细的 Markdown 语法入门指南&#xff0c;专门针对在 VSCode 上使用 Markdown 的零基础用户。这份指南将包括 Markdown 的基础语法、在 VSCode 中的安装与使用方式、常见问题及注意事项。 Markdown 是一种轻量级标记语言&#xff0c;使用纯文本符号来标记格式&am…

PostgreSQL学习笔记:PostgreSQL vs MySQL

PostgreSQL 和 MySQL 都是广泛使用的关系型数据库管理系统&#xff0c;它们有以下一些对比&#xff1a; 一、功能特性 1. 数据类型支持 PostgreSQL&#xff1a;支持丰富的数据类型&#xff0c;包括数组、JSON、JSONB、范围类型、几何类型等。对于复杂数据结构的存储和处理非…

内存检测工具——Qt Creator

前言 检测内存错误的工具&#xff0c;有很多个&#xff0c;我今天粗浅的学了一下可在Qt上使用的工具们&#xff1a; Dr.Memory 工具之前我曾在关注的博主上看到相关的博客&#xff1a;C(Qt)软件调试---内存调试器Dr.Memory&#xff08;21&#xff09;_dr. memory-CSDN博客 今…

2.4 基于Vitest的单元测试基础设施搭建

文章目录 1. 现代单元测试体系解析测试金字塔演进Vitest核心定位2. 基础设施架构设计整体架构图3. 环境配置全流程3.1 基础环境搭建3.2 配置文件`vitest.config.ts`3.3 测试环境初始化4. 测试用例编写规范4.1 基础测试示例4.2 Vue组件测试4.3 异步逻辑测试5. Mock策略深度优化5…

⭐算法OJ⭐链表排序【归并排序】(C++/JavaScript 实现)

文章目录 148. Sort List解题思路归并排序的基本思想归并排序的步骤 实现实现步骤C 实现JavaScript 实现 复杂度总结 148. Sort List Given the head of a linked list, return the list after sorting it in ascending order. 解题思路 链表排序问题可以通过多种方法解决&am…

SegMAN模型详解及代码复现

SegMAN模型概述 模型背景 在深入探讨SegMAN模型之前&#xff0c;我们需要了解其研究背景。在SegMAN出现之前&#xff0c;计算机视觉领域的研究主要集中在以下几个方面&#xff1a; 手工制作方法&#xff0c;如SIFT基于卷积神经网络(CNN)的方法&#xff0c;如STN和PTN对平移、…

Manus AI:多语言手写识别的技术革命与未来图景

摘要&#xff1a;在全球化浪潮下&#xff0c;跨语言沟通的需求日益迫切&#xff0c;但手写文字的多样性却成为技术突破的难点。Manus AI凭借其多语言手写识别技术&#xff0c;将潦草笔迹转化为精准数字文本&#xff0c;覆盖全球超百种语言。本文从技术原理、应用场景、行业价值…

Stable Diffusion游戏底模推荐

一、基础通用型底模 SDXLbase &#x1f4da; 官方原版底模&#xff0c;支持1024x1024高清出图&#xff0c;适用于各类游戏场景和角色的基础生成&#xff0c;建议作为微调训练的基准模型。 来源: 相关搜索结果 写实风格搭配推荐 &#x1f3a8; 搭配 9realisticSDXL 或 麻袋real…

理解字符流和字节流,节点流和处理流、缓冲流、InputStreamReader、BufferInputStream、BufferReader...

DAY10.2 Java核心基础 IO流 字符流和字节流 字符流和字节流在每次处理数据的单位不同&#xff0c;一个是字符&#xff0c;一个是字节 如果复制文件类型是文本类型&#xff0c;字节流字符流都可以 如果复制的文件类型是非文本类型&#xff0c;则只能使用字节流&#xff0c;使…

DBeaver安装教程+连接TDengine数据库

为TDengine安装的DBeaver教程 安装 23.1.1 版本以上的DBeaver 因为官方文档说这个版本之上的DBeaver才支持TDengine内嵌前往DBeaver 官方文档进行版本下载滑到链接最下面点击进入 点击download&#xff0c;进入选择下载版本 等待下载成功即可双击自行安装 打开数据库连接TDen…

【三维重建】Proc-GS:使用3DGS的程序性城市建筑生成

标题&#xff1a;《Proc-GS: Procedural Building Generation for City Assembly with 3D Gaussians》 项目&#xff1a;https://city-super.github.io/procgs/ 来源&#xff1a;香港中文大学&#xff1b;上海人工智能实验室 等 文章目录 摘要一、 程序代码定义 (Procedural Co…