IC验证——perl脚本ccode_standard——c代码寄存器配置标准化
目录
1 脚本名称
2 脚本路径
3 脚本参数说明
4 脚本操作说明
5 脚本代码
1 脚本名称
ccode_standard
2 脚本路径
/scripts/bin/ccode_standard
3 脚本参数说明
| 次序 | 参数名 | 说明 |
| 1 | address (./rfdig;.;..;./boot) | 指定脚本执行路径(可以为脚本所在路径的任意相对路径) |
| help | 脚本使用帮助,打印说明信息 | |
| 2 | all | 运行脚本后,脚本会解析全部rbus.cfg,并对参数1指定的目录中的.c文件进行寄存器配置代码标准化修改 |
| module_name_seca.cfg | 运行脚本后,脚本会解析参数2指定的rbus.cfg,并对参数1指定的目录中的.c文件进行寄存器配置代码标准化修改 | |
| 示例 | 命令:ccode_standard help 输出ccode_standard使用帮助信息 | |
| 命令:ccode_standard ./rfdig rfdig_seca.cfg 对当前路径下的rfdig文件夹及其子文件夹下所有的.c文件中,有关rfdig_seca.cfg中的寄存器配置代码进行标准化修改; | ||
| 命令:ccode_standard . crg_seca.cfg 对当前路径下的所有文件夹及其子文件夹下的所有.c文件中,所有有关crg_seca.cfg中的寄存器配置代码进行标准化修改; | ||
| 命令:ccode_standard . all 对当前路径下的所有文件夹及其子文件夹下的所有.c文件中,所有模块的寄存器配置代码进行标准化修改; | ||
| 说明 | 配置文件module_name_seca.cfg是被验模块寄存器的详细配置文件;脚本索引该文件的具体路径为: $PROJ_SRC/design/scripts/rbus_scripts/rbus_cfg/ 使用该脚本前请先gen_rbus; | |
4 脚本操作说明
该脚本用于将如下所示的代码标准化,加速ST用例调整,该脚本可以反复多次调用,不会产生异常问题:

- 使用脚本前,首先gen_rbus,source,产生脚本需要解析的rbus.cfg文件,以及环境变量$ENV{PROJ_SRC};

- 根据脚本存放路径,指定脚本工作目录,如下例所示,脚本存放于case目录,我想修改rfdig文件夹下的.c文件,参考rfdig_seca.cfg;执行:ccode_standard ./rfdig rfdig_seca.cfg > 1.log


-
如图所示,脚本只替换了有关rfdig的寄存器,iomux,crg等没有替换:
-

- 如下例所示,脚本存放于case目录,我想修改rfdig/rfdig_tc017文件夹下的.c文件,参考所有rbus.cfg文件;
-
执行:ccode_standard ./rfdig/rfdig_m33_tc017 all > 1.log
-

-
如图所示,脚本不仅替换了有关rfdig的寄存器,iomux,crg等都有替换:
-


