From 430d690f04f375d0762eebc0d7cf26cae41b02ee Mon Sep 17 00:00:00 2001 From: "Cui, Yifeng" Date: Wed, 23 Oct 2024 23:48:09 -0700 Subject: [PATCH] Init MKL for Pytorch XPU and enable fft_c2c --- CMakeLists.txt | 1 + cmake/Modules/FindONEMKL.cmake | 66 ++++ cmake/ONEMKL.cmake | 11 + src/ATen/CMakeLists.txt | 3 + src/ATen/native/xpu/SpectralOps.cpp | 28 ++ src/ATen/native/xpu/XPUFallback.template | 1 - src/ATen/native/xpu/mkl/SpectralOps.cpp | 484 +++++++++++++++++++++++ src/ATen/native/xpu/mkl/SpectralOps.h | 18 + src/BuildOnLinux.cmake | 1 + src/BuildOnWindows.cmake | 1 + src/CMakeLists.txt | 4 + test/xpu/skip_list_common.py | 25 ++ test/xpu/test_spectral_ops_xpu.py | 81 ++++ test/xpu/xpu_test_utils.py | 21 +- yaml/native/native_functions.yaml | 11 + 15 files changed, 754 insertions(+), 2 deletions(-) create mode 100644 cmake/Modules/FindONEMKL.cmake create mode 100644 cmake/ONEMKL.cmake create mode 100644 src/ATen/native/xpu/SpectralOps.cpp create mode 100644 src/ATen/native/xpu/mkl/SpectralOps.cpp create mode 100644 src/ATen/native/xpu/mkl/SpectralOps.h create mode 100644 test/xpu/test_spectral_ops_xpu.py diff --git a/CMakeLists.txt b/CMakeLists.txt index d67ead44e..6b20296b4 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -31,6 +31,7 @@ set(TORCH_XPU_OPS_ROOT ${PROJECT_SOURCE_DIR}) list(APPEND CMAKE_MODULE_PATH ${TORCH_XPU_OPS_ROOT}/cmake/Modules) include(${TORCH_XPU_OPS_ROOT}/cmake/SYCL.cmake) +include(${TORCH_XPU_OPS_ROOT}/cmake/ONEMKL.cmake) include(${TORCH_XPU_OPS_ROOT}/cmake/BuildFlags.cmake) if(BUILD_TEST) diff --git a/cmake/Modules/FindONEMKL.cmake b/cmake/Modules/FindONEMKL.cmake new file mode 100644 index 000000000..6204cdc64 --- /dev/null +++ b/cmake/Modules/FindONEMKL.cmake @@ -0,0 +1,66 @@ +set(ONEMKL_FOUND FALSE) + +set(ONEMKL_LIBRARIES) + +# In order to be compatible with various situations of Pytorch development +# bundle setup, ENV{MKLROOT} and SYCL_ROOT will be checked sequentially to get +# the root directory of oneMKL. +if(DEFINED ENV{MKLROOT}) + # Directly get the root directory of oneMKL if ENV{MKLROOT} exists. + set(ONEMKL_ROOT $ENV{MKLROOT}) +elseif(SYCL_FOUND) + # oneMKL configuration may not be imported into the build system. Get the root + # directory of oneMKL based on the root directory of compiler relatively. + get_filename_component(ONEMKL_ROOT "${SYCL_ROOT}/../../mkl/latest" REALPATH) +endif() + +if(NOT DEFINED ONEMKL_ROOT) + message( + WARNING + "Cannot find either ENV{MKLROOT} or SYCL_ROOT, please setup oneAPI environment before building!!" + ) + return() +endif() + +if(NOT EXISTS ${ONEMKL_ROOT}) + message( + WARNING + "${ONEMKL_ROOT} not found, please setup oneAPI environment before building!!" + ) + return() +endif() + +find_file( + ONEMKL_INCLUDE_DIR + NAMES include + HINTS ${ONEMKL_ROOT} + NO_DEFAULT_PATH) + +find_file( + ONEMKL_LIB_DIR + NAMES lib + HINTS ${ONEMKL_ROOT} + NO_DEFAULT_PATH) + +if((ONEMKL_INCLUDE_DIR STREQUAL "ONEMKL_INCLUDE_DIR-NOTFOUND") + OR (ONEMKL_LIB_DIR STREQUAL "ONEMKL_LIB_DIR-NOTFOUND")) + message(WARNING "oneMKL sdk is incomplete!!") + return() +endif() + +if(WIN32) + set(MKL_LIB_NAMES "mkl_intel_lp64" "mkl_intel_thread" "mkl_core" "mkl_sycl") +else() + set(MKL_LIB_NAMES "mkl_intel_lp64" "mkl_gnu_thread" "mkl_core" "mkl_sycl_dft") +endif() + +foreach(LIB_NAME IN LISTS MKL_LIB_NAMES) + find_library( + ${LIB_NAME}_library + NAMES ${LIB_NAME} + HINTS ${ONEMKL_LIB_DIR} + NO_CMAKE_PATH NO_CMAKE_ENVIRONMENT_PATH) + list(APPEND ONEMKL_LIBRARIES ${${LIB_NAME}_library}) +endforeach() + +set(ONEMKL_FOUND TRUE) diff --git a/cmake/ONEMKL.cmake b/cmake/ONEMKL.cmake new file mode 100644 index 000000000..84a19816f --- /dev/null +++ b/cmake/ONEMKL.cmake @@ -0,0 +1,11 @@ +find_package(ONEMKL) +if(NOT ONEMKL_FOUND) + message(FATAL_ERROR "Can NOT find ONEMKL cmake helpers module!") +endif() + +set(TORCH_XPU_OPS_ONEMKL_INCLUDE_DIR ${ONEMKL_INCLUDE_DIR}) + +set(TORCH_XPU_OPS_ONEMKL_LIBRARIES ${ONEMKL_LIBRARIES}) + +list(INSERT TORCH_XPU_OPS_ONEMKL_LIBRARIES 0 "-Wl,--no-as-needed") +list(APPEND TORCH_XPU_OPS_ONEMKL_LIBRARIES "-Wl,--as-needed") diff --git a/src/ATen/CMakeLists.txt b/src/ATen/CMakeLists.txt index 3c4326f47..49ffd7433 100644 --- a/src/ATen/CMakeLists.txt +++ b/src/ATen/CMakeLists.txt @@ -2,14 +2,17 @@ file(GLOB xpu_h "xpu/*.h") file(GLOB xpu_cpp "xpu/*.cpp") +file(GLOB xpu_mkl "native/xpu/mkl/*.cpp") file(GLOB xpu_native_cpp "native/xpu/*.cpp" "native/sparse/*.cpp") file(GLOB xpu_sycl "native/xpu/sycl/*.cpp") list(APPEND ATen_XPU_CPP_SRCS ${xpu_cpp}) +list(APPEND ATen_XPU_MKL_SRCS ${xpu_mkl}) list(APPEND ATen_XPU_NATIVE_CPP_SRCS ${xpu_native_cpp}) list(APPEND ATen_XPU_SYCL_SRCS ${xpu_sycl}) set(ATen_XPU_CPP_SRCS ${ATen_XPU_CPP_SRCS} PARENT_SCOPE) +set(ATen_XPU_MKL_SRCS ${ATen_XPU_MKL_SRCS} PARENT_SCOPE) set(ATen_XPU_NATIVE_CPP_SRCS ${ATen_XPU_NATIVE_CPP_SRCS} PARENT_SCOPE) set(ATen_XPU_SYCL_SRCS ${ATen_XPU_SYCL_SRCS} PARENT_SCOPE) diff --git a/src/ATen/native/xpu/SpectralOps.cpp b/src/ATen/native/xpu/SpectralOps.cpp new file mode 100644 index 000000000..af82394f1 --- /dev/null +++ b/src/ATen/native/xpu/SpectralOps.cpp @@ -0,0 +1,28 @@ +#include +#include +#include + +namespace at::native { + +Tensor _fft_c2c_xpu( + const Tensor& self, + IntArrayRef dim, + int64_t normalization, + bool forward) { + TORCH_CHECK(self.is_complex()); + + return native::xpu::_fft_c2c_mkl(self, dim, normalization, forward); +} + +Tensor& _fft_c2c_xpu_out( + const Tensor& self, + IntArrayRef dim, + int64_t normalization, + bool forward, + Tensor& out) { + TORCH_CHECK(self.is_complex()); + + return native::xpu::_fft_c2c_mkl_out(self, dim, normalization, forward, out); +} + +} // namespace at::native diff --git a/src/ATen/native/xpu/XPUFallback.template b/src/ATen/native/xpu/XPUFallback.template index 8e89df47c..526a9eacf 100644 --- a/src/ATen/native/xpu/XPUFallback.template +++ b/src/ATen/native/xpu/XPUFallback.template @@ -160,7 +160,6 @@ TORCH_LIBRARY_IMPL(aten, XPU, m) { "_cummin_helper", "dot", "_efficient_attention_forward", - "_fft_c2c", "_fft_c2r", "_fft_r2c", "_flash_attention_forward", diff --git a/src/ATen/native/xpu/mkl/SpectralOps.cpp b/src/ATen/native/xpu/mkl/SpectralOps.cpp new file mode 100644 index 000000000..b66966b25 --- /dev/null +++ b/src/ATen/native/xpu/mkl/SpectralOps.cpp @@ -0,0 +1,484 @@ +#include +#include +#include +#include +#include +#include +#include + +using namespace oneapi::mkl::dft; + +namespace at::native::xpu { + +namespace impl { + +constexpr int64_t mkl_max_ndim = 3; + +// Sort transform dimensions by input layout, for best performance +// exclude_last is for onesided transforms where the last dimension cannot be +// reordered +static DimVector _sort_dims( + const Tensor& self, + IntArrayRef dim, + bool exclude_last = false) { + DimVector sorted_dims(dim.begin(), dim.end()); + auto self_strides = self.strides(); + std::sort( + sorted_dims.begin(), + sorted_dims.end() - exclude_last, + [&](int64_t a, int64_t b) { return self_strides[a] > self_strides[b]; }); + return sorted_dims; +} + +class dft_config_t { + public: + using config_int64_t = std::unordered_map; + using config_float_t = std::unordered_map; + using config_double_t = std::unordered_map; + + dft_config_t() { + val_int64_.clear(); + val_float_.clear(); + val_double_.clear(); + fwd_strides_.clear(); + bwd_strides_.clear(); + } + + void set_strides( + std::vector& fwd_strides, + std::vector& bwd_strides) { + fwd_strides_ = fwd_strides; + bwd_strides_ = bwd_strides; + } + + template + void set_value(config_param key, T value) { + if (std::is_same::value || + std::is_same::value) { + val_int64_.insert({key, value}); + } else if (std::is_same::value) { + val_float_.insert({key, value}); + } else if (std::is_same::value) { + val_double_.insert({key, value}); + } else { + TORCH_CHECK(0, "Unsupported value type in FFT config!"); + } + } + + template + void commit_values(descriptor& desc) { +#define COMMIT_VAL(val_map) \ + for (auto& value : (val_map)) { \ + desc.set_value(value.first, value.second); \ + } + + COMMIT_VAL(val_int64_); + COMMIT_VAL(val_float_); + COMMIT_VAL(val_double_); + + if (!fwd_strides_.empty()) { + desc.set_value(config_param::FWD_STRIDES, fwd_strides_.data()); + } + if (!bwd_strides_.empty()) { + desc.set_value(config_param::BWD_STRIDES, bwd_strides_.data()); + } + } + + private: + config_int64_t val_int64_; + config_float_t val_float_; + config_double_t val_double_; + std::vector fwd_strides_; + std::vector bwd_strides_; +}; + +template +class dft_desc_t { + public: + using mkl_desc_t = descriptor; + + dft_desc_t( + sycl::queue& q, + std::vector& dimensions, + std::shared_ptr configs) + : desc_(dimensions), configs_(configs) { + configs_->commit_values(desc_); + desc_.set_value( + oneapi::mkl::dft::config_param::WORKSPACE, + oneapi::mkl::dft::config_value::WORKSPACE_EXTERNAL); + desc_.commit(q); + } + + mkl_desc_t& raw() { + return desc_; + } + + private: + mkl_desc_t desc_; + std::shared_ptr configs_; +}; + +template +void _mkl_dft( + const Tensor& input, + Tensor& output, + int64_t signal_ndim, + bool complex_input, + bool complex_output, + bool inverse, + IntArrayRef checked_signal_sizes, + bool onesided, + int64_t batch) { + auto& queue = at::xpu::getCurrentSYCLQueue(); + std::vector mkl_signal_sizes( + checked_signal_sizes.begin() + 1, checked_signal_sizes.end()); + + std::shared_ptr desc_config(new dft_config_t); + desc_config->set_value(config_param::PLACEMENT, DFTI_NOT_INPLACE); + desc_config->set_value(config_param::NUMBER_OF_TRANSFORMS, batch); + + auto istrides = input.strides(); + auto ostrides = output.strides(); + int64_t idist = istrides[0]; + int64_t odist = ostrides[0]; + + if (!inverse) { + desc_config->set_value(config_param::FWD_DISTANCE, idist); + desc_config->set_value(config_param::BWD_DISTANCE, odist); + } else { + desc_config->set_value(config_param::FWD_DISTANCE, odist); + desc_config->set_value(config_param::BWD_DISTANCE, idist); + } + + std::vector fwd_strides(1 + signal_ndim, 0), + bwd_strides(1 + signal_ndim, 0); + + for (int64_t i = 1; i <= signal_ndim; i++) { + if (!inverse) { + fwd_strides[i] = istrides[i]; + bwd_strides[i] = ostrides[i]; + } else { + fwd_strides[i] = ostrides[i]; + bwd_strides[i] = istrides[i]; + } + } + + desc_config->set_strides(fwd_strides, bwd_strides); + + if (!complex_input || !complex_output) { + desc_config->set_value( + config_param::CONJUGATE_EVEN_STORAGE, DFTI_COMPLEX_COMPLEX); + } + + auto desc = + dft_desc_t(queue, mkl_signal_sizes, desc_config); + + // Obtain the size of workspace required after commit. + size_t workspaceSizeBytes = 0; + desc.raw().get_value( + oneapi::mkl::dft::config_param::WORKSPACE_BYTES, &workspaceSizeBytes); + + // Allocate USM workspace and provide it to the descriptor. + Tensor workspaceBuf = at::empty( + {(long)(workspaceSizeBytes / sizeof(double))}, + input.options().dtype(at::kDouble), + c10::nullopt); + desc.raw().set_workspace((double*)workspaceBuf.data_ptr()); + + auto in_data = (scalar_t*)input.data_ptr(); + auto out_data = (scalar_t*)output.data_ptr(); + + sycl::event event; + if (!inverse) { + event = compute_forward(desc.raw(), in_data, out_data); + } else { + event = compute_backward(desc.raw(), in_data, out_data); + } + event.wait_and_throw(); + queue.throw_asynchronous(); +} + +void _fft_with_size( + Tensor& output, + const Tensor& self, + int64_t signal_ndim, + bool complex_input, + bool complex_output, + bool inverse, + IntArrayRef checked_signal_sizes, + bool onesided) { + int64_t batch = self.size(0); + Tensor input_ = self; + // real/imag dimension must aligned when viewed as of complex type + + if (complex_input) { + bool need_contiguous = input_.stride(-1) != 1; + + for (int64_t i = 0; !need_contiguous && i <= signal_ndim; i++) { + need_contiguous |= input_.stride(i) % 2 != 0; + } + + if (need_contiguous) { + input_ = input_.contiguous(); + } + } + + bool complex_type = inverse ? complex_output : complex_input; + + void (*dft_func)( + const class at::Tensor&, + class at::Tensor&, + int64_t, + bool, + bool, + bool, + class c10::ArrayRef, + bool, + int64_t); + Tensor input = input_; + + if (input.scalar_type() == ScalarType::Float || + input.scalar_type() == ScalarType::ComplexFloat) { + dft_func = complex_type + ? _mkl_dft + : _mkl_dft; + } else if ( + input.scalar_type() == ScalarType::Double || + input.scalar_type() == ScalarType::ComplexDouble) { + dft_func = complex_type + ? _mkl_dft + : _mkl_dft; + } else { + AT_ERROR("MKL FFT doesn't support tensor of type"); + } + + dft_func( + input, + output, + signal_ndim, + complex_input, + complex_output, + inverse, + checked_signal_sizes, + onesided, + batch); +} + +// Execute a general fft operation (can be c2c, onesided r2c or onesided c2r) +Tensor& _exec_fft( + Tensor& out, + Tensor self, + IntArrayRef out_sizes, + IntArrayRef dim, + bool onesided, + bool forward) { + const auto ndim = self.dim(); + const int64_t signal_ndim = dim.size(); + const auto batch_dims = ndim - signal_ndim; + + // Permute dimensions so batch dimensions come first, and in stride order + // This maximizes data locality when collapsing to a single batch dimension + DimVector dim_permute(ndim); + std::iota(dim_permute.begin(), dim_permute.end(), int64_t{0}); + + c10::SmallVector is_transformed_dim(ndim); + for (const auto& d : dim) { + is_transformed_dim[d] = true; + } + + auto batch_end = + std::partition(dim_permute.begin(), dim_permute.end(), [&](int64_t d) { + return !is_transformed_dim[d]; + }); + + auto self_strides = self.strides(); + std::sort(dim_permute.begin(), batch_end, [&](int64_t a, int64_t b) { + return self_strides[a] > self_strides[b]; + }); + std::copy(dim.cbegin(), dim.cend(), batch_end); + + auto input = self.permute(dim_permute); + + // Collapse batch dimensions into a single dimension + DimVector batched_sizes(signal_ndim + 1); + batched_sizes[0] = -1; + std::copy( + input.sizes().cbegin() + batch_dims, + input.sizes().cend(), + batched_sizes.begin() + 1); + input = input.reshape(batched_sizes); + + const auto batch_size = input.sizes()[0]; + DimVector signal_size(signal_ndim + 1); + signal_size[0] = batch_size; + + for (int64_t i = 0; i < signal_ndim; ++i) { + auto in_size = input.sizes()[i + 1]; + auto out_size = out_sizes[dim[i]]; + signal_size[i + 1] = std::max(in_size, out_size); + TORCH_INTERNAL_ASSERT( + in_size == signal_size[i + 1] || + in_size == (signal_size[i + 1] / 2) + 1); + TORCH_INTERNAL_ASSERT( + out_size == signal_size[i + 1] || + out_size == (signal_size[i + 1] / 2) + 1); + } + + batched_sizes[0] = batch_size; + DimVector batched_out_sizes(batched_sizes.begin(), batched_sizes.end()); + + for (size_t i = 0; i < dim.size(); ++i) { + batched_out_sizes[i + 1] = out_sizes[dim[i]]; + } + + out.resize_(batched_out_sizes, MemoryFormat::Contiguous); + + // run the FFT + _fft_with_size( + out, + input, + signal_ndim, + input.is_complex(), + out.is_complex(), + !forward, + signal_size, + onesided); + + // Inplace reshaping to original batch shape and inverting the dimension + // permutation + DimVector out_strides(ndim); + int64_t batch_numel = 1; + + for (int64_t i = batch_dims - 1; i >= 0; --i) { + out_strides[dim_permute[i]] = batch_numel * out.strides()[0]; + batch_numel *= out_sizes[dim_permute[i]]; + } + + for (int64_t i = batch_dims; i < ndim; ++i) { + out_strides[dim_permute[i]] = out.strides()[1 + (i - batch_dims)]; + } + + out.as_strided_(out_sizes, out_strides, out.storage_offset()); + + return out; +} + +double _dft_scale( + IntArrayRef dim, + IntArrayRef input_sizes, + IntArrayRef out_sizes, + int64_t normalization) { + const auto norm = static_cast(normalization); + double double_scale = 1.0; + + if (norm == fft_norm_mode::none) { + return double_scale; + } + + const int64_t signal_ndim = dim.size(); + int64_t signal_numel = 1; + + for (int64_t i = 0; i < signal_ndim; ++i) { + auto in_size = input_sizes[dim[i]]; + auto out_size = out_sizes[dim[i]]; + auto signal_size = std::max(in_size, out_size); + + signal_numel *= signal_size; + TORCH_INTERNAL_ASSERT( + in_size == signal_size || in_size == (signal_size / 2) + 1); + TORCH_INTERNAL_ASSERT( + out_size == signal_size || out_size == (signal_size / 2) + 1); + } + + if (norm == fft_norm_mode::by_root_n) { + double_scale = 1.0 / std::sqrt(signal_numel); + } else { + double_scale = 1.0 / static_cast(signal_numel); + } + + return double_scale; +} + +const Tensor& _fft_apply_normalization( + const Tensor& self, + int64_t normalization, + IntArrayRef sizes, + IntArrayRef dims) { + auto scale = _dft_scale(dims, sizes, self.sizes(), normalization); + return (scale == 1.0) ? self : self.mul_(scale); +} + +Tensor& _fft_apply_normalization_out( + Tensor& out, + const Tensor& self, + int64_t normalization, + IntArrayRef sizes, + IntArrayRef dims) { + auto scale = _dft_scale(dims, sizes, self.sizes(), normalization); + return at::mul_out(out, self, c10::scalar_to_tensor(scale)); +} + +} // namespace impl + +Tensor _fft_c2c_mkl( + const Tensor& self, + IntArrayRef dim, + int64_t normalization, + bool forward) { + if (dim.empty()) { + return self.clone(); + } + + auto sorted_dims = impl::_sort_dims(self, dim); + auto out_sizes = self.sizes(); + auto out = at::empty(out_sizes, self.options()); + auto input_sizes = self.sizes(); + auto working_tensor = self; + + while (!sorted_dims.empty()) { + const auto max_dims = + std::min(static_cast(impl::mkl_max_ndim), sorted_dims.size()); + auto fft_dims = + IntArrayRef(sorted_dims).slice(sorted_dims.size() - max_dims, max_dims); + + impl::_exec_fft( + out, + working_tensor, + out_sizes, + fft_dims, + /*onesided=*/false, + forward); + + sorted_dims.resize(sorted_dims.size() - max_dims); + + if (sorted_dims.empty()) { + break; + } + + sorted_dims = impl::_sort_dims(self, sorted_dims); + + if (working_tensor.is_same(self)) { + working_tensor = std::move(out); + out = at::empty(out_sizes, self.options()); + } else { + std::swap(out, working_tensor); + } + } + + return impl::_fft_apply_normalization(out, normalization, input_sizes, dim); +} + +Tensor& _fft_c2c_mkl_out( + const Tensor& self, + IntArrayRef dim, + int64_t normalization, + bool forward, + Tensor& out) { + auto result = _fft_c2c_mkl( + self, dim, static_cast(fft_norm_mode::none), forward); + at::native::resize_output(out, result.sizes()); + return impl::_fft_apply_normalization_out( + out, result, normalization, result.sizes(), dim); +} + +} // namespace at::native::xpu diff --git a/src/ATen/native/xpu/mkl/SpectralOps.h b/src/ATen/native/xpu/mkl/SpectralOps.h new file mode 100644 index 000000000..39763e428 --- /dev/null +++ b/src/ATen/native/xpu/mkl/SpectralOps.h @@ -0,0 +1,18 @@ +#pragma once + +namespace at::native::xpu { + +Tensor _fft_c2c_mkl( + const Tensor& self, + IntArrayRef dim, + int64_t normalization, + bool forward); + +Tensor& _fft_c2c_mkl_out( + const Tensor& self, + IntArrayRef dim, + int64_t normalization, + bool forward, + Tensor& out); + +} // namespace at::native::xpu diff --git a/src/BuildOnLinux.cmake b/src/BuildOnLinux.cmake index 3e304b31f..68d5941a3 100644 --- a/src/BuildOnLinux.cmake +++ b/src/BuildOnLinux.cmake @@ -7,6 +7,7 @@ add_library( torch_xpu_ops STATIC ${ATen_XPU_CPP_SRCS} + ${ATen_XPU_MKL_SRCS} ${ATen_XPU_NATIVE_CPP_SRCS} ${ATen_XPU_GEN_SRCS}) diff --git a/src/BuildOnWindows.cmake b/src/BuildOnWindows.cmake index 3b463f9c4..04fea6cc9 100644 --- a/src/BuildOnWindows.cmake +++ b/src/BuildOnWindows.cmake @@ -12,6 +12,7 @@ add_library( torch_xpu_ops STATIC ${ATen_XPU_CPP_SRCS} + ${ATen_XPU_MKL_SRCS} "bridge.cpp") set(PATH_TO_TORCH_XPU_OPS_ATEN_LIB \"torch_xpu_ops_aten.dll\") target_compile_options(torch_xpu_ops PRIVATE -DPATH_TO_TORCH_XPU_OPS_ATEN_LIB=${PATH_TO_TORCH_XPU_OPS_ATEN_LIB}) diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 0716ca5af..de2a5ea7b 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -2,6 +2,7 @@ include(${TORCH_XPU_OPS_ROOT}/cmake/Codegen.cmake) set(ATen_XPU_CPP_SRCS) +set(ATen_XPU_MKL_SRCS) set(ATen_XPU_NATIVE_CPP_SRCS) set(ATen_XPU_SYCL_SRCS) @@ -27,3 +28,6 @@ if(CLANG_FORMAT) add_custom_target(CL_FORMAT_CSRCS COMMAND ${CLANG_FORMAT_EXEC} -i -style=file ${ALL_CSRCS}) add_dependencies(torch_xpu_ops CL_FORMAT_CSRCS) endif() + +target_include_directories(torch_xpu_ops PUBLIC ${TORCH_XPU_OPS_ONEMKL_INCLUDE_DIR}) +target_link_libraries(torch_xpu_ops PUBLIC ${TORCH_XPU_OPS_ONEMKL_LIBRARIES}) diff --git a/test/xpu/skip_list_common.py b/test/xpu/skip_list_common.py index cbd6822bd..2da5880df 100644 --- a/test/xpu/skip_list_common.py +++ b/test/xpu/skip_list_common.py @@ -696,6 +696,26 @@ # FATAL: Unexpected page fault from GPU at 0x0, ctx_id: 1 (CCS) type: 0 (NotPresent), level: 3 (PML4), access: 0 (Read), banned: 1, aborting. # FATAL: Unexpected page fault from GPU at 0x0, ctx_id: 1 (CCS) type: 0 (NotPresent), level: 3 (PML4), access: 0 (Read), banned: 1, aborting. "test_dtypes__refs_nn_functional_pdist_xpu", + + # The following dtypes did not work in backward but are listed by the OpInfo: {torch.bfloat16}. + "test_dtypes_fft_fft2_xpu", + "test_dtypes_fft_fft_xpu", + "test_dtypes_fft_fftn_xpu", + "test_dtypes_fft_hfft2_xpu", + "test_dtypes_fft_hfft_xpu", + "test_dtypes_fft_hfftn_xpu", + "test_dtypes_fft_ifft2_xpu", + "test_dtypes_fft_ifft_xpu", + "test_dtypes_fft_ifftn_xpu", + "test_dtypes_fft_ihfft2_xpu", + "test_dtypes_fft_ihfft_xpu", + "test_dtypes_fft_ihfftn_xpu", + "test_dtypes_fft_irfft2_xpu", + "test_dtypes_fft_irfft_xpu", + "test_dtypes_fft_irfftn_xpu", + "test_dtypes_fft_rfft2_xpu", + "test_dtypes_fft_rfft_xpu", + "test_dtypes_fft_rfftn_xpu", ), "test_binary_ufuncs_xpu.py": ( @@ -3475,4 +3495,9 @@ ), "test_distributions_xpu.py": None, + + "test_spectral_ops_xpu.py": ( + # CUDA specific case + "test_cufft_plan_cache_xpu_float64", + ), } diff --git a/test/xpu/test_spectral_ops_xpu.py b/test/xpu/test_spectral_ops_xpu.py new file mode 100644 index 000000000..bc60cf2ae --- /dev/null +++ b/test/xpu/test_spectral_ops_xpu.py @@ -0,0 +1,81 @@ +# Owner(s): ["module: intel"] + +import torch +import numpy as np +from packaging import version +from itertools import product + +from torch.testing._internal.common_device_type import ( + instantiate_device_type_tests, ops, onlyNativeDeviceTypes) +from torch.testing._internal.common_methods_invocations import ( + spectral_funcs, SpectralFuncType) +from torch.testing._internal.common_utils import run_tests + +try: + from .xpu_test_utils import XPUPatchForImport +except Exception as e: + from ..xpu_test_utils import XPUPatchForImport + +with XPUPatchForImport(False): + from test_spectral_ops import TestFFT + +has_scipy_fft = False +try: + import scipy.fft + has_scipy_fft = True +except ModuleNotFoundError: + pass + +REFERENCE_NORM_MODES = ( + (None, "forward", "backward", "ortho") + if version.parse(np.__version__) >= version.parse('1.20.0') and ( + not has_scipy_fft or version.parse(scipy.__version__) >= version.parse('1.6.0')) + else (None, "ortho")) + +@ops([op for op in spectral_funcs if op.ndimensional == SpectralFuncType.OneD], + allowed_dtypes=(torch.float, torch.cfloat)) +def _test_reference_1d(self, device, dtype, op): + if op.ref is None: + raise unittest.SkipTest("No reference implementation") + + norm_modes = REFERENCE_NORM_MODES + test_args = [ + *product( + # input + (torch.randn(67, device=device, dtype=dtype), + torch.randn(80, device=device, dtype=dtype), + torch.randn(12, 14, device=device, dtype=dtype), + torch.randn(9, 6, 3, device=device, dtype=dtype)), + # n + (None, 50, 6), + # dim + (-1, 0), + # norm + norm_modes + ), + # Test transforming middle dimensions of multi-dim tensor + *product( + (torch.randn(4, 5, 6, 7, device=device, dtype=dtype),), + (None,), + (1, 2, -2,), + norm_modes + ) + ] + + for iargs in test_args: + args = list(iargs) + input = args[0] + args = args[1:] + + expected = op.ref(input.cpu().numpy(), *args) + exact_dtype = dtype in (torch.double, torch.complex128) + actual = op(input, *args) + self.assertEqual(actual, expected, exact_dtype=exact_dtype, atol=1e-4, rtol=1e-5) + +TestFFT.test_reference_1d = _test_reference_1d + +instantiate_device_type_tests(TestFFT, globals(), only_for=("xpu"), allow_xpu=True) + + +if __name__ == "__main__": + run_tests() diff --git a/test/xpu/xpu_test_utils.py b/test/xpu/xpu_test_utils.py index 56d8ce9aa..aa294282f 100644 --- a/test/xpu/xpu_test_utils.py +++ b/test/xpu/xpu_test_utils.py @@ -31,7 +31,6 @@ _xpu_computation_op_list = [ "empty", "eye", - "fill", "zeros", "zeros_like", "clone", @@ -72,6 +71,26 @@ "exp2", "expm1", "exponential", + "fft.fft", + "fft.fft2", + "fft.fftn", + "fft.hfft", + "fft.hfft2", + "fft.hfftn", + "fft.rfft", + "fft.rfft2", + "fft.rfftn", + "fft.ifft", + "fft.ifft2", + "fft.ifftn", + "fft.ihfft", + "fft.ihfft2", + "fft.ihfftn", + "fft.irfft", + "fft.irfft2", + "fft.irfftn", + "fft.fftshift", + "fft.ifftshift", "fill", "fmod", "gcd", diff --git a/yaml/native/native_functions.yaml b/yaml/native/native_functions.yaml index 545bced73..dffed34e1 100644 --- a/yaml/native/native_functions.yaml +++ b/yaml/native/native_functions.yaml @@ -6534,3 +6534,14 @@ - func: index_copy.dimname(Tensor self, Dimname dim, Tensor index, Tensor source) -> Tensor variants: function, method + +# Standard complex to complex FFT (forward or backward) +- func: _fft_c2c(Tensor self, SymInt[] dim, int normalization, bool forward) -> Tensor + variants: function + dispatch: + XPU: _fft_c2c_xpu + +- func: _fft_c2c.out(Tensor self, SymInt[] dim, int normalization, bool forward, *, Tensor(a!) out) -> Tensor(a!) + variants: function + dispatch: + XPU: _fft_c2c_xpu_out