UVM验证效率提升:利用仿真器保存恢复机制消除冗余配置周期

news2026/5/21 22:22:58
1. 验证环境中的冗余周期之痛一个普遍存在的效率瓶颈在芯片验证领域尤其是使用UVMUniversal Verification Methodology构建的复杂验证环境中我们常常会面临一个看似不起眼、实则消耗巨大的问题冗余的仿真周期。想象一下你正在验证一个集成了多个接口如DDR控制器、PCIe、以太网MAC的SoC。在启动每一个具体的功能测试用例之前无论这个测试是验证数据路径的吞吐量还是检查某个特定中断的处理逻辑DUTDesign Under Test和整个验证平台都需要经历一个漫长而固定的“启动配置”阶段。这个阶段通常包括释放复位、等待PLL锁定、配置各个接口控制器的基础寄存器、初始化内部存储器、加载引导程序或固件镜像等。这个过程可能耗时数秒甚至数十秒的仿真时间。对于单个测试来说这是必要的成本。但当你需要运行成百上千个测试用例时问题就凸显出来了——每个测试都从时间零点Time 0开始重复执行这一模一样的配置序列。大量的计算资源CPU时间、内存和工程师的等待时间都被浪费在了这些重复劳动上。这就像每次做饭前都要重新从零开始种菜、收割、生火而不是直接利用已经准备好的食材和灶台。更具体地说这种冗余体现在两个层面。一是仿真时间的冗余即仿真工具如VCS、Xcelium需要反复模拟相同的信号跳变和事务消耗大量算力。二是工程师生产力的冗余验证工程师需要反复等待漫长的启动过程或者编写复杂的脚本去跳过这些阶段增加了环境维护的复杂度和出错风险。本文的核心就是探讨如何利用Verilog和主流仿真器提供的“保存-恢复”Save-Restore机制结合UVM框架的灵活性巧妙地规避这些冗余周期实现验证效率的显著提升。2. 核心思路拆解为何“保存-恢复”是解药以及UVM如何适配2.1 仿真器“存档点”机制的原理解读解决重复配置问题的直觉想法是“能不能把配置好的状态存起来下次直接用” 幸运的是主流数字仿真器如Synopsys VCS, Cadence Xcelium, Siemens EDA Questa都内置了这个能力通常称为Checkpoint或Save/Restore。其底层原理可以类比于操作系统的休眠功能或游戏中的存档点。仿真器在运行时会将整个仿真的状态包括所有寄存器变量、线网值、队列内容、甚至部分内存镜像序列化并保存到硬盘上的一个或多个文件中。这个状态快照Snapshot包含了恢复到该仿真时间点所需的一切信息。当执行恢复操作时仿真器从这些文件读回数据精确地将内存中的仿真状态重建出来然后可以从这个“存档点”继续运行。在Verilog层面这通常通过系统任务如$save,$restart或通过仿真器的命令行接口CLI/TCL来控制。关键在于这个保存的状态是完整的、确定性的。它不仅保存了DUT内部所有触发器和锁存器的值也保存了验证平台Testbench中所有UVM组件如uvm_sequencer,uvm_driver,uvm_monitor以及所有事务级模型TLM端口、队列、配置数据库uvm_config_db的状态。这意味着恢复后整个系统仿佛从未中断过可以无缝衔接。2.2 UVM测试的动态性与静态配置的冲突然而直接使用仿真器的保存/恢复功能会遇到一个与UVM动态特性相关的挑战。UVM测试的核心激励来源于序列Sequence。序列是动态生成的其内容和随机性通常在run_phase或main_phase中才被确定和执行。如果我们简单地在配置完成后保存状态然后恢复并运行另一个测试我们会面临一个问题恢复后的环境会试图重复执行之前测试已经执行过的序列或者因为序列已经结束而无事可做。我们真正期望的流程是执行一次公共的、耗时的配置阶段并保存状态 - 从保存的状态恢复 - 为每次恢复的仿真注入不同的、目标明确的测试序列。这就要求我们将UVM环境的“静态配置”和“动态激励”进行解耦。传统的UVM测试写法往往在build_phase或connect_phase通过uvm_config_db设置好整个测试运行的default_sequence。这种方式是静态的、一次性的无法适应“一次配置多次不同测试”的恢复场景。因此我们的策略需要进行一个关键转变将测试序列的启动时机从环境的构建/连接阶段延迟到main_phase真正开始执行的时刻。并且序列的类型需要能在恢复仿真时通过外部参数如命令行plusarg动态指定。这样恢复后的仿真才能跳过重复的配置直接奔向不同的测试目标。3. 实战改造以UVM UBUS示例为例的代码级详解理论说再多不如一行代码。我们以UVM自带的经典ubus示例项目为基础演示如何实施这一改造。这个环境包含一个简单的总线、主从设备非常适合说明问题。假设我们有两个测试test_read_modify_write和test_r8_w8_r4_w4它们仅在顶层运行的序列read_modify_write_seq和r8_w8_r4_w4_seq上不同而之前的总线初始化、设备配置等步骤完全一致。3.1 改造第一步重构测试基类延迟序列启动首先我们创建一个所有测试都可以继承的基础测试类base_test。这个基类的核心任务是提供一个灵活的框架允许在main_phase中动态接收并启动指定的序列。class ubus_save_restore_base_test extends uvm_test; uvm_component_utils(ubus_save_restore_base_test) // 环境实例 ubus_example_tb ubus_tb; // 用于存储从命令行获取的序列名 string sequence_name; // 标志位指示是否处于“恢复后运行”模式 bit restored_mode 0; function new(string name ubus_save_restore_base_test, uvm_component parent null); super.new(name, parent); // 从命令行参数获取“SEQ序列名”的参数 if ($value$plusargs(SEQ%s, sequence_name)) begin uvm_info(get_type_name(), $sformatf(Will run sequence from command line: %s, sequence_name), UVM_LOW) restored_mode 1; // 如果指定了SEQ则认为是从恢复点启动 end endfunction virtual function void build_phase(uvm_phase phase); super.build_phase(phase); // 创建环境这部分是每个测试都需要的静态构造 ubus_tb ubus_example_tb::type_id::create(ubus_tb, this); endfunction virtual task main_phase(uvm_phase phase); // 关键将phase.raise_objection放在这里而不是在build_phase设置default_sequence phase.raise_objection(this); if (restored_mode) begin // 模式A恢复后模式运行动态指定的序列 uvm_info(get_type_name(), Running in RESTORED mode, starting dynamic sequence..., UVM_LOW) run_dynamic_sequence(); end else { // 模式B全新仿真模式执行完整的配置和默认序列 uvm_info(get_type_name(), Running in FRESH mode, performing full configuration..., UVM_LOW) // 这里可以模拟耗时的初始化例如等待复位完成、PLL锁定、DDR初始化等 #1000; // 举例模拟1us的初始化时间 perform_common_configuration(); // 可以在这里执行一个基础的、用于建立环境的默认序列或者什么都不做直接保存 uvm_info(get_type_name(), Common configuration done. Ready for save point., UVM_LOW) // 注意此时不主动启动任何测试序列为保存状态做准备 end phase.drop_objection(this); endtask // 公共配置任务模拟耗时操作 virtual task perform_common_configuration(); uvm_info(get_type_name(), Simulating PLL lock and DDR init..., UVM_LOW) // 在实际项目中这里可能是对虚拟序列virtual sequence的调用 // 该序列负责配置所有接口的寄存器。 // 例如通过uvm_config_db设置虚拟序列并在虚拟序列器中启动。 #500; uvm_info(get_type_name(), Common configuration completed., UVM_LOW) endtask // 动态运行序列的任务 virtual task run_dynamic_sequence(); uvm_sequence_base seq; uvm_sequencer_base seqr; // 1. 根据字符串sequence_name找到对应的序列类型并创建实例 case (sequence_name) read_modify_write_seq: seq read_modify_write_seq::type_id::create(seq); r8_w8_r4_w4_seq: seq r8_w8_r4_w4_seq::type_id::create(seq); // 可以添加更多序列... default: begin uvm_fatal(get_type_name(), $sformatf(Unknown sequence requested: %s, sequence_name)) end endcase // 2. 找到目标sequencer这里以ubus_master_sequencer为例 // 在实际环境中你可能需要定位一个虚拟sequencer或特定的物理sequencer if (!uvm_config_db#(ubus_master_sequencer)::get(this, , master_sequencer, seqr)) begin uvm_fatal(get_type_name(), Failed to get master_sequencer from config DB) end // 3. 启动序列 uvm_info(get_type_name(), $sformatf(Starting sequence %s on sequencer %s, sequence_name, seqr.get_full_name()), UVM_LOW) seq.start(seqr); endtask endclass关键修改解读移除了build_phase中的default_sequence设置传统方式uvm_config_db#(uvm_object_wrapper)::set(this, “*.master_sequencer.main_phase”, “default_sequence”, read_modify_write_seq::get_type())被移除。序列启动的控制权完全交给了main_phase。引入了restored_mode和命令行参数SEQ通过$value$plusargs检测运行时是否指定了要运行的序列名。如果指定了则进入restored_mode跳过公共配置直接运行指定序列。分离了perform_common_configuration任务将耗时的公共配置步骤抽象成一个独立任务只在全新仿真!restored_mode时调用。run_dynamic_sequence任务实现了根据字符串动态创建并启动序列的机制。这是恢复后执行不同测试的关键。3.2 改造第二步创建具体的测试用例原有的测试类现在变得非常精简它们只需要继承我们改造后的基类甚至可以不需要任何额外代码因为所有逻辑都在基类中。当然你也可以在子类中覆盖某些行为。// 原来的 test_read_modify_write 可以简化成几乎一个空壳 class test_read_modify_write extends ubus_save_restore_base_test; uvm_component_utils(test_read_modify_write) function new(string name test_read_modify_write, uvm_component parent null); super.new(name, parent); endfunction // 可以覆盖 perform_common_configuration 来定义本测试特定的公共配置如果需要 endclass // test_r8_w8_r4_w4 同理 class test_r8_w8_r4_w4 extends ubus_save_restore_base_test; uvm_component_utils(test_r8_w8_r4_w4) function new(string name test_r8_w8_r4_w4, uvm_component parent null); super.new(name, parent); endfunction endclass3.3 仿真流程与操作指南现在我们来看完整的、利用保存-恢复策略的仿真流程。我们以VCS仿真器为例其他仿真器如Xcelium的-input命令Questa的-restore原理类似具体命令请参考其手册。步骤一执行公共配置并创建保存点这是耗时最长但只做一次的步骤。# 编译阶段与普通仿真无异 vcs -sverilog -ntb_opts uvm-1.2 [其他编译选项] ... -o simv # 运行仿真使用一个测试比如test_read_modify_write但不指定SEQ参数。 # 这个运行的目的不是完成测试而是走到公共配置结束的点然后保存状态。 ./simv UVM_TESTNAMEtest_read_modify_write UVM_TIMEOUT1000000在仿真运行期间当我们的测试基类判断restored_mode0它会执行perform_common_configuration()。在这个任务完成后、main_phase结束前即phase.drop_objection之前是保存状态的理想时机。我们需要通过仿真器交互命令或系统任务来触发保存。方法A在Testbench中硬编码保存点不推荐缺乏灵活性task perform_common_configuration(); // ... 执行配置 ... uvm_info(get_type_name(), Common configuration completed. Saving checkpoint..., UVM_LOW) $save(common_configuration_checkpoint); // Verilog系统任务 // 或者使用仿真器特定的PLI任务 endtask方法B通过仿真器命令行交互推荐更灵活在仿真运行时在终端按下CtrlC中断仿真进入VCS的UCLIUnified Command Line Interface模式然后执行ucli% save -file common_configuration_checkpoint ucli% quit -f或者你可以提前编写一个Tcl脚本(save_at_time.tcl)after 1000ns // 在配置预计完成的时间点 save common_configuration_checkpoint quit -f然后运行仿真时加载它./simv ... -ucli -i save_at_time.tcl步骤二从保存点恢复并运行不同的测试保存点创建后后续的所有测试都可以从这个点开始节省了配置时间。# 恢复仿真并指定要运行的新序列 ./simv -restore common_configuration_checkpoint UVM_TESTNAMEtest_read_modify_write SEQr8_w8_r4_w4_seq此时仿真器直接从保存的状态开始运行。测试基类的构造函数会检测到SEQr8_w8_r4_w4_seq参数设置restored_mode1。在main_phase中它跳过公共配置直接调用run_dynamic_sequence()启动r8_w8_r4_w4_seq序列。尽管我们指定的UVM_TESTNAME还是test_read_modify_write但由于处于恢复模式实际运行的序列已经变成了我们命令行指定的那个。你可以用同样的方式用不同的SEQ参数反复从同一个保存点恢复快速运行大量测试用例。./simv -restore common_configuration_checkpoint UVM_TESTNAMEtest_read_modify_write SEQread_modify_write_seq ./simv -restore common_configuration_checkpoint UVM_TESTNAMEtest_read_modify_write SEQsome_other_seq4. 深入解析实现细节与关键注意事项4.1 保存/恢复的粒度与时机选择选择在哪个时间点保存状态是一门艺术需要仔细权衡。最佳时机通常选择在所有必要的、确定性的初始化完成之后在任何随机的、测试特异的激励开始之前。在我们的例子中就是在perform_common_configuration()完成后。这确保了恢复后的环境处于一个干净、一致的“起跑线”状态。需要避免的时机复位期间复位信号可能还未稳定状态不确定。正在执行动态序列的过程中序列可能已经消耗了部分transaction或driver/monitor内部有状态残留恢复后可能导致行为错乱。有未决的时钟或定时器如果保存时存在即将触发的定时事件恢复后这些事件的定时可能会受到影响。涉及外部模型或C代码如果验证环境中有不透明的C/C模型或DPI-C调用需要确认它们是否支持保存/恢复。有些模型可能需要特殊的“存根”函数来序列化其内部状态。注意保存的状态文件可能非常大因为它包含了整个仿真内存的镜像。对于大型设计文件可能达到GB甚至TB级别。确保你的磁盘有足够空间并且考虑使用仿真器提供的增量保存或压缩选项如果支持。4.2 UVM组件状态与回调的处理UVM环境在保存/恢复后是否能正常工作取决于其组件的状态是否被正确保存。大部分UVM内置类uvm_component,uvm_sequence_item,uvm_transaction如果只使用标准的UVM字段宏uvm_field_*注册其字段那么其状态通常能被仿真器自动捕获。但是你需要特别注意以下几点静态变量和全局变量Verilog中的static变量和SystemVerilog中的class静态变量static可能不会被自动保存/恢复。如果它们存储了关键状态你需要考虑其他管理方式或者避免在需要保存/恢复的上下文使用它们。文件描述符和外部连接打开的文件句柄、网络套接字等外部资源的状态无法通过保存/恢复来维持。通常需要在恢复后重新初始化这些连接。UVM Phase 状态机UVM运行时的phase状态例如当前正在执行哪个phase是UVM内核管理的。仿真器保存的是整个进程的内存状态因此phase状态也会被保存。恢复后UVM会从保存点所在的phase继续执行。这就是为什么我们的策略能正常工作我们在main_phase中保存恢复后依然在main_phase中继续。uvm_event、uvm_barrier等同步原语这些对象内部有等待的进程列表。保存/恢复后这些等待关系可能会变得复杂。建议在公共配置阶段避免使用复杂的、跨多个phase的同步机制或者确保在恢复后能重新建立正确的同步。4.3 随机数生成器RNG的状态管理这是保存-恢复策略中最容易踩坑的地方之一。UVM和SystemVerilog的随机化依赖于伪随机数生成器RNG的状态。如果保存的状态包含了RNG的当前种子seed和状态那么每次从同一个保存点恢复后生成的随机数序列将是完全相同的。这对于验证来说是不可接受的因为我们希望每次测试都有不同的随机激励。解决方案在恢复后主动重置或重播RNG种子。virtual task main_phase(uvm_phase phase); phase.raise_objection(this); if (restored_mode) begin // 关键步骤在恢复模式重新播种RNG确保随机性 int seed; seed $urandom; // 从系统获取一个新的随机种子 uvm_info(get_type_name(), $sformatf(Reseeding RNG with new seed: %0d, seed), UVM_LOW) this.srandom(seed); // 为当前组件及其子组件重新播种 // 如果需要也可以为特定的sequencer或sequence单独播种 // ubus_tb.master_sequencer.srandom(seed); uvm_info(get_type_name(), Running in RESTORED mode..., UVM_LOW) run_dynamic_sequence(); end else { // 全新仿真模式使用默认或命令行传入的种子 uvm_info(get_type_name(), Running in FRESH mode..., UVM_LOW) perform_common_configuration(); // 可以在这里保存状态 end phase.drop_objection(this); endtask通过调用component.srandom(seed)可以为该组件及其所有子组件层次结构下的RNG设置一个新的种子从而打破恢复后随机数序列的确定性保证每次恢复运行的测试激励都是随机的、不同的。5. 常见问题排查与效能提升技巧5.1 问题排查速查表问题现象可能原因排查步骤与解决方案恢复后仿真立即结束或挂起1.main_phase中没有raise_objection。2. 恢复后restored_mode判断错误执行了错误的代码分支。1. 检查main_phase确保在恢复模式下也有phase.raise_objection(this)。2. 打印restored_mode和sequence_name的值确认命令行参数SEQ是否正确传递并被解析。恢复后序列没有启动1. 命令行指定的序列名与case语句中的字符串不匹配。2. 未能通过uvm_config_db获取到目标sequencer。3. 序列的body()任务中有错误。1. 检查SEQ参数和case语句确保大小写一致。2. 在run_dynamic_sequence中增加调试信息打印获取到的sequencer句柄。3. 在序列body()任务开始处添加uvm_info确认其是否被调用。恢复后的仿真行为与全新仿真不一致1. RNG状态未重置导致随机激励重复。2. 某些组件如记分板、覆盖率收集器在保存前已有历史数据恢复后累积计算错误。3. 存在仿真器与保存/恢复相关的已知bug。1. 实施RNG重播种策略。2. 在公共配置完成后、保存状态前重置所有记分板、覆盖率模型等分析组件的历史状态。或者在恢复模式下的start_of_simulation_phase中重置它们。3. 查阅仿真器发行说明升级到已修复该bug的版本。保存的文件异常巨大设计规模大且保存了完整的波形数据或调试信息。1. 在保存前关闭波形记录如$dumpfile/$dumpvars。2. 使用仿真器的增量保存功能如VCS的-save选项组合。3. 考虑只保存必要的层次结构。恢复后出现X态或仿真错误1. 保存的时机不当设计或Testbench处于不稳定状态。2. 某些第三方IP或VIP不支持保存/恢复。1. 尝试在时钟稳定边沿、且无关键信号变化时保存状态。2. 联系IP/VIP供应商确认其兼容性。对于不兼容的模块可能需要将其排除在保存范围之外如果仿真器支持部分保存或者在恢复后重新初始化这些模块。5.2 效能提升与进阶技巧分层保存策略对于超大型设计一次性保存全部状态开销太大。可以考虑“分层保存”先保存一个最小化的、仅包含总线基础设施和存储控制器配置的“轻量级”检查点。然后针对不同子系统或功能模块的测试从这个轻量级检查点恢复后再快速加载该模块的特定配置形成“二级检查点”。这需要更精细的脚本控制。与回归测试框架集成将保存-恢复流程集成到你的Makefile或Python/Perl回归测试脚本中。脚本可以自动运行一次“黄金配置”仿真并创建保存点。并行启动多个恢复仿真作业每个作业分配不同的SEQ参数和随机种子。收集并汇总所有恢复仿真的日志和报告。 这能最大化利用多核服务器资源将回归测试时间从“线性增长”变为“常数时间线性增长仅计算不同测试本身”。状态验证在从保存点恢复后可以加入一个简短的“健康检查”序列。这个序列读取DUT和验证平台的一些关键状态寄存器或变量与预期值进行比较确保恢复的状态是正确的然后再开始正式的测试序列。这能及早发现因保存/恢复引入的隐蔽错误。利用仿真器的快照管理功能像VCS的-save/-restoreXcelium的-snapshot等通常有丰富的选项。例如可以指定保存的名称、压缩格式、是否包含波形信息等。深入阅读仿真器手册能帮你找到最优的配置组合。实施这套保存-恢复策略初期会有一点学习和改造的成本但一旦流程跑通对于拥有长配置周期的验证项目来说其带来的仿真效率提升是立竿见影的。它尤其适合在项目中期和后期当测试用例数量爆炸性增长时作为加速回归验证的利器。从我的经验来看对于一个配置阶段占单次仿真时间30%以上的项目采用此策略后整体验证周期缩短20%-40%是完全可以期待的。关键在于精确地找到那个公共配置结束的“甜蜜点”并妥善处理好恢复后的环境状态与随机性。

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

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

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…