unity变体收集工具

项目目录:D:\GIT\YooAsset\Temp

收集代码:

using System.Collections;

using System.Collections.Generic;

using UnityEditor;

#if UNITY_EDITOR

using UnityEditor.Build;

using UnityEditor.Rendering;

#if UNITY_2019_4_OR_NEWER

using UnityEditor.Build.Pipeline.Utilities;

#endif

#endif

using UnityEngine;

using System.Linq;

using System.Text;

using ShaderVariant = UnityEngine.ShaderVariantCollection.ShaderVariant;

// 增量打包时已经打包的Shader不会触发该回调,只有新打包的Shader才会。不管是Addressable打包或者是其他的打包方式都类似。

// 如果需要通过IPreprocessShaders来进行Shader变体剔除,那么还是建议重新全量打包一次,或者打包时设置ForceRebuild。

// 实现 IPreprocessShaders 接口的类有VersionedCallbackAttribute 属性, 修改version更新sbp的hash,跳过打包缓存。

namespace Soco.ShaderVariantsStripper

{

#if UNITY_EDITOR

#if UNITY_2019_4_OR_NEWER

    [VersionedCallback(1.47f)]

#endif

    public class ShaderVariantsStripperCode : IPreprocessShaders

    {

        public static StringBuilder sbCache = new StringBuilder();

        public static StringBuilder sb = new StringBuilder();

        public int callbackOrder { get { return 0; } }

        private static string[] sAllPath = { "Assets", "Packages" };

        private static ShaderVariantsStripperConfig[] sConfigs;

        private List<(ConditionPair conditionPair, ShaderVariantsStripperConfig config)> mConditionList = new List<(ConditionPair condition, ShaderVariantsStripperConfig config)>();

        public static ShaderVariantsStripperConfig[] LoadConfigs()

        {

            string[] guids = AssetDatabase.FindAssets("t:ShaderVariantsStripperConfig", sAllPath);

            sConfigs = (from guid in guids

                    select AssetDatabase.LoadAssetAtPath<ShaderVariantsStripperConfig>(

                        AssetDatabase.GUIDToAssetPath(guid)))

                .ToArray();

            return sConfigs;

        }

        public static void ClearConfigs()

        {

            sConfigs = null;

        }

        public void OnProcessShader(Shader shader, ShaderSnippetData snippet, IList<ShaderCompilerData> data)

        {

            sbCache.AppendLine(shader.name);

            sbCache.AppendLine(data.Count.ToString());

            sbCache.AppendLine(snippet.passType.ToString());

           

            // print to file

            sb.AppendLine("");

            sb.AppendLine($"name:{shader.name}, passType:{snippet.passType}, shaderType:{snippet.shaderType}");

            foreach(var v in data) {

                List<string> keyList = new List<string>();

                string keys = "";

                foreach(var key in v.shaderKeywordSet.GetShaderKeywords()) {

                    keys += key.name + ",";

                    keyList.Add(key.name);

                }

                sb.AppendLine($"     变体 Shader:{shader.name} keys:{keys} ");

                var str_ = "";

                foreach(var str in keyList) {

                    str_ += str + " ";

                }

                sbCache.AppendLine(str_);

            }

            sbCache.AppendLine("");



 

            if (sConfigs == null) {

                LoadConfigs();

            }

            ShaderCompilerData workaround = data[0];

            int stripCount = 0;

            for (int i = data.Count - 1; i >= 0 ; --i) {

                mConditionList.Clear();

                StripVariant(shader, snippet, data[i], sConfigs, mConditionList);

                foreach (var conditionPair_fromConfig in mConditionList) {

                    if (conditionPair_fromConfig.conditionPair.strip)

                    {

                        data.RemoveAt(i);

                        stripCount++;

                        break;

                    }

                }

            }

            // Debug.Log($"Shader:{shader.name} Pass:{snippet.passType}-PassName:{snippet.passName} 剔除个数:{stripCount}");

            if (data.Count == 0)

            {

                // Debug.Log($"Shader:{shader.name} Pass:{snippet.passType} 因剔除全部保留变体一个");

                data.Add(workaround);

            } else {

                foreach(var v in data)

                {

                    string keys = "";

                    foreach(var key in v.shaderKeywordSet.GetShaderKeywords())

                    {

                        keys += key.name + ",";

                    }

                    // Debug.Log($"     变体 Shader:{shader.name} keys:{keys} ");

                }

            }

        }

       