5 脚本代码
#! /usr/bin/perl -w#==========================================================
# PERL MODULE
#==========================================================use Cwd;
use File::Path;
use File::Find;#==========================================================
# PERL FUNCTION GetOptions (get command parameters)
#==========================================================
$g_work_dir = $ARGV[0];
$g_seca_fl = $ARGV[1];#our $work_dir = 'rfdig_m33_tc017';#==========================================================
# SCRIPT MAIN
#==========================================================our @seca_fl_list;
our @seca_name;
our @seca_type;
our @seca_width;
our @seca_init_value;
our @seca_with_field;
our @seca_field_list_bgn;
our @seca_field_list_end;
our @seca_field_name;
our @seca_field_width;
our @seca_field_bgn_idx;
our @seca_field_end_idx;our $ccode_standard_mark = 0;Main();sub RegMem{@seca_name = ();@seca_type = ();@seca_width = ();@seca_init_value = ();@seca_with_field = ();@seca_field_list_bgn = ();@seca_field_list_end = ();@seca_field_name = ();@seca_field_width = ();@seca_field_bgn_idx = ();@seca_field_end_idx = ();if (open(SECA_ID,"$g_seca_fl") or die "cannot open $g_seca_fl, no such file") {@seca_array = <SECA_ID>;my $i = 0;my $j = 0;my $rbus_ok = 0;foreach $seca_line (@seca_array) {$seca_line =~ s/[\n\r]*//g;if ($seca_line =~ m/^\s*field_(\S+)\s+\[(\S+)\]\s*$/) {if($rbus_ok eq 1) {$seca_with_field[$i-1] = 1 ;$seca_field_list_bgn[$i-1] = $j;$rbus_ok = 0 ;}$seca_field_list_end[$i-1] = $j;my $field_name_tmp = "\L$1";my $field_name_size = push(@seca_field_name ,$field_name_tmp);my $field_port_tmp = $2;if ($field_port_tmp =~ m/(\d+):(\d+)/) {push(@seca_field_bgn_idx,int($2));push(@seca_field_end_idx,int($1));push(@seca_field_width,(int($1)-int($2)+1));} else {push(@seca_field_bgn_idx,int($field_port_tmp));push(@seca_field_end_idx,int($field_port_tmp));push(@seca_field_width,1);}#print ("seca_with_field[",$i-1,"] = ",$seca_with_field[$i-1] ,"\n");#print ("seca_field_list_bgn[",$i-1,"] = ",$seca_field_list_bgn[$i-1] ,"\n");#print ("seca_field_list_end[",$i-1,"] = ",$seca_field_list_end[$i-1] ,"\n");#print ("seca_field_name[",$j,"] = ",$seca_field_name[$j] ,"\n");#print ("seca_field_width[",$j,"] = ",$seca_field_width[$j] ,"\n");#print ("seca_field_bgn_idx[",$j,"] = ",$seca_field_bgn_idx[$j] ,"\n");#print ("seca_field_end_idx[",$j,"] = ",$seca_field_end_idx[$j] ,"\n");$j = $j + 1;}if ($seca_line =~ m/^\s*rbus_(\S+)\s+(\S+)\s+(\w+)\s+(\d+)('\w+)\s+(\S+)\s*$/) {#print ("before match string: ",$`, "\n");#print ("match string : ",$&, "\n");#print ("after match string: ",$', "\n");#print ("seca_postfix : ",$1, "\n");#print ("seca_type : ",$2, "\n");#print ("seca_scope : ",$3, "\n");#print ("seca_width : ",int($4),"\n");#print ("seca_init_value : ",$5, "\n");#print ("seca_prefix : ",$6, "\n");#print ("seca_name : ","$6_$1","\n");$seca_name[$i] = "\L$6_$1";$seca_type[$i] = $2;$seca_width[$i] = int($4);$seca_init_value[$i] = $5;$seca_with_field[$i] = 0;$i = $i+1;$rbus_ok = 1;}}}
}sub GenMacroDefine{#print "c file is : $_[0]\n";#print "c macro_define file is : $_[1]\n";$ccode_standard_mark = 0;open(ORIEN_FILE,">>$_[1]");if (open(SOURCE_FILE,"$_[0]")) {@context = <SOURCE_FILE>;foreach $line (@context) {for ( $i = 0; $i < @seca_name; $i = $i +1) {if ($seca_with_field[$i] eq 1) {if ($line =~ m/^\s*(\w+)\[($seca_name[$i])\]\s*=\s*0x(\w+)\s*;.*$/) {$ccode_standard_mark = 1;print ORIEN_FILE "\n"."//"."$line";my $seca_field_list_num = $seca_field_list_end[$i] - $seca_field_list_bgn[$i] + 1;for ($j = 0; $j < $seca_field_list_num; $j = $j+1) {my $seca_field_width_tmp = $seca_field_width[$seca_field_list_bgn[$i]+$j];my $seca_field_name_tmp = "$seca_name[$i]"."_"."$seca_field_name[$seca_field_list_bgn[$i]+$j]";my $seca_field_mask_tmp = "$seca_field_name_tmp"."_mask ((1<<$seca_field_width[$seca_field_list_bgn[$i]+$j])-1)";my $seca_field_shift_tmp = "$seca_field_name_tmp"."_shift $seca_field_bgn_idx[$seca_field_list_bgn[$i]+$j]";my $seca_field_bit_tmp = "$seca_field_name_tmp"."_bit (1<<$seca_field_bgn_idx[$seca_field_list_bgn[$i]+$j])";if($seca_field_width_tmp eq 1){print ORIEN_FILE "#define $seca_field_bit_tmp\n";} else {print ORIEN_FILE "#define $seca_field_mask_tmp\n";print ORIEN_FILE "#define $seca_field_shift_tmp\n";}}}}}}close SOURCE_FILE;}close ORIEN_FILE;system ("awk '!a[\$0]++' $_[1] > $_[1]_tmp \&\& mv -f $_[1]_tmp $_[1]");
}sub AddMacroDefine{#print "old c file is : $_[0]\n";#print "new c file is : $_[1]\n";#print "add file is : $_[2]\n";my $add_ready = 0;open(ORIEN_FILE,">$_[1]");if (open(SOURCE_FILE,"$_[0]")) {@context = <SOURCE_FILE>;foreach $line (@context) {if($line =~ m/^#include \"$_[2]\"$/) {} else {print ORIEN_FILE "$line";}if ($add_ready eq 0) {if ($line =~ m/^\s*\#include\s*\"\S+_define.c\"\s*$/) {print ORIEN_FILE "\#include "."\"$_[2]\"\n";$add_ready = 1;}}}close SOURCE_FILE;}close ORIEN_FILE;
}sub GenSubContent{#print "old c file is : $_[0]\n";#print "new c file is : $_[1]\n";open(ORIEN_FILE,">>$_[1]");if (open(SOURCE_FILE,"$_[0]")) {@context = <SOURCE_FILE>;foreach $line (@context) {my $line_match = 0;for ( $i = 0; $i < @seca_name; $i = $i + 1) {if ($seca_with_field[$i] eq 1) {if ($line =~ m/^(\s*)(\w+\[$seca_name[$i]\])\s*=\s*0x(\w+)\s*;.*$/) {my $reg_value = hex($3);if ($reg_value != 0) {$line_match = 1;print ORIEN_FILE "//"."$line";my $seca_field_list_num = $seca_field_list_end[$i] - $seca_field_list_bgn[$i] + 1;for ($j = 0; $j < $seca_field_list_num; $j = $j+1) {my $seca_field_name_tmp = "$seca_name[$i]"."_"."$seca_field_name[$seca_field_list_bgn[$i]+$j]";my $seca_field_mask_tmp = "$seca_field_name_tmp"."_mask";my $seca_field_value = (($reg_value) >> ($seca_field_bgn_idx[$seca_field_list_bgn[$i]+$j])) & ((1<<$seca_field_width[$seca_field_list_bgn[$i]+$j])-1);my $seca_field_width_tmp = $seca_field_width[$seca_field_list_bgn[$i]+$j];my $seca_field_shift_tmp = "$seca_field_name_tmp"."_shift";my $seca_field_bit_tmp = "$seca_field_name_tmp"."_bit";#print ("seca_field_mask_value = ",((1<<$seca_field_width[$seca_field_list_bgn[$i]+$j])-1),";");#print ("reg_value = ",$reg_value,";");#print ("seca_field_value = $seca_field_value\n");if($j eq 0){if ($j eq ($seca_field_list_num - 1)) {if($seca_field_width_tmp eq 1) {printf ORIEN_FILE ("%s%-40s ((%-50s * 0x%-8x) %-51s) ;\n","$1","$2 =","$seca_field_bit_tmp",$seca_field_value," ");} else {printf ORIEN_FILE ("%s%-40s ((%-50s & 0x%-8x) << %-51s) ;\n","$1","$2 =","$seca_field_mask_tmp",$seca_field_value,"$seca_field_shift_tmp");}} else {if($seca_field_width_tmp eq 1) {printf ORIEN_FILE ("%s%-40s ((%-50s * 0x%-8x) %-51s) |\n","$1","$2 =","$seca_field_bit_tmp",$seca_field_value," ");} else {printf ORIEN_FILE ("%s%-40s ((%-50s & 0x%-8x) << %-51s) |\n","$1","$2 =","$seca_field_mask_tmp",$seca_field_value,"$seca_field_shift_tmp");}}} elsif($j eq ($seca_field_list_num - 1)) { if($seca_field_width_tmp eq 1) {printf ORIEN_FILE ("%s%-40s ((%-50s * 0x%-8x) %-51s) ;\n","$1"," ","$seca_field_bit_tmp",$seca_field_value," ");} else {printf ORIEN_FILE ("%s%-40s ((%-50s & 0x%-8x) << %-51s) ;\n","$1"," ","$seca_field_mask_tmp",$seca_field_value,"$seca_field_shift_tmp");}} else { if($seca_field_width_tmp eq 1) {printf ORIEN_FILE ("%s%-40s ((%-50s * 0x%-8x) %-51s) |\n","$1"," ","$seca_field_bit_tmp",$seca_field_value," ");} else {printf ORIEN_FILE ("%s%-40s ((%-50s & 0x%-8x) << %-51s) |\n","$1"," ","$seca_field_mask_tmp",$seca_field_value,"$seca_field_shift_tmp");}}}}}}}if ($line_match eq 0) {print ORIEN_FILE "$line";}}close SOURCE_FILE;}close ORIEN_FILE;
}sub CopyFile{#print "source file name is $_[0]\n";#print "orien file name is $_[1]\n";open(ORIEN_FILE,">$_[1]");if (open(SOURCE_FILE,"$_[0]")) {@context = <SOURCE_FILE>;foreach $line (@context) {print ORIEN_FILE "$line";}close SOURCE_FILE;}close ORIEN_FILE;
}sub FindConfigFile{if (-f $File::Find::name) {if ($_ =~ m/^(\w+_sec[a-z]\.cfg)$/) {#print "find file is $1 \n";push(@seca_fl_list,$1); }}
}sub FindWanted{if (-f $File::Find::name) {if ($_ =~ m/(\S+)\.c$/) {print "process file: $File::Find::name \n";GenMacroDefine("$_","$1"."_macro_define.h");unlink "$1"."_macro_define.h";if($ccode_standard_mark eq 1) {unlink "$_".".tmp";GenSubContent("$_","$_".".tmp");#AddMacroDefine("$_".".tmp","$_","$1"."_macro_define.h");system ("mv -f $_.tmp $_");unlink "$_".".tmp";}}}
}sub Main{my $cur_dir = getcwd;my $proj_src = $ENV{PROJ_SRC};my $seca_dir = "$ENV{PROJ_SRC}/design/scripts/rbus_scripts/rbus_cfg/";if($g_seca_fl eq "all") {chdir($seca_dir) or die "$seca_dir was not found!!! please gen_rbus firstly.";find(\&FindConfigFile,"$seca_dir");} else {push(@seca_fl_list,$g_seca_fl);}#for ($i = 0; $i < @seca_fl_list; $i = $i+1) {# print "seca_fl_list[$i] = $seca_fl_list[$i] \n";#}for ($z = 0; $z < @seca_fl_list; $z = $z + 1) {print "read config_file : $seca_fl_list[$z]\n";$g_seca_fl = "$seca_dir"."$seca_fl_list[$z]";RegMem();chdir($cur_dir) or die "$cur_dir does not exist!!!";find(\&FindWanted,"$cur_dir/$g_work_dir");}
}
相关文章:
IC验证——perl脚本ccode_standard——c代码寄存器配置标准化
目录 1 脚本名称 2 脚本路径 3 脚本参数说明 4 脚本操作说明 5 脚本代码 1 脚本名称 ccode_standard 2 脚本路径 /scripts/bin/ccode_standard 3 脚本参数说明 次序 参数名 说明 1 address (./rfdig;.;..;./boot) 指定脚本执行路…...
Qt单个字符判断
1.相关说明 字符的Unicode编码、单个字符的判断 2.界面绘制 3.相关主要代码 #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this); }Widget::~Widget() {delete ui;…...
通过OpenIddict设计一个授权服务器02-创建asp.net项目
在这一部分中,我们将创建一个ASPNET核心项目,作为我们授权服务器的最低设置。我们将使用MVC来提供页面,并将身份验证添加到项目中,包括一个基本的登录表单。 创建一个空的asp.net core项目 正如前一篇文章中所说,授权…...
2.6、云负载均衡产品详述
一、定义 弹性负载均衡(Elastic Load Balance,简称ELB)可将来自公网的访问流量分发到后端云主机,可选多种负载均衡策略,并支持自动检测云主机健康状况,消除单点故障,保障应用系统的高可用。 二、产品架构 1&am…...
黑马程序员 Docker笔记
本篇学习笔记文档对应B站视频: 同学们,在前两天我们学习了Linux操作系统的常见命令以及如何在Linux上部署一个单体项目。大家想一想自己最大的感受是什么? 我相信,除了个别天赋异禀的同学以外,大多数同学都会有相同的…...
游戏素材永不缺,免费在线AI工具Scenario功能齐全,简单易用
Scenario是一个在线的AI驱动的工具,主要用于游戏艺术创作。它提供了一套全面的功能,旨在帮助游戏开发者创建与其独特风格和艺术方向相符的独特、高质量的游戏艺术。Scenario的突出特点之一是它的微调能力,允许用户根据独特的风格和艺术方向训…...
ChatGPT和文心一言哪个好用?
#ChatGPT 和文心一言哪个更好用?# 在当今信息爆炸的时代,人们对于文本生成和创作工具的需求越来越高。在这个背景下,ChatGPT和文心一言作为备受瞩目的工具,各自拥有独特的功能和用途。在本文中,我们将深入探讨这两个工…...
纯c++简易的迷宫小游戏
一个用c写的黑框框迷宫 适合新手入门学习 也适合大学生小作业 下面附上代码 总体思路 初始化游戏界面:设置迷宫的大小(WIDTH和HEIGH),生成迷宫地图(map),包括墙壁、空地、起点和终点。显示…...
基于python舆情分析可视化系统+情感分析+爬虫+机器学习(源码)✅
大数据毕业设计:Python招聘数据采集分析可视化系统✅ 毕业设计:2023-2024年计算机专业毕业设计选题汇总(建议收藏) 毕业设计:2023-2024年最新最全计算机专业毕设选题推荐汇总 🍅感兴趣的可以先收藏起来&…...
2024年1月16日Arxiv热门NLP大模型论文:Multi-Candidate Speculative Decoding
大幅提速NLP任务,无需牺牲准确性!南京大学提出新算法,大幅提升AI文本生成效率飞跃 引言:探索大型语言模型的高效文本生成 在自然语言处理(NLP)的领域中,大型语言模型(LLMs…...
AI对决:ChatGPT与文心一言的比较
文章目录 引言ChatGPT与文心一言的比较Chatgpt的看法文心一言的看法Copilot的观点chatgpt4.0的回答 模型的自我评价自我评价 ChatGPT的优势在这里插入图片描述 文心一言的优势AI技术发展趋势总结 引言 在过去的几年里,人工智能(AI)技术取得了…...
uni-app引用矢量库图标
矢量库引用 导入黑色图标 1.生成连接,下载样式 2.导入项目(字体样式) 3.引入css样式 4.替换font-face 5.使用图标(字体图标,只有黑色) 导入彩色图标 1.安装插件 npm install -g iconfont-tools2.…...
Android的setContentView流程
一.Activity里面的mWindow是啥 在ActivityThread的performLaunchActivity方法里面: private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {ActivityInfo aInfo r.activityInfo;if (r.packageInfo null) {r.packageInfo getP…...
【加速排坑】docker设置国内image镜像源
第零步,查看阿里最新的镜像源:https://cr.console.aliyun.com/cn-hangzhou/instances/mirrors 第一步:在/etc/docker/daemon.json中添加镜像源 sudo mkdir -p /etc/docker sudo tee /etc/docker/daemon.json <<-EOF {"registry-m…...
el-table嵌套两层el-dropdown-menu导致样式错乱
问题: 解决方式: <el-table-column label"操作" fixed"right" width"132" align"center"><template slot-scope"scope"><div v-if"scope.row._index ! 合计"><el-d…...
自动化测试:fixture学得好,Pytest测试框架用到老
在pytest中,fixture是一种非常有用的特性,它允许我们在测试函数中注入数据或状态,以便进行测试。而参数化则是fixture的一个特性,它允许我们将不同的数据传递给fixture,从而进行多次测试。 本文将介绍如何在pytest中使…...
Linux上常用网络操作
主机名配置 hostname 查看主机名 hostname xxx 修改主机名 重启后无效 如果想要永久生效,可以修改/etc/sysconfig/network文件 IP地址配置 ifconfig 查看(修改)ip地址(重启后无效) ifconfig eth0 192.168.12.22 修改ip地址 如果想要永久生效,修改 /etc/sysco…...
POI:对Excel的基本读操作 整理2
1 简单读取操作 public class ExcelRead {String PATH "D:\\Idea-projects\\POI\\POI_projects";// 读取的一系列方法// ...... } 因为07版本和03版本操作流程大差不差,所以这边就以03版本为例 Testpublic void testRead03() throws IOException {//获取…...
LeetCode每周五题_2024/01/15~01/19
文章目录 82. 删除排序链表中的重复元素 II题目题解 2744. 最大字符串配对数目题目题解 82. 删除排序链表中的重复元素 II 82. 删除排序链表中的重复元素 II 题目 给定一个已排序的链表的头 head , 删除原始链表中所有重复数字的节点,只留下不同的数字…...
免费chartGPT网站汇总
https://s.suolj.com - (支持文心、科大讯飞、智谱等国内大语言模型,Midjourney绘画、语音对讲、聊天插件)国内可以直连,响应速度很快 很稳定 https://seboai.github.io - 国内可以直连,响应速度很快 很稳定 http://gp…...
Rolify 项目部署指南:从开发环境到生产环境的完整迁移流程
Rolify 项目部署指南:从开发环境到生产环境的完整迁移流程 【免费下载链接】rolify Role management library with resource scoping 项目地址: https://gitcode.com/gh_mirrors/ro/rolify Rolify 是一款功能强大的角色管理库,支持资源范围的权限…...
Bing Wallpaper自动化部署:GitHub Actions与持续集成
Bing Wallpaper自动化部署:GitHub Actions与持续集成 【免费下载链接】bing-wallpaper 项目地址: https://gitcode.com/gh_mirrors/bi/bing-wallpaper Bing Wallpaper项目是一个专注于收集和展示Bing每日壁纸的开源项目,通过自动化部署可以确保壁…...
2025版等离子体期刊分区解析:从PRL到PPAP的投稿指南
1. 2025版等离子体期刊分区概览 对于从事等离子体研究的科研人员来说,选择合适的期刊投稿是研究成果传播的关键一步。2025版中科院期刊分区将等离子体相关期刊划分为三个主要层级,每个层级都有其独特的定位和特点。 先说说最顶级的中科院一区期刊。这个层…...
从ResNet到mHC:DeepSeek重构残差连接,额外开销仅6.7%,附复现代码
2015年,由微软亚洲研究院的何恺明团队提出ResNet,ResNet引入残差连接的概念,用以解决深层神经网络训练中的梯度消失/爆炸和网络退化问题,使得训练极深的网络成为可能。 ��1��&#x…...
Janus-Pro-7B模型部署精讲:VMware虚拟机中的隔离环境搭建
Janus-Pro-7B模型部署精讲:VMware虚拟机中的隔离环境搭建 想在自己的电脑上测试Janus-Pro-7B这类大模型,但又担心搞乱本地环境,或者影响其他工作?用虚拟机搭建一个隔离的测试环境,是个非常稳妥的选择。它就像在你的电…...
DLSS Swapper完整指南:高效管理游戏DLSS、FSR与XeSS版本
DLSS Swapper完整指南:高效管理游戏DLSS、FSR与XeSS版本 【免费下载链接】dlss-swapper 项目地址: https://gitcode.com/GitHub_Trending/dl/dlss-swapper DLSS Swapper是一款专业的游戏性能优化工具,专门用于管理NVIDIA DLSS、AMD FSR和Intel X…...
手机端能用嘎嘎降AI吗:移动端使用完整指南和注意事项
手机端能用嘎嘎降AI吗:移动端使用完整指南和注意事项 上周室友第一次用降AI工具,操作错了好几步,差点浪费机会。觉得有必要写一篇详细教程。 我用的是嘎嘎降AI(www.aigcleaner.com),4.8元一篇,…...
别再到处找教程了!Ubuntu 18.04 + Carla 0.9.13 + ROS Melodic 联合仿真环境保姆级搭建实录
Ubuntu 18.04 Carla 0.9.13 ROS Melodic 联合仿真环境实战指南 自动驾驶仿真环境的搭建往往充满挑战,特别是当多个复杂系统需要协同工作时。本文将带你一步步完成Ubuntu 18.04系统下Carla 0.9.13与ROS Melodic的联合仿真环境搭建,避开那些令人头疼的&…...
QMCDecode:解放加密音乐的格式转换专家指南
QMCDecode:解放加密音乐的格式转换专家指南 【免费下载链接】QMCDecode QQ音乐QMC格式转换为普通格式(qmcflac转flac,qmc0,qmc3转mp3, mflac,mflac0等转flac),仅支持macOS,可自动识别到QQ音乐下载目录,默认转换结果存储…...
模型微调加持:百川2-13B+OpenClaw定制化个人助手实践
模型微调加持:百川2-13BOpenClaw定制化个人助手实践 1. 为什么需要定制化个人助手? 去年我尝试用现成大模型搭建自动化助手时,发现一个尴尬现象:当我让AI帮我整理会议纪要时,它总把技术术语解释得像科普读物…...
