xtensor
Loading...
Searching...
No Matches
xvectorize.hpp
1/***************************************************************************
2 * Copyright (c) Johan Mabille, Sylvain Corlay and Wolf Vollprecht *
3 * Copyright (c) QuantStack *
4 * *
5 * Distributed under the terms of the BSD 3-Clause License. *
6 * *
7 * The full license is in the file LICENSE, distributed with this software. *
8 ****************************************************************************/
9
10#ifndef XTENSOR_VECTORIZE_HPP
11#define XTENSOR_VECTORIZE_HPP
12
13#include <type_traits>
14#include <utility>
15
16#include "xfunction.hpp"
17#include "xutils.hpp"
18
19namespace xt
20{
21
22 /***************
23 * xvectorizer *
24 ***************/
25
26 template <class F, class R>
28 {
29 public:
30
31 template <class... E>
33
34 template <class Func, class = std::enable_if_t<!std::is_same<std::decay_t<Func>, xvectorizer>::value>>
36
37 template <class... E>
38 xfunction_type<E...> operator()(E&&... e) const;
39
40 private:
41
42 typename std::remove_reference<F>::type m_f;
43 };
44
45 namespace detail
46 {
47 template <class F>
48 using get_function_type = remove_class_t<decltype(&std::remove_reference_t<F>::operator())>;
49 }
50
51 template <class R, class... Args>
52 xvectorizer<R (*)(Args...), R> vectorize(R (*f)(Args...));
53
54 template <class F, class R, class... Args>
55 xvectorizer<F, R> vectorize(F&& f, R (*)(Args...));
56
57// Workaround for Visual Studio 15.7.1.
58// Error C2668 (ambiguous call to overloaded function) mistaking a declarations
59// for the definition of another overload.
60#ifndef _MSC_VER
61 template <class F>
62 auto vectorize(F&& f)
63 -> decltype(vectorize(std::forward<F>(f), std::declval<detail::get_function_type<F>*>()));
64#endif
65
66 /******************************
67 * xvectorizer implementation *
68 ******************************/
69
70 template <class F, class R>
71 template <class Func, class>
72 inline xvectorizer<F, R>::xvectorizer(Func&& f)
73 : m_f(std::forward<Func>(f))
74 {
75 }
76
77 template <class F, class R>
78 template <class... E>
79 inline auto xvectorizer<F, R>::operator()(E&&... e) const -> xfunction_type<E...>
80 {
81 return xfunction_type<E...>(m_f, std::forward<E>(e)...);
82 }
83
84 template <class R, class... Args>
85 inline xvectorizer<R (*)(Args...), R> vectorize(R (*f)(Args...))
86 {
87 return xvectorizer<R (*)(Args...), R>(f);
88 }
89
90 template <class F, class R, class... Args>
91 inline xvectorizer<F, R> vectorize(F&& f, R (*)(Args...))
92 {
93 return xvectorizer<F, R>(std::forward<F>(f));
94 }
95
96 template <class F>
97 inline auto vectorize(F&& f)
98 -> decltype(vectorize(std::forward<F>(f), std::declval<detail::get_function_type<F>*>()))
99 {
100 return vectorize(std::forward<F>(f), static_cast<detail::get_function_type<F>*>(nullptr));
101 }
102}
103
104#endif
standard mathematical functions for xexpressions