        //对外开放接口,用于检查keyword是否需要被剔除

        private static List<(ConditionPair conditionPair, ShaderVariantsStripperConfig config)> sConditionList = new List<(ConditionPair condition, ShaderVariantsStripperConfig config)>();

        public static bool IsVariantStrip(Shader shader, ShaderSnippetData snippet, ShaderCompilerData data, ShaderVariantsStripperConfig[] configs)

        {

            sConditionList.Clear();

            StripVariant(shader, snippet, data, sConfigs, sConditionList);

            return sConditionList.Any(conditionPair_fromConfig =>

                conditionPair_fromConfig.conditionPair.strip);

        }

       

        public static void StripVariant(Shader shader, ShaderSnippetData snippet, ShaderCompilerData data,

            ShaderVariantsStripperConfig[] configs,

            List<(ConditionPair conditionPair, ShaderVariantsStripperConfig config)> conditionList)

        {

            StripVariant(shader, ShaderVariantsData.GetShaderVariantsData(snippet, data), configs, conditionList);

        }

       

        public static void StripVariant(Shader shader, ShaderVariantsData variantData, ShaderVariantsStripperConfig[] configs, List<(ConditionPair conditionPair, ShaderVariantsStripperConfig config)> conditionList)

        {

            int FindConditionEqual(ConditionPair pair, out int index)

            {

                for (int condList_i = 0; condList_i < conditionList.Count; ++condList_i)

                {

                    if (pair.condition.EqualTo(conditionList[condList_i].conditionPair.condition, variantData))

                    {

                        index = condList_i;

                        return condList_i;

                    }

                }

                index = -1;

                return -1;

            }

               

            foreach (ShaderVariantsStripperConfig config in configs)

            {

                if (!config.mEnable)

                    continue;

               

                bool applyGlobalConfig = true;

                // 如果这个配置文件中能找到当前shader,则应用配置文件中“应用global config选项”

                if (config.mShaderConditions.TryGetValue(shader, out ShaderVariantsItem item))

                    applyGlobalConfig = item.applyGlobalConfig;

                // 如果Shader View中没有Shader,则Global Setting应用于全体Shader

                else if (config.mShaderConditions.Count == 0)

                    applyGlobalConfig = true;

                else

                    applyGlobalConfig = false;

                   

                //Global condition

                if (applyGlobalConfig)

                {

                    foreach (ConditionPair pair in config.mGlobalConditions)

                    {

                        if (pair.condition != null && pair.condition.Completion(shader, variantData))

                        {

                            //如果有相同的条件,

                            if (FindConditionEqual(pair, out int findIndex) != -1)

                            {

                                //且优先级更高

                                if(pair.priority > conditionList[findIndex].conditionPair.priority)

                                    conditionList[findIndex] = (pair, config);

                                //优先级更低则直接丢弃

                            }

                            else//否则加入列表

                                conditionList.Add((pair, config));

                        }

                    }

                }

                //Shader local condition

                if (item != null)

                {

                    foreach (ConditionPair pair in item.conditionPairs)

                    {

                        if (pair.condition.Completion(shader, variantData))

                        {

                            if (FindConditionEqual(pair, out int findIndex) != -1)

                            {

                                if (pair.priority > conditionList[findIndex].conditionPair.priority)

                                    conditionList[findIndex] = (pair, config);

                            }

                            else

                                conditionList.Add((pair, config));

                        }

                    }

                }

            }

        }

    }

#endif

}


 

保存代码:生成一个svc 和一个txt

using System;

using System.Collections;

using System.Collections.Generic;

using System.Reflection;

using System.Diagnostics;

using UnityEngine;

using ShaderVariant = UnityEngine.ShaderVariantCollection.ShaderVariant;

using Rendering = UnityEngine.Rendering;

using System.IO;

namespace YooAsset.Editor

{

