@@ -58,7 +58,7 @@ def use_global_cache() -> bool:
58
58
_IS_MACOS = sys .platform .startswith ("darwin" )
59
59
_IS_WINDOWS = sys .platform == "win32"
60
60
61
- SUBPROCESS_DECODE_ARGS = ("oem " ,) if _IS_WINDOWS else ()
61
+ SUBPROCESS_DECODE_ARGS = ("utf-8 " ,) if _IS_WINDOWS else ()
62
62
63
63
log = logging .getLogger (__name__ )
64
64
@@ -198,6 +198,33 @@ def _is_msvc_cl(cpp_compiler: str) -> bool:
198
198
return False
199
199
200
200
201
+ @functools .lru_cache (None )
202
+ def _is_intel_compiler (cpp_compiler : str ) -> bool :
203
+ try :
204
+ output_msg = (
205
+ subprocess .check_output (
206
+ [cpp_compiler , "--version" ], stderr = subprocess .DEVNULL
207
+ )
208
+ .strip ()
209
+ .decode (* SUBPROCESS_DECODE_ARGS )
210
+ )
211
+ is_intel_compiler = "Intel" in output_msg .splitlines ()[0 ]
212
+ if is_intel_compiler :
213
+ if _IS_WINDOWS :
214
+ if re .search (r"((icx$)|(icx-cc$))" , cpp_compiler ):
215
+ raise RuntimeError (
216
+ "Please use icx-cl, due to torch.compile only support MSVC-like CLI (compiler flags syntax)."
217
+ )
218
+ return is_intel_compiler
219
+ except FileNotFoundError as exc :
220
+ return False
221
+ except subprocess .SubprocessError :
222
+ # --version args not support.
223
+ return False
224
+
225
+ return False
226
+
227
+
201
228
@functools .lru_cache (None )
202
229
def is_gcc () -> bool :
203
230
return _is_gcc (get_cpp_compiler ())
@@ -208,6 +235,11 @@ def is_clang() -> bool:
208
235
return _is_clang (get_cpp_compiler ())
209
236
210
237
238
+ @functools .lru_cache (None )
239
+ def is_intel_compiler () -> bool :
240
+ return _is_intel_compiler (get_cpp_compiler ())
241
+
242
+
211
243
@functools .lru_cache (None )
212
244
def is_apple_clang () -> bool :
213
245
return _is_apple_clang (get_cpp_compiler ())
@@ -798,6 +830,20 @@ def perload_clang_libomp_win(cpp_compiler: str, omp_name: str) -> None:
798
830
pass
799
831
800
832
833
+ @functools .lru_cache (None )
834
+ def perload_icx_libomp_win (cpp_compiler : str ) -> None :
835
+ try :
836
+ output = subprocess .check_output (
837
+ [cpp_compiler , "-print-file-name=libiomp5md.dll" ], stderr = subprocess .DEVNULL
838
+ ).decode (* SUBPROCESS_DECODE_ARGS )
839
+ omp_path = output .rstrip ()
840
+ if os .path .isfile (omp_path ):
841
+ os .environ ["KMP_DUPLICATE_LIB_OK" ] = "TRUE"
842
+ omp_module = cdll .LoadLibrary (omp_path )
843
+ except subprocess .SubprocessError :
844
+ pass
845
+
846
+
801
847
def _get_openmp_args (
802
848
cpp_compiler : str ,
803
849
) -> Tuple [List [str ], List [str ], List [str ], List [str ], List [str ], List [str ]]:
@@ -854,10 +900,28 @@ def _get_openmp_args(
854
900
# if openmp is still not available, we let the compiler to have a try,
855
901
# and raise error together with instructions at compilation error later
856
902
elif _IS_WINDOWS :
903
+ """
904
+ On Windows, `clang` and `icx` have their specific openmp implenmention.
905
+ And the openmp lib is in compiler's some sub-directory.
906
+ For dynamic library(DLL) load, the Windows native APIs are `LoadLibraryA` and `LoadLibraryExA`, and their search
907
+ dependencies have some rules:
908
+ https://learn.microsoft.com/en-us/windows/win32/api/libloaderapi/nf-libloaderapi-loadlibraryexa#searching-for-dlls-and-dependencies
909
+ In some case, the rules may not include compiler's sub-directories.
910
+ So, it can't search and load compiler's openmp library correctly.
911
+ And then, the whole application would be broken.
912
+
913
+ To avoid the openmp load failed, we can automatic locate the openmp binary and preload it.
914
+ 1. For clang, the function is `perload_clang_libomp_win`.
915
+ 2. For icx, the function is `perload_icx_libomp_win`.
916
+ """
857
917
if _is_clang (cpp_compiler ):
858
918
cflags .append ("openmp" )
859
919
libs .append ("libomp" )
860
920
perload_clang_libomp_win (cpp_compiler , "libomp.dll" )
921
+ elif _is_intel_compiler (cpp_compiler ):
922
+ cflags .append ("Qiopenmp" )
923
+ libs .append ("libiomp5md" )
924
+ perload_icx_libomp_win (cpp_compiler )
861
925
else :
862
926
# /openmp, /openmp:llvm
863
927
# llvm on Windows, new openmp: https://devblogs.microsoft.com/cppblog/msvc-openmp-update/
0 commit comments