我们正在测试一个项目设置,要求我们根据编译器的版本包含或排除源文件。测试项目位于CRC-Test的GitHub上。对于PoC,我们正在开发CRC-32C的过程,因为Intel和ARM都有它。稍后它将应用于AES,CLMUL,SHA,AVX {2},BMI {2},ARMv7a NEON,ARMv8等。(MS编译器通过内在函数支持NEON和ARMv8)。
我尝试有条件编译的源文件是crc-simd.cpp
。它具有使用SSE4.2内在函数和ARMv8内在函数的实现(crc.cpp
提供了通用的C ++实现)。
我在crc-test.vcxproj添加了一个VCX项目文件。 ItemGroup
有一个ClCompile
:
<ItemGroup>
<ClCompile Include="main.cpp" >
<PrecompiledHeader />
</ClCompile>
<ClCompile Include="crc.cpp" >
<PrecompiledHeader />
</ClCompile>
<ClCompile Include="crc-simd.cpp" >
<PrecompiledHeader />
</ClCompile>
</ItemGroup>
我需要<ClCompile Include="crc-simd.cpp" >
的条件,其中“如果cl.exe
为15.00或更高,则包含源文件”。在预处理器中,我会检查_MSC_VER >= 1500
。在MSBuild中,我想我正在寻找类似的东西:
<ItemGroup Condition="'$(CL_VERSION)'>='15.00'" Label="SSE4.2 acceleration">
<ClCompile Include="crc-simd.cpp" >
<PrecompiledHeader />
</ClCompile>
</ItemDefinitionGroup>
我找到了List of MSBuild built-in variables,但我不知道$(CL_VERSION)
在现实生活中是什么。 “cl”的搜索没有引起有趣的命中。我在MSBuild vs compiler找到了一个问题,但它提出了一个不同的问题。
我的问题是,如何在MSBuild条件中引用cl.exe
版本?
相关问题是,有时我需要完整版。例如,在VS2005 SP1,cl.exe
版本15.00.30729(_MSC_FULL_VER >= 150030729
)添加了AES支持。我的相关问题是,如何在MSBuild条件中引用cl.exe
完整版本?
尝试在开发人员提示中使用以下内容:
<ItemDefinitionGroup Condition="'$(CL_VERSION)'>='15.00'" Label="SSE4.2 acceleration">
<ClCompile Include="crc-simd.cpp" >
<PrecompiledHeader />
</ClCompile>
</ItemDefinitionGroup>
结果:
> cls && msbuild
Microsoft (R) Build Engine version 4.6.1087.0
[Microsoft .NET Framework, version 4.0.30319.42000]
Copyright (C) Microsoft Corporation. All rights reserved.
Build started 7/25/2017 1:23:31 PM.
Project "C:\Users\Test\CRC-Test\crc-test.vcxproj" on node 1 (default ta
rgets).
C:\Users\Test\CRC-Test\crc-test.vcxproj(127,14): error MSB4066: The att
ribute "Include" in element <ClCompile> is unrecognized.
Done Building Project "C:\Users\Test\CRC-Test\crc-test.vcxproj" (defaul
t targets) -- FAILED.
Build FAILED.
"C:\Users\Test\CRC-Test\crc-test.vcxproj" (default target) (1) ->
C:\Users\Test\CRC-Test\crc-test.vcxproj(127,14): error MSB4066: The a
ttribute "Include" in element <ClCompile> is unrecognized.
0 Warning(s)
1 Error(s)
Time Elapsed 00:00:00.11
该项目目前正在为Visual Studio 2010设置。如果您使用的是其他版本,请将其更改为符合您的口味。不需要VCUpgrade
:
<PlatformToolset>v100</PlatformToolset>
以下是GNUmakefile
中Linux项目的精髓。当编译器和平台支持硬件加速时,我们会引入crc-simd.cpp
文件以加快实施速度。这就是我们想要用MSBuild做的事情。
HAS_ARMV8 ?= $(shell uname -m | $(EGREP) -i -c 'aarch32|aarch64')
HAS_SSE42 ?= $(shell uname -m | $(EGREP) -i -c 'amd64|x86_64|i*86')
GCC43_OR_LATER := $(shell $(CXX) -v 2>&1 | $(EGREP) -i -c "gcc version (4\.[3-9]|[5-9]\.)")
GCC48_OR_LATER := $(shell $(CXX) -v 2>&1 | $(EGREP) -i -c "gcc version (4\.[8-9]|[5-9]\.)")
...
OBJECTS = main.o crc.o
ifeq ($(HAS_ARMV8)$(GCC48_OR_LATER),11)
OBJECTS += crc-simd.o
endif
ifeq ($(HAS_SSE42)$(GCC43_OR_LATER),11)
OBJECTS += crc-simd.o
endif
...
ifeq ($(HAS_ARMV8)$(GCC48_OR_LATER),11)
crc-simd.o : crc-simd.cpp
$(CXX) $(CXXFLAGS) -march=armv8-a+crc -c $<
endif
ifeq ($(HAS_SSE42)$(GCC43_OR_LATER),11)
crc-simd.o : crc-simd.cpp
$(CXX) $(CXXFLAGS) -msse4.2 -c $<
endif
...
答案 0 :(得分:4)
可以使用与makefile类似的方法:运行cl.exe,使用正则表达式获取版本,如果版本大于某个数字,则包含源文件。在代码中:
<Target Name="GetClVersion">
<!-- Run cl, store output in variable -->
<Exec Command="cl.exe" ConsoleToMSBuild="True" IgnoreExitCode="True">
<Output TaskParameter="ConsoleOutput" PropertyName="ClOut" />
</Exec>
<PropertyGroup>
<!-- cl.exe version number is something like 18.00.31101 -->
<ClVersion>$([System.Text.RegularExpressions.Regex]::Match('$(ClOut)','(\d+\.\d+\.\d+)'))</ClVersion>
<!-- Turn it into an integer like 180031101 by getting rid of dots -->
<ClVersion>$([System.String]::Copy('$(ClVersion)').Replace('.', ''))</ClVersion>
</PropertyGroup>
</Target>
<!-- This will run early in the build, before compilation,
so we can modify the ClCompile ItemGroup. Needs to be done
in a target, else we cannot use results from other targets
(GetClVersion in this case) -->
<Target Name="ShowClVersion" DependsOnTargets="GetClVersion"
BeforeTargets="BuildGenerateSources">
<!-- > = escaped > sign -->
<ItemGroup Condition="$(ClVersion) > 180031100">
<ClCompile Include="somefile.cpp" />
</ItemGroup>
</Target>
注意:您找到的内置变量列表适用于构建系统msbuild,它与编译器分开;这就是为什么没有CL_VERSION就像makefile默认没有GCC_VERSION一样。还值得尝试使用
<PlatformToolset>$(DefaultPlatformToolset)</PlatformToolset>
而不是坚持一个或保留多个项目,每个VS版本一个。它还使VS不要求升级(因为它看到它正在使用的默认平台工具集)。我没有VS版本&lt; 2012年测试它是否在那里工作,但它确实从VS2012开始,因此已经有4个,并且还在计算。