    public class BuildRunner

    {

        private static Stopwatch _buildWatch;

        /// <summary>

        /// 总耗时

        /// </summary>

        public static int TotalSeconds = 0;

   

        /// <summary>

        /// 执行构建流程

        /// </summary>

        /// <returns>如果成功返回TRUE,否则返回FALSE</returns>

        public static BuildResult Run(List<IBuildTask> pipeline, BuildContext context)

        {

            if (pipeline == null)

                throw new ArgumentNullException("pipeline");

            if (context == null)

                throw new ArgumentNullException("context");

            BuildResult buildResult = new BuildResult();

            buildResult.Success = true;

            TotalSeconds = 0;

            for (int i = 0; i < pipeline.Count; i++)

            {

                IBuildTask task = pipeline[i];

                try

                {

                    _buildWatch = Stopwatch.StartNew();

                    var taskAttribute = task.GetType().GetCustomAttribute<TaskAttribute>();

                    if (taskAttribute != null)

                        BuildLogger.Log($"---------------------------------------->{taskAttribute.TaskDesc}<---------------------------------------");

                    task.Run(context);

                    _buildWatch.Stop();

                    // 统计耗时

                    int seconds = GetBuildSeconds();

                    TotalSeconds += seconds;

                    if (taskAttribute != null)

                        BuildLogger.Log($"{taskAttribute.TaskDesc}耗时:{seconds}秒");

                }

                catch (Exception e)

                {

                    EditorTools.ClearProgressBar();

                    buildResult.FailedTask = task.GetType().Name;

                    buildResult.ErrorInfo = e.ToString();

                    buildResult.Success = false;

                    break;

                }

            }

            // 返回运行结果

            BuildLogger.Log($"构建过程总计耗时:{TotalSeconds}秒");

            Print();

            return buildResult;

        }

        private static void Print()

        {

            System.Text.StringBuilder sb = Soco.ShaderVariantsStripper.ShaderVariantsStripperCode.sb;  

            SVC.FileHelper.WriteToFile(sb, "Temp/OnProcessShader.txt");

           

            System.Text.StringBuilder sbCache = Soco.ShaderVariantsStripper.ShaderVariantsStripperCode.sbCache;  

            SVC.FileHelper.WriteToFile(sbCache, "Temp/sbCache.txt", false);

            ShaderVariantCollection svc = new ShaderVariantCollection();

            using (System.IO.StringReader reader = new System.IO.StringReader(sbCache.ToString()))

            {

                string line;

                while ((line = reader.ReadLine()) != null)

                {

                    UnityEngine.Debug.Log(line);

                    string shaderName = line.Trim();

                    Shader mshader = Shader.Find(shaderName);

                    if(mshader == null)

                    {

                        UnityEngine.Debug.LogError("没有找到shader: " + shaderName);

                    } else {

                    }

                    line = reader.ReadLine();

                    int count = int.Parse(line.Trim());

                    line = reader.ReadLine();

                    string mpassType = line.Trim();  

                    Rendering.PassType mpass = (Rendering.PassType) Enum.Parse(typeof(Rendering.PassType), mpassType);

                    for(int i = 0, iMax = count; i < iMax; i++)

                    {

                        line = reader.ReadLine();

                        string keysLine = line.Trim();

                        string[] keyNames = keysLine.Split(' ');

                        ShaderVariant _sv = new ShaderVariant(mshader, mpass, keyNames);

                        svc.Add(_sv);

                    }

                       

                    line = reader.ReadLine(); // 跳过分割行

                }

                    var tempPath = "Assets/TestSVC/Tool2/UnityBuildResultSVC.shadervariants";

                if ( File.Exists( tempPath ) ) {

                    UnityEditor.AssetDatabase.DeleteAsset( tempPath );

                }  

                UnityEngine.Debug.LogError("--------- Save" + svc);

                UnityEditor.AssetDatabase.CreateAsset(svc, tempPath);

                UnityEditor.AssetDatabase.SaveAssets();

                UnityEditor.AssetDatabase.Refresh();

            }

        }

        private static int GetBuildSeconds()

