http://blogs.msdn.com/b/chuckw/archive/2012/05/07/hlsl-fxc-and-d3dcompile.aspx
上面的链接指出“注意:此自动集成仅适用于 C++ 项目,不适用于 C# 项目。”。我正在使用 SlimDX 并且肯定有一种方法可以使 Visual Studio 在 C# 项目的构建时编译 HLSL 着色器?
http://blogs.msdn.com/b/chuckw/archive/2012/05/07/hlsl-fxc-and-d3dcompile.aspx
上面的链接指出“注意:此自动集成仅适用于 C++ 项目,不适用于 C# 项目。”。我正在使用 SlimDX 并且肯定有一种方法可以使 Visual Studio 在 C# 项目的构建时编译 HLSL 着色器?
我想你会发现这可能对你有用。
您必须在 bin 文件夹中有 fxc.exe directx 编译器,它可以在 direct x sdk 中找到。
此类提供可在 WPF 中使用的可观察集合,以显示着色器配置文件列表。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.IO;
using System.Diagnostics;
using System.Collections.ObjectModel;
using System.Threading;
using System.Reflection;
namespace Mars.HLSLEditor
{
public class ShaderModel {
public string Key {get; set;}
public string Name { get; set; }
}
public class EffectCompiler
{
private static Dictionary<string, string> profiles = new Dictionary<string,string>();
public static ObservableCollection<shadermodel> Profiles
{
get {
var x = from p in profiles select new ShaderModel { Key = p.Key, Name = p.Value};
return new ObservableCollection<shadermodel>(x);
}
}
public static ShaderModel ProfileByKey(string key) {
return new ShaderModel { Key = key, Name = profiles[key] };
}
public static int GetProfileIndex(ShaderModel model) {
return Array.IndexOf(profiles.Keys.ToArray(), model.Key);
}
static EffectCompiler() {
profiles.Add("cs_4_0", "Compute Shader model 4.0");
profiles.Add("cs_4_1", "Compute Shader model 4.1");
profiles.Add("cs_5_0", "Compute Shader model 5.0");
profiles.Add("ds_5_0", "Domain Shader model 5.0");
profiles.Add("fx_2_0", "Effect model 2.0");
profiles.Add("fx_4_0", "Effect model 4.0");
profiles.Add("fx_4_1", "Effect model 4.1");
profiles.Add("fx_5_0", "Effect model 5.0");
profiles.Add("gs_4_0", "Geometry Shader model 4.0");
profiles.Add("gs_4_1", "Geometry Shader model 4.1");
profiles.Add("gs_5_0", "Geometry Shader model 5.0");
profiles.Add("hs_5_0", "Hull Shader model 5.0");
profiles.Add("ps_2_0", "Pixel Shader model 2.0");
profiles.Add("ps_2_a", "Pixel Shader model 2.0 A");
profiles.Add("ps_2_b", "Pixel Shader model 2.0 B");
profiles.Add("ps_2_sw", "Pixel Shader model 2.0 (software)");
profiles.Add("ps_3_0", "Pixel Shader model 3.0");
profiles.Add("ps_3_sw", "Pixel Shader model 3.0 (software)");
profiles.Add("ps_4_0", "Pixel Shader model 4.0");
profiles.Add("ps_4_0_level_9_0", "Pixel Shader model 4.0 (level 9.0)");
profiles.Add("ps_4_0_level_9_1", "Pixel Shader model 4.0 (level 9.1)");
profiles.Add("ps_4_0_level_9_3", "Pixel Shader model 4.0 (level 9.3)");
profiles.Add("ps_4_1", "Pixel Shader model 4.1");
profiles.Add("ps_5_0", "Pixel Shader model 5.0");
profiles.Add("tx_1_0", "Texture Shader model 1.0 (software)");
profiles.Add("vs_1_1", "Vertex Shader model 1.1");
profiles.Add("vs_2_0", "Vertex Shader model 2.0");
profiles.Add("vs_2_a", "Vertex Shader model 2.0 A");
profiles.Add("vs_2_sw", "Vertex Shader model 2.0 (software)");
profiles.Add("vs_3_0", "Vertex Shader model 3.0");
profiles.Add("vs_3_sw", "Vertex Shader model 3.0 (software)");
profiles.Add("vs_4_0", "Vertex Shader model 4.0");
profiles.Add("vs_4_0_level_9_0", "Vertex Shader model 4.0 (level 9.0)");
profiles.Add("vs_4_0_level_9_1", "Vertex Shader model 4.0 (level 9.1)");
profiles.Add("vs_4_0_level_9_3", "Vertex Shader model 4.0 (level 9.3)");
profiles.Add("vs_4_1", "Vertex Shader model 4.1");
profiles.Add("vs_5_0", "Vertex Shader model 5.0");
}
static object locker = new object();
public static bool TryCompile(string code, ShaderModel model, string entrypoint, out string error) {
lock (locker)
{
string id = Thread.CurrentThread.ManagedThreadId.ToString();
string path = string.Format("{0}\\tmp{1}.fx", Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData), id);
using (FileStream fs = new FileStream(path, FileMode.Create))
{
byte[] data = Encoding.ASCII.GetBytes(code);
fs.Write(data, 0, data.Length);
}
string fxc = Path.Combine(new Uri(System.IO.Path.GetDirectoryName(Assembly.GetExecutingAssembly().GetName().CodeBase)).AbsolutePath, @"fxc.exe");
if (!File.Exists(fxc))
{
error = "No effect compiler executable!";
return false;
}
ProcessStartInfo psi = new ProcessStartInfo(fxc);
psi.CreateNoWindow = true;
psi.UseShellExecute = false;
psi.RedirectStandardError = true;
psi.Arguments = string.Format("/T {1} /E {2} /Fo\"{0}.obj\" \"{0}\"", path, model.Key, entrypoint);
error = string.Empty;
using (Process p = Process.Start(psi))
{
StreamReader sr = p.StandardError;
error = sr.ReadToEnd().Replace(path, "Line ");
if (!p.WaitForExit(5000))
{
error = "General failure while compiling (timeout).";
return false;
}
}
if (File.Exists(path))
File.Delete(path);
if (File.Exists(path + ".obj"))
File.Delete(path + ".obj");
if (error == string.Empty)
{
return true;
}
error = error.Replace("compilation failed; no code produced", "");
error = error.Trim();
return false;
}
}
}
}
这段代码属于 火星:(他真的应该得到赏金) http://marscode.blogspot.com.au/2011/04/directx-xna-shader-test-compiler-in-c.html
这是另一个有用的资源:
它展示了如何在public abstract class ShaderEffect : Effect
C# 中使用System.Windows.Media.Effects Namespace
他提到了 public class EffectCompiler
http://msdn.microsoft.com/library/windows/desktop/bb232919(v=vs.85).aspx
编辑:
请包括
using System.IO;
前
using System.Windows.Shapes;
它会抛出一个错误,但它会按照这个顺序工作,并发出警告,它仍然可以编译。Path
使用两个名称空间存在歧义 。
请告知这是否解决了您的问题。
没有简单的方法可以将 HLSL 的 Visual C++ MSBuild 规则添加到 C# 项目。通常,您必须使用自定义构建步骤,或者正如 StackOverflow 上其他地方所建议的那样,您可以在您的解决方案中拥有一个仅构建着色器的 C++ 项目。
您可以手动将 3D 资产和内容支持添加到 C# 项目,就像在 Stenning 的Direct3D 渲染食谱中对代码所做的那样,以获得 DDS、FBX->CMO 和 DGSL 着色器管道支持。您可以在GitHub 上查看项目。