        {

            float seconds = _buildWatch.ElapsedMilliseconds / 1000f;

            return (int)seconds;

        }

    }

}

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

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

相关文章

学习笔记|串口通信实战|简易串口控制器|sprintf函数|STC32G单片机视频开发教程(冲哥)|第二十一集(下):串口与PC通信

目录 3.串口通信实战实操简易的工作原理Tips:sprintf函数简介 总结课后练习 3.串口通信实战 做一个简易串口控制器。发送对应指令&#xff0c;让板子做相应的事情&#xff0c;或者传输数据&#xff08;文本模式下发送&#xff0c;不要选择HEX&#xff09;。 1.串口发送字符Ax\…

2020年亚太杯APMCM数学建模大赛B题美国总统的经济影响分析求解全过程文档及程序

2020年亚太杯APMCM数学建模大赛 B题 美国总统的经济影响分析 原题再现&#xff1a; 美国总统选举每四年举行一次。 2020年是美国总统大选年&#xff0c;共和党候选人唐纳德特朗普和民主党对手乔拜登竞选总统。 甲乙双方候选人在金融贸易&#xff0c;经济金融治理&#xff0c;…

云计算系统与传统计算系统的比较

随着技术的不断发展&#xff0c;云计算系统逐渐成为了企业和个人使用的主要计算方式之一。然而&#xff0c;很多人对云计算系统与传统计算系统之间的区别和相似之处还存在一些疑惑。本文将以云计算系统和传统计算系统为方向&#xff0c;探讨它们之间的异同点。 首先&#xff0…

python爬虫实战-京东商品数据

前言 大家早好、午好、晚好吖 ❤ ~欢迎光临本文章 今天介绍一下如何用 Python 来批量获取京东商品信息&#xff01;&#xff01; 如果有什么疑惑/资料需要的可以点击文章末尾名片领取源码 第三方库: requests >>> pip install requests 开发环境: python 3.8 py…

【iOS】——用单例类封装网络请求

文章目录 一、JSONModel1.JSONModel的简单介绍2.JSONModel的使用 二、单例类和Block传值 一、JSONModel 1.JSONModel的简单介绍 JSONModel一个第三方库&#xff0c;这个库用来将网络请求到的JSON格式的数据转化成Foundation框架下的Model类的属性&#xff0c;这样我们就可以直…

mysql中date/datetime类型自动转go的时间类型time.Time

在DSN中需要加入parseTimetrue&&locLocal&#xff0c;或 charsetutf8mb4&locAsia%2FShanghai&parseTimetrue。 package main_testimport ("database/sql""fmt""testing""time"_ "github.com/go-sql-driver/mysq…

Elasticsearch基础篇(五):创建es索引并学习分析器、过滤器、分词器的作用和配置

创建es索引并学习分析器、过滤器、分词器的作用和配置 一、基础概念Elasticsearch与MySQL的类比1. ES与MySQL的结构类比图2. ES与MySQL的类比示意表格3. 索引中重要概念索引&#xff08;Index&#xff09;文档&#xff08;Document&#xff09;字段&#xff08;Field&#xff0…

2023.10.18 区别 对象 和 类对象

目录 对象 类对象 总结 对象 对象是类的实例化结果它是内存中的一块区域&#xff0c;包含了该类的属性和方法的具体值和实现对象具有唯一的标识、状态、行为通过创建类的实例&#xff0c;我们可以在程序中操作和处理具体的对象 简单实例 class Person {public int high 1…

车载开发学习——CAN总线

CAN总线又称为汽车总线&#xff0c;全程为“控制器局域网&#xff08;Controller Area Network&#xff09;”&#xff0c;即区域网络控制器&#xff0c;它将区域内的单一控制单元以某种形式连接在一起&#xff0c;形成一个系统。在这个系统内&#xff0c;大家以一种大家都认可…

Netty系列教程之NIO基础知识

近30集的孙哥视频课程&#xff0c;看完一集整理一集来的&#xff0c;内容有点多&#xff0c;请大家放心食用~ 1. 网络通讯的演变 1.1 多线程版网络通讯 在传统的开发模式中&#xff0c;客户端发起一个 HTTP 请求的过程就是建立一个 socket 通信的过程&#xff0c;服务端在建立…

VR全景图片如何拍摄制作,拍摄制作过程中要注意什么?

引言&#xff1a; VR全景图片就是通过专业的相机设备捕捉到的一个空间的高清图像&#xff0c;再经过专业工具进行拼合&#xff0c;呈现出一种环绕式的视觉效果。想象一下&#xff0c;当你站在一个完全真实的环境中&#xff0c;可以自由地转动视角&#xff0c;看到四周的景色&a…

高数定理集合啦

haha~ 最近在准备数学竞赛&#xff0c;好久没有发布笔记啦&#xff0c;今天就来一波高数里常用的定理吧&#xff0c;不全面的话后续会更新哒~ 费马定理&#xff1a;对于一个函数的局部极值点&#xff0c;如果导数存在&#xff0c;那么导数在该点处必须为零&#xff0c;即 f(x)0…

SQL数据库管理工具RazorSQL mac中文版特点与功能

RazorSQL mac是一款功能强大的SQL数据库管理工具&#xff0c;它支持多种数据库&#xff0c;包括MySQL、Oracle、Microsoft SQL Server、SQLite、PostgreSQL等。 RazorSQL mac 软件特点和功能 多种数据库支持&#xff1a;RazorSQL支持多种数据库&#xff0c;用户可以通过一个工…

基于R语言的Meta分析【全流程、不确定性分析】方法与Meta机器学习高级应用

查看原文>>>【案例教程】基于R语言的Meta分析【全流程、不确定性分析】方法与Meta机器学习高级应用 Meta分析是针对某一科研问题&#xff0c;根据明确的搜索策略、选择筛选文献标准、采用严格的评价方法&#xff0c;对来源不同的研究成果进行收集、合并及定量统计分析…

安装Elasticsearch步骤(包含遇到的问题及解决方案)

注&#xff1a;笔者是在centos云服务器环境下安装的Elasticsearch 目录 1.安装前准备 2.下载Elasticsearch 3.启动Elasticsearch 非常容易出问题 第一次运行时&#xff0c;可能出现如下错误&#xff1a; 一、内存不足原因启动失败 二、使用root用户启动问题 三、启动ES自…

uniapp使用uQRCode绘制二维码,下载到本地,调起微信扫一扫二维码核销

1.效果 2.在utils文件夹下创建uqrcode.js // uqrcode.js //--------------------------------------------------------------------- // github https://github.com/Sansnn/uQRCode //---------------------------------------------------------------------let uQRCode {…

第二证券:A股三季报披露全面启动 多领域上市公司业绩表现亮点纷呈

A股上市公司三季报宣告全面发动。Wind数据闪现&#xff0c;到10月17日记者发稿&#xff0c;来自沪深北三大交易所近80家上市公司首要晒出了最新运营效果体现的“效果单”。本周&#xff0c;相关财报宣告家数也将增至270家左右。与此同时&#xff0c;10月以来&#xff0c;亦有不…

怎么把图片改成jpg格式?

怎么把图片改成jpg格式&#xff1f;大家都知道&#xff0c;随着计算机被发明到现在已经存在了很多年&#xff0c;在这么多的的技术发展过程中&#xff0c;也形成了种类非常多的图片文件格式&#xff0c;例如平时我们能接触到的图片格式有jpg、png、gif、bmp、heic、tiff、jfif、…

vcpkg manifest 的使用

最近项目上要使用 CMakeLists 管理&#xff0c;由于 Windows 版本有依赖到 vcpkg 提供的库&#xff0c;所以需要使用 vcpkg manifest 来统一设置库的版本&#xff0c;方便后续维护 推荐一个文章&#xff0c;介绍的可以说是非常全面了 VCPKG 特性 - Versioning 不过里面也有一些…

C++标准库算法整理

目录 1、数值操作 1.1、std::accumulate 1.2、std::inner_product 1.3、std::partial_sum 1.4、std::exclusive_scan 1.5、std::inclusive_scan 1.6、std::reduce 2、相邻元素 2.1、std::adjacent_difference 2.2、std::adjacent_find 2.3、std::unique 2.4、std::u…