blob: 812e5f909e11a1c9839e286c5ebb874f573c6946 [file] [log] [blame]
Eugene Zelenko06becf82017-11-01 23:09:49 +00001//===- Marshallers.h - Generic matcher function marshallers -----*- C++ -*-===//
Manuel Klimek24db0f02013-05-14 09:13:00 +00002//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Manuel Klimek24db0f02013-05-14 09:13:00 +00006//
7//===----------------------------------------------------------------------===//
Eugene Zelenko06becf82017-11-01 23:09:49 +00008//
Manuel Klimek24db0f02013-05-14 09:13:00 +00009/// \file
Adrian Prantl9fc8faf2018-05-09 01:00:01 +000010/// Functions templates and classes to wrap matcher construct functions.
Manuel Klimek24db0f02013-05-14 09:13:00 +000011///
12/// A collection of template function and classes that provide a generic
13/// marshalling layer on top of matcher construct functions.
14/// These are used by the registry to export all marshaller constructors with
15/// the same generic interface.
Eugene Zelenko06becf82017-11-01 23:09:49 +000016//
Manuel Klimek24db0f02013-05-14 09:13:00 +000017//===----------------------------------------------------------------------===//
18
Benjamin Kramer2f5db8b2014-08-13 16:25:19 +000019#ifndef LLVM_CLANG_LIB_ASTMATCHERS_DYNAMIC_MARSHALLERS_H
20#define LLVM_CLANG_LIB_ASTMATCHERS_DYNAMIC_MARSHALLERS_H
Manuel Klimek24db0f02013-05-14 09:13:00 +000021
Eugene Zelenko06becf82017-11-01 23:09:49 +000022#include "clang/AST/ASTTypeTraits.h"
23#include "clang/AST/OperationKinds.h"
24#include "clang/ASTMatchers/ASTMatchersInternal.h"
Manuel Klimek24db0f02013-05-14 09:13:00 +000025#include "clang/ASTMatchers/Dynamic/Diagnostics.h"
26#include "clang/ASTMatchers/Dynamic/VariantValue.h"
Eugene Zelenko06becf82017-11-01 23:09:49 +000027#include "clang/Basic/AttrKinds.h"
Manuel Klimek24db0f02013-05-14 09:13:00 +000028#include "clang/Basic/LLVM.h"
Eugene Zelenko06becf82017-11-01 23:09:49 +000029#include "llvm/ADT/ArrayRef.h"
30#include "llvm/ADT/None.h"
Samuel Benzaquen0239b692013-08-13 14:54:51 +000031#include "llvm/ADT/STLExtras.h"
Eugene Zelenko06becf82017-11-01 23:09:49 +000032#include "llvm/ADT/StringRef.h"
33#include "llvm/ADT/StringSwitch.h"
34#include "llvm/ADT/Twine.h"
35#include <cassert>
36#include <cstddef>
37#include <iterator>
38#include <limits>
39#include <memory>
Chandler Carruth5553d0d2014-01-07 11:51:46 +000040#include <string>
Eugene Zelenko06becf82017-11-01 23:09:49 +000041#include <utility>
42#include <vector>
Manuel Klimek24db0f02013-05-14 09:13:00 +000043
44namespace clang {
45namespace ast_matchers {
46namespace dynamic {
Manuel Klimek24db0f02013-05-14 09:13:00 +000047namespace internal {
48
Adrian Prantl9fc8faf2018-05-09 01:00:01 +000049/// Helper template class to just from argument type to the right is/get
Manuel Klimek24db0f02013-05-14 09:13:00 +000050/// functions in VariantValue.
51/// Used to verify and extract the matcher arguments below.
52template <class T> struct ArgTypeTraits;
53template <class T> struct ArgTypeTraits<const T &> : public ArgTypeTraits<T> {
54};
55
56template <> struct ArgTypeTraits<std::string> {
57 static bool is(const VariantValue &Value) { return Value.isString(); }
Eugene Zelenko06becf82017-11-01 23:09:49 +000058
Manuel Klimek24db0f02013-05-14 09:13:00 +000059 static const std::string &get(const VariantValue &Value) {
60 return Value.getString();
61 }
Eugene Zelenko06becf82017-11-01 23:09:49 +000062
Peter Collingbourned32e28c2014-01-23 22:48:38 +000063 static ArgKind getKind() {
64 return ArgKind(ArgKind::AK_String);
65 }
Manuel Klimek24db0f02013-05-14 09:13:00 +000066};
67
Samuel Benzaquenc6f2c9b2013-06-21 15:51:31 +000068template <>
69struct ArgTypeTraits<StringRef> : public ArgTypeTraits<std::string> {
70};
71
Eugene Zelenko06becf82017-11-01 23:09:49 +000072template <class T> struct ArgTypeTraits<ast_matchers::internal::Matcher<T>> {
Samuel Benzaquen81ef9292013-06-20 14:28:32 +000073 static bool is(const VariantValue &Value) {
Samuel Benzaquen998cda232013-08-30 15:09:52 +000074 return Value.isMatcher() && Value.getMatcher().hasTypedMatcher<T>();
Samuel Benzaquen81ef9292013-06-20 14:28:32 +000075 }
Eugene Zelenko06becf82017-11-01 23:09:49 +000076
Manuel Klimek24db0f02013-05-14 09:13:00 +000077 static ast_matchers::internal::Matcher<T> get(const VariantValue &Value) {
Samuel Benzaquen998cda232013-08-30 15:09:52 +000078 return Value.getMatcher().getTypedMatcher<T>();
Manuel Klimek24db0f02013-05-14 09:13:00 +000079 }
Eugene Zelenko06becf82017-11-01 23:09:49 +000080
Peter Collingbourned32e28c2014-01-23 22:48:38 +000081 static ArgKind getKind() {
82 return ArgKind(ast_type_traits::ASTNodeKind::getFromNodeKind<T>());
83 }
Samuel Benzaquenc31b3522013-06-04 15:46:22 +000084};
Manuel Klimek24db0f02013-05-14 09:13:00 +000085
Peter Wuc04b1982017-06-08 22:00:38 +000086template <> struct ArgTypeTraits<bool> {
87 static bool is(const VariantValue &Value) { return Value.isBoolean(); }
Eugene Zelenko06becf82017-11-01 23:09:49 +000088
Peter Wuc04b1982017-06-08 22:00:38 +000089 static bool get(const VariantValue &Value) {
90 return Value.getBoolean();
91 }
Eugene Zelenko06becf82017-11-01 23:09:49 +000092
Peter Wuc04b1982017-06-08 22:00:38 +000093 static ArgKind getKind() {
94 return ArgKind(ArgKind::AK_Boolean);
95 }
96};
97
Peter Wu2bbed502017-06-08 22:00:50 +000098template <> struct ArgTypeTraits<double> {
99 static bool is(const VariantValue &Value) { return Value.isDouble(); }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000100
Peter Wu2bbed502017-06-08 22:00:50 +0000101 static double get(const VariantValue &Value) {
102 return Value.getDouble();
103 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000104
Peter Wu2bbed502017-06-08 22:00:50 +0000105 static ArgKind getKind() {
106 return ArgKind(ArgKind::AK_Double);
107 }
108};
109
Samuel Benzaquenc31b3522013-06-04 15:46:22 +0000110template <> struct ArgTypeTraits<unsigned> {
111 static bool is(const VariantValue &Value) { return Value.isUnsigned(); }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000112
Samuel Benzaquenc31b3522013-06-04 15:46:22 +0000113 static unsigned get(const VariantValue &Value) {
114 return Value.getUnsigned();
115 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000116
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000117 static ArgKind getKind() {
118 return ArgKind(ArgKind::AK_Unsigned);
119 }
Manuel Klimek24db0f02013-05-14 09:13:00 +0000120};
121
Manuel Klimek3fe8a382014-08-25 11:23:50 +0000122template <> struct ArgTypeTraits<attr::Kind> {
123private:
Richard Smith354abec2017-12-08 23:29:59 +0000124 static Optional<attr::Kind> getAttrKind(llvm::StringRef AttrKind) {
125 return llvm::StringSwitch<Optional<attr::Kind>>(AttrKind)
Manuel Klimek3fe8a382014-08-25 11:23:50 +0000126#define ATTR(X) .Case("attr::" #X, attr:: X)
127#include "clang/Basic/AttrList.inc"
Richard Smith354abec2017-12-08 23:29:59 +0000128 .Default(llvm::None);
Manuel Klimek3fe8a382014-08-25 11:23:50 +0000129 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000130
Manuel Klimek3fe8a382014-08-25 11:23:50 +0000131public:
132 static bool is(const VariantValue &Value) {
Richard Smith354abec2017-12-08 23:29:59 +0000133 return Value.isString() && getAttrKind(Value.getString());
Manuel Klimek3fe8a382014-08-25 11:23:50 +0000134 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000135
Manuel Klimek3fe8a382014-08-25 11:23:50 +0000136 static attr::Kind get(const VariantValue &Value) {
Richard Smith354abec2017-12-08 23:29:59 +0000137 return *getAttrKind(Value.getString());
Manuel Klimek3fe8a382014-08-25 11:23:50 +0000138 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000139
Manuel Klimek3fe8a382014-08-25 11:23:50 +0000140 static ArgKind getKind() {
141 return ArgKind(ArgKind::AK_String);
142 }
143};
144
Eugene Zelenko06becf82017-11-01 23:09:49 +0000145template <> struct ArgTypeTraits<CastKind> {
Etienne Bergeron75e52722016-05-13 19:36:55 +0000146private:
Richard Smith354abec2017-12-08 23:29:59 +0000147 static Optional<CastKind> getCastKind(llvm::StringRef AttrKind) {
148 return llvm::StringSwitch<Optional<CastKind>>(AttrKind)
Etienne Bergeron75e52722016-05-13 19:36:55 +0000149#define CAST_OPERATION(Name) .Case( #Name, CK_##Name)
150#include "clang/AST/OperationKinds.def"
Richard Smith354abec2017-12-08 23:29:59 +0000151 .Default(llvm::None);
Etienne Bergeron75e52722016-05-13 19:36:55 +0000152 }
153
154public:
155 static bool is(const VariantValue &Value) {
Richard Smith354abec2017-12-08 23:29:59 +0000156 return Value.isString() && getCastKind(Value.getString());
Etienne Bergeron75e52722016-05-13 19:36:55 +0000157 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000158
159 static CastKind get(const VariantValue &Value) {
Richard Smith354abec2017-12-08 23:29:59 +0000160 return *getCastKind(Value.getString());
Etienne Bergeron75e52722016-05-13 19:36:55 +0000161 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000162
Etienne Bergeron75e52722016-05-13 19:36:55 +0000163 static ArgKind getKind() {
164 return ArgKind(ArgKind::AK_String);
165 }
166};
167
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000168/// Matcher descriptor interface.
Manuel Klimek24db0f02013-05-14 09:13:00 +0000169///
Peter Collingbournef43e6942013-11-23 01:34:36 +0000170/// Provides a \c create() method that constructs the matcher from the provided
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000171/// arguments, and various other methods for type introspection.
Peter Collingbournef43e6942013-11-23 01:34:36 +0000172class MatcherDescriptor {
Manuel Klimek24db0f02013-05-14 09:13:00 +0000173public:
Eugene Zelenko06becf82017-11-01 23:09:49 +0000174 virtual ~MatcherDescriptor() = default;
175
Craig Toppere335f252015-10-04 04:53:55 +0000176 virtual VariantMatcher create(SourceRange NameRange,
Peter Collingbournef43e6942013-11-23 01:34:36 +0000177 ArrayRef<ParserValue> Args,
178 Diagnostics *Error) const = 0;
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000179
180 /// Returns whether the matcher is variadic. Variadic matchers can take any
181 /// number of arguments, but they must be of the same type.
182 virtual bool isVariadic() const = 0;
183
184 /// Returns the number of arguments accepted by the matcher if not variadic.
185 virtual unsigned getNumArgs() const = 0;
186
187 /// Given that the matcher is being converted to type \p ThisKind, append the
188 /// set of argument types accepted for argument \p ArgNo to \p ArgKinds.
189 // FIXME: We should provide the ability to constrain the output of this
190 // function based on the types of other matcher arguments.
191 virtual void getArgKinds(ast_type_traits::ASTNodeKind ThisKind, unsigned ArgNo,
192 std::vector<ArgKind> &ArgKinds) const = 0;
193
194 /// Returns whether this matcher is convertible to the given type. If it is
195 /// so convertible, store in *Specificity a value corresponding to the
196 /// "specificity" of the converted matcher to the given context, and in
197 /// *LeastDerivedKind the least derived matcher kind which would result in the
198 /// same matcher overload. Zero specificity indicates that this conversion
199 /// would produce a trivial matcher that will either always or never match.
200 /// Such matchers are excluded from code completion results.
201 virtual bool isConvertibleTo(
Craig Topper210e1ad2014-05-17 18:49:24 +0000202 ast_type_traits::ASTNodeKind Kind, unsigned *Specificity = nullptr,
203 ast_type_traits::ASTNodeKind *LeastDerivedKind = nullptr) const = 0;
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000204
205 /// Returns whether the matcher will, given a matcher of any type T, yield a
206 /// matcher of type T.
207 virtual bool isPolymorphic() const { return false; }
Manuel Klimek24db0f02013-05-14 09:13:00 +0000208};
209
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000210inline bool isRetKindConvertibleTo(
Craig Topper00bbdcf2014-06-28 23:22:23 +0000211 ArrayRef<ast_type_traits::ASTNodeKind> RetKinds,
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000212 ast_type_traits::ASTNodeKind Kind, unsigned *Specificity,
213 ast_type_traits::ASTNodeKind *LeastDerivedKind) {
Samuel Benzaquen646f23b2014-08-12 21:11:37 +0000214 for (const ast_type_traits::ASTNodeKind &NodeKind : RetKinds) {
215 if (ArgKind(NodeKind).isConvertibleTo(Kind, Specificity)) {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000216 if (LeastDerivedKind)
Samuel Benzaquen646f23b2014-08-12 21:11:37 +0000217 *LeastDerivedKind = NodeKind;
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000218 return true;
219 }
220 }
221 return false;
222}
223
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000224/// Simple callback implementation. Marshaller and function are provided.
Samuel Benzaquenb5dd69f2013-06-11 18:51:07 +0000225///
226/// This class wraps a function of arbitrary signature and a marshaller
Peter Collingbournef43e6942013-11-23 01:34:36 +0000227/// function into a MatcherDescriptor.
Samuel Benzaquenb5dd69f2013-06-11 18:51:07 +0000228/// The marshaller is in charge of taking the VariantValue arguments, checking
229/// their types, unpacking them and calling the underlying function.
Peter Collingbournef43e6942013-11-23 01:34:36 +0000230class FixedArgCountMatcherDescriptor : public MatcherDescriptor {
Manuel Klimek24db0f02013-05-14 09:13:00 +0000231public:
Eugene Zelenko06becf82017-11-01 23:09:49 +0000232 using MarshallerType = VariantMatcher (*)(void (*Func)(),
233 StringRef MatcherName,
234 SourceRange NameRange,
235 ArrayRef<ParserValue> Args,
236 Diagnostics *Error);
Samuel Benzaquenb5dd69f2013-06-11 18:51:07 +0000237
Dmitri Gribenkocb63baf2013-05-17 17:50:16 +0000238 /// \param Marshaller Function to unpack the arguments and call \c Func
239 /// \param Func Matcher construct function. This is the function that
240 /// compile-time matcher expressions would use to create the matcher.
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000241 /// \param RetKinds The list of matcher types to which the matcher is
242 /// convertible.
243 /// \param ArgKinds The types of the arguments this matcher takes.
244 FixedArgCountMatcherDescriptor(
245 MarshallerType Marshaller, void (*Func)(), StringRef MatcherName,
Craig Topper00bbdcf2014-06-28 23:22:23 +0000246 ArrayRef<ast_type_traits::ASTNodeKind> RetKinds,
247 ArrayRef<ArgKind> ArgKinds)
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000248 : Marshaller(Marshaller), Func(Func), MatcherName(MatcherName),
249 RetKinds(RetKinds.begin(), RetKinds.end()),
250 ArgKinds(ArgKinds.begin(), ArgKinds.end()) {}
Manuel Klimek24db0f02013-05-14 09:13:00 +0000251
Craig Toppere335f252015-10-04 04:53:55 +0000252 VariantMatcher create(SourceRange NameRange,
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000253 ArrayRef<ParserValue> Args,
254 Diagnostics *Error) const override {
Manuel Klimek24db0f02013-05-14 09:13:00 +0000255 return Marshaller(Func, MatcherName, NameRange, Args, Error);
256 }
257
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000258 bool isVariadic() const override { return false; }
259 unsigned getNumArgs() const override { return ArgKinds.size(); }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000260
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000261 void getArgKinds(ast_type_traits::ASTNodeKind ThisKind, unsigned ArgNo,
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000262 std::vector<ArgKind> &Kinds) const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000263 Kinds.push_back(ArgKinds[ArgNo]);
264 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000265
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000266 bool isConvertibleTo(
267 ast_type_traits::ASTNodeKind Kind, unsigned *Specificity,
268 ast_type_traits::ASTNodeKind *LeastDerivedKind) const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000269 return isRetKindConvertibleTo(RetKinds, Kind, Specificity,
270 LeastDerivedKind);
271 }
272
Manuel Klimek24db0f02013-05-14 09:13:00 +0000273private:
274 const MarshallerType Marshaller;
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000275 void (* const Func)();
Manuel Klimek24db0f02013-05-14 09:13:00 +0000276 const std::string MatcherName;
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000277 const std::vector<ast_type_traits::ASTNodeKind> RetKinds;
278 const std::vector<ArgKind> ArgKinds;
Manuel Klimek24db0f02013-05-14 09:13:00 +0000279};
280
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000281/// Helper methods to extract and merge all possible typed matchers
Samuel Benzaquenc6f2c9b2013-06-21 15:51:31 +0000282/// out of the polymorphic object.
283template <class PolyMatcher>
Samuel Benzaquen0239b692013-08-13 14:54:51 +0000284static void mergePolyMatchers(const PolyMatcher &Poly,
Samuel Benzaquenf34ac3e2013-10-29 14:37:15 +0000285 std::vector<DynTypedMatcher> &Out,
Samuel Benzaquenc6f2c9b2013-06-21 15:51:31 +0000286 ast_matchers::internal::EmptyTypeList) {}
287
288template <class PolyMatcher, class TypeList>
Samuel Benzaquen0239b692013-08-13 14:54:51 +0000289static void mergePolyMatchers(const PolyMatcher &Poly,
Samuel Benzaquenf34ac3e2013-10-29 14:37:15 +0000290 std::vector<DynTypedMatcher> &Out, TypeList) {
291 Out.push_back(ast_matchers::internal::Matcher<typename TypeList::head>(Poly));
Samuel Benzaquenc6f2c9b2013-06-21 15:51:31 +0000292 mergePolyMatchers(Poly, Out, typename TypeList::tail());
293}
294
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000295/// Convert the return values of the functions into a VariantMatcher.
Samuel Benzaquenc6f2c9b2013-06-21 15:51:31 +0000296///
297/// There are 2 cases right now: The return value is a Matcher<T> or is a
Samuel Benzaquen0239b692013-08-13 14:54:51 +0000298/// polymorphic matcher. For the former, we just construct the VariantMatcher.
299/// For the latter, we instantiate all the possible Matcher<T> of the poly
300/// matcher.
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000301static VariantMatcher outvalueToVariantMatcher(const DynTypedMatcher &Matcher) {
Samuel Benzaquen0239b692013-08-13 14:54:51 +0000302 return VariantMatcher::SingleMatcher(Matcher);
Samuel Benzaquenc6f2c9b2013-06-21 15:51:31 +0000303}
304
305template <typename T>
Samuel Benzaquen0239b692013-08-13 14:54:51 +0000306static VariantMatcher outvalueToVariantMatcher(const T &PolyMatcher,
307 typename T::ReturnTypes * =
Eugene Zelenko1660a5d2016-01-26 19:01:06 +0000308 nullptr) {
Samuel Benzaquenf34ac3e2013-10-29 14:37:15 +0000309 std::vector<DynTypedMatcher> Matchers;
Samuel Benzaquen0239b692013-08-13 14:54:51 +0000310 mergePolyMatchers(PolyMatcher, Matchers, typename T::ReturnTypes());
Benjamin Kramer3f755aa2014-03-10 17:55:02 +0000311 VariantMatcher Out = VariantMatcher::PolymorphicMatcher(std::move(Matchers));
Samuel Benzaquen0239b692013-08-13 14:54:51 +0000312 return Out;
Samuel Benzaquenc6f2c9b2013-06-21 15:51:31 +0000313}
314
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000315template <typename T>
316inline void buildReturnTypeVectorFromTypeList(
317 std::vector<ast_type_traits::ASTNodeKind> &RetTypes) {
318 RetTypes.push_back(
319 ast_type_traits::ASTNodeKind::getFromNodeKind<typename T::head>());
320 buildReturnTypeVectorFromTypeList<typename T::tail>(RetTypes);
321}
322
323template <>
324inline void
325buildReturnTypeVectorFromTypeList<ast_matchers::internal::EmptyTypeList>(
326 std::vector<ast_type_traits::ASTNodeKind> &RetTypes) {}
327
328template <typename T>
329struct BuildReturnTypeVector {
330 static void build(std::vector<ast_type_traits::ASTNodeKind> &RetTypes) {
331 buildReturnTypeVectorFromTypeList<typename T::ReturnTypes>(RetTypes);
332 }
333};
334
335template <typename T>
Eugene Zelenko06becf82017-11-01 23:09:49 +0000336struct BuildReturnTypeVector<ast_matchers::internal::Matcher<T>> {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000337 static void build(std::vector<ast_type_traits::ASTNodeKind> &RetTypes) {
338 RetTypes.push_back(ast_type_traits::ASTNodeKind::getFromNodeKind<T>());
339 }
340};
341
342template <typename T>
Eugene Zelenko06becf82017-11-01 23:09:49 +0000343struct BuildReturnTypeVector<ast_matchers::internal::BindableMatcher<T>> {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000344 static void build(std::vector<ast_type_traits::ASTNodeKind> &RetTypes) {
345 RetTypes.push_back(ast_type_traits::ASTNodeKind::getFromNodeKind<T>());
346 }
347};
348
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000349/// Variadic marshaller function.
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000350template <typename ResultT, typename ArgT,
351 ResultT (*Func)(ArrayRef<const ArgT *>)>
352VariantMatcher
Craig Toppere335f252015-10-04 04:53:55 +0000353variadicMatcherDescriptor(StringRef MatcherName, SourceRange NameRange,
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000354 ArrayRef<ParserValue> Args, Diagnostics *Error) {
355 ArgT **InnerArgs = new ArgT *[Args.size()]();
356
357 bool HasError = false;
358 for (size_t i = 0, e = Args.size(); i != e; ++i) {
Eugene Zelenko06becf82017-11-01 23:09:49 +0000359 using ArgTraits = ArgTypeTraits<ArgT>;
360
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000361 const ParserValue &Arg = Args[i];
362 const VariantValue &Value = Arg.Value;
363 if (!ArgTraits::is(Value)) {
364 Error->addError(Arg.Range, Error->ET_RegistryWrongArgType)
365 << (i + 1) << ArgTraits::getKind().asString() << Value.getTypeAsString();
366 HasError = true;
367 break;
368 }
369 InnerArgs[i] = new ArgT(ArgTraits::get(Value));
370 }
371
372 VariantMatcher Out;
373 if (!HasError) {
Craig Topper8c2a2a02014-08-30 16:55:39 +0000374 Out = outvalueToVariantMatcher(Func(llvm::makeArrayRef(InnerArgs,
375 Args.size())));
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000376 }
377
378 for (size_t i = 0, e = Args.size(); i != e; ++i) {
379 delete InnerArgs[i];
380 }
381 delete[] InnerArgs;
382 return Out;
383}
384
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000385/// Matcher descriptor for variadic functions.
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000386///
387/// This class simply wraps a VariadicFunction with the right signature to export
388/// it as a MatcherDescriptor.
389/// This allows us to have one implementation of the interface for as many free
390/// functions as we want, reducing the number of symbols and size of the
391/// object file.
392class VariadicFuncMatcherDescriptor : public MatcherDescriptor {
393public:
Eugene Zelenko06becf82017-11-01 23:09:49 +0000394 using RunFunc = VariantMatcher (*)(StringRef MatcherName,
395 SourceRange NameRange,
396 ArrayRef<ParserValue> Args,
397 Diagnostics *Error);
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000398
399 template <typename ResultT, typename ArgT,
400 ResultT (*F)(ArrayRef<const ArgT *>)>
Samuel Benzaquenc1384c12016-03-25 16:29:30 +0000401 VariadicFuncMatcherDescriptor(
402 ast_matchers::internal::VariadicFunction<ResultT, ArgT, F> Func,
403 StringRef MatcherName)
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000404 : Func(&variadicMatcherDescriptor<ResultT, ArgT, F>),
405 MatcherName(MatcherName.str()),
406 ArgsKind(ArgTypeTraits<ArgT>::getKind()) {
407 BuildReturnTypeVector<ResultT>::build(RetKinds);
408 }
409
Craig Toppere335f252015-10-04 04:53:55 +0000410 VariantMatcher create(SourceRange NameRange,
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000411 ArrayRef<ParserValue> Args,
412 Diagnostics *Error) const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000413 return Func(MatcherName, NameRange, Args, Error);
414 }
415
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000416 bool isVariadic() const override { return true; }
417 unsigned getNumArgs() const override { return 0; }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000418
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000419 void getArgKinds(ast_type_traits::ASTNodeKind ThisKind, unsigned ArgNo,
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000420 std::vector<ArgKind> &Kinds) const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000421 Kinds.push_back(ArgsKind);
422 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000423
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000424 bool isConvertibleTo(
425 ast_type_traits::ASTNodeKind Kind, unsigned *Specificity,
426 ast_type_traits::ASTNodeKind *LeastDerivedKind) const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000427 return isRetKindConvertibleTo(RetKinds, Kind, Specificity,
428 LeastDerivedKind);
429 }
430
431private:
432 const RunFunc Func;
433 const std::string MatcherName;
434 std::vector<ast_type_traits::ASTNodeKind> RetKinds;
435 const ArgKind ArgsKind;
436};
437
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000438/// Return CK_Trivial when appropriate for VariadicDynCastAllOfMatchers.
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000439class DynCastAllOfMatcherDescriptor : public VariadicFuncMatcherDescriptor {
440public:
441 template <typename BaseT, typename DerivedT>
442 DynCastAllOfMatcherDescriptor(
443 ast_matchers::internal::VariadicDynCastAllOfMatcher<BaseT, DerivedT> Func,
444 StringRef MatcherName)
445 : VariadicFuncMatcherDescriptor(Func, MatcherName),
446 DerivedKind(ast_type_traits::ASTNodeKind::getFromNodeKind<DerivedT>()) {
447 }
448
Craig Toppera798a9d2014-03-02 09:32:10 +0000449 bool
450 isConvertibleTo(ast_type_traits::ASTNodeKind Kind, unsigned *Specificity,
451 ast_type_traits::ASTNodeKind *LeastDerivedKind) const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000452 // If Kind is not a base of DerivedKind, either DerivedKind is a base of
453 // Kind (in which case the match will always succeed) or Kind and
454 // DerivedKind are unrelated (in which case it will always fail), so set
455 // Specificity to 0.
456 if (VariadicFuncMatcherDescriptor::isConvertibleTo(Kind, Specificity,
457 LeastDerivedKind)) {
458 if (Kind.isSame(DerivedKind) || !Kind.isBaseOf(DerivedKind)) {
459 if (Specificity)
460 *Specificity = 0;
461 }
462 return true;
463 } else {
464 return false;
465 }
466 }
467
468private:
469 const ast_type_traits::ASTNodeKind DerivedKind;
470};
471
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000472/// Helper macros to check the arguments on all marshaller functions.
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000473#define CHECK_ARG_COUNT(count) \
474 if (Args.size() != count) { \
475 Error->addError(NameRange, Error->ET_RegistryWrongArgCount) \
476 << count << Args.size(); \
477 return VariantMatcher(); \
478 }
479
480#define CHECK_ARG_TYPE(index, type) \
481 if (!ArgTypeTraits<type>::is(Args[index].Value)) { \
482 Error->addError(Args[index].Range, Error->ET_RegistryWrongArgType) \
483 << (index + 1) << ArgTypeTraits<type>::getKind().asString() \
484 << Args[index].Value.getTypeAsString(); \
485 return VariantMatcher(); \
486 }
487
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000488/// 0-arg marshaller function.
Manuel Klimek24db0f02013-05-14 09:13:00 +0000489template <typename ReturnType>
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000490static VariantMatcher matcherMarshall0(void (*Func)(), StringRef MatcherName,
Craig Toppere335f252015-10-04 04:53:55 +0000491 SourceRange NameRange,
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000492 ArrayRef<ParserValue> Args,
493 Diagnostics *Error) {
Eugene Zelenko06becf82017-11-01 23:09:49 +0000494 using FuncType = ReturnType (*)();
Manuel Klimek24db0f02013-05-14 09:13:00 +0000495 CHECK_ARG_COUNT(0);
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000496 return outvalueToVariantMatcher(reinterpret_cast<FuncType>(Func)());
Manuel Klimek24db0f02013-05-14 09:13:00 +0000497}
498
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000499/// 1-arg marshaller function.
Samuel Benzaquenc31b3522013-06-04 15:46:22 +0000500template <typename ReturnType, typename ArgType1>
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000501static VariantMatcher matcherMarshall1(void (*Func)(), StringRef MatcherName,
Craig Toppere335f252015-10-04 04:53:55 +0000502 SourceRange NameRange,
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000503 ArrayRef<ParserValue> Args,
504 Diagnostics *Error) {
Eugene Zelenko06becf82017-11-01 23:09:49 +0000505 using FuncType = ReturnType (*)(ArgType1);
Manuel Klimek24db0f02013-05-14 09:13:00 +0000506 CHECK_ARG_COUNT(1);
507 CHECK_ARG_TYPE(0, ArgType1);
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000508 return outvalueToVariantMatcher(reinterpret_cast<FuncType>(Func)(
509 ArgTypeTraits<ArgType1>::get(Args[0].Value)));
Manuel Klimek24db0f02013-05-14 09:13:00 +0000510}
511
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000512/// 2-arg marshaller function.
Samuel Benzaquenc31b3522013-06-04 15:46:22 +0000513template <typename ReturnType, typename ArgType1, typename ArgType2>
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000514static VariantMatcher matcherMarshall2(void (*Func)(), StringRef MatcherName,
Craig Toppere335f252015-10-04 04:53:55 +0000515 SourceRange NameRange,
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000516 ArrayRef<ParserValue> Args,
517 Diagnostics *Error) {
Eugene Zelenko06becf82017-11-01 23:09:49 +0000518 using FuncType = ReturnType (*)(ArgType1, ArgType2);
Samuel Benzaquenc31b3522013-06-04 15:46:22 +0000519 CHECK_ARG_COUNT(2);
520 CHECK_ARG_TYPE(0, ArgType1);
521 CHECK_ARG_TYPE(1, ArgType2);
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000522 return outvalueToVariantMatcher(reinterpret_cast<FuncType>(Func)(
523 ArgTypeTraits<ArgType1>::get(Args[0].Value),
524 ArgTypeTraits<ArgType2>::get(Args[1].Value)));
Samuel Benzaquenc31b3522013-06-04 15:46:22 +0000525}
526
Manuel Klimek24db0f02013-05-14 09:13:00 +0000527#undef CHECK_ARG_COUNT
528#undef CHECK_ARG_TYPE
529
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000530/// Helper class used to collect all the possible overloads of an
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000531/// argument adaptative matcher function.
532template <template <typename ToArg, typename FromArg> class ArgumentAdapterT,
533 typename FromTypes, typename ToTypes>
534class AdaptativeOverloadCollector {
535public:
Justin Lebar82380d82016-10-10 16:26:40 +0000536 AdaptativeOverloadCollector(
537 StringRef Name, std::vector<std::unique_ptr<MatcherDescriptor>> &Out)
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000538 : Name(Name), Out(Out) {
539 collect(FromTypes());
540 }
541
542private:
Eugene Zelenko06becf82017-11-01 23:09:49 +0000543 using AdaptativeFunc = ast_matchers::internal::ArgumentAdaptingMatcherFunc<
544 ArgumentAdapterT, FromTypes, ToTypes>;
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000545
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000546 /// End case for the recursion
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000547 static void collect(ast_matchers::internal::EmptyTypeList) {}
548
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000549 /// Recursive case. Get the overload for the head of the list, and
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000550 /// recurse to the tail.
Peter Collingbournef43e6942013-11-23 01:34:36 +0000551 template <typename FromTypeList>
552 inline void collect(FromTypeList);
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000553
Craig Topperbf3e3272014-08-30 16:55:52 +0000554 StringRef Name;
Justin Lebar82380d82016-10-10 16:26:40 +0000555 std::vector<std::unique_ptr<MatcherDescriptor>> &Out;
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000556};
557
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000558/// MatcherDescriptor that wraps multiple "overloads" of the same
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000559/// matcher.
560///
561/// It will try every overload and generate appropriate errors for when none or
562/// more than one overloads match the arguments.
Peter Collingbournef43e6942013-11-23 01:34:36 +0000563class OverloadedMatcherDescriptor : public MatcherDescriptor {
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000564public:
Justin Lebar82380d82016-10-10 16:26:40 +0000565 OverloadedMatcherDescriptor(
566 MutableArrayRef<std::unique_ptr<MatcherDescriptor>> Callbacks)
567 : Overloads(std::make_move_iterator(Callbacks.begin()),
568 std::make_move_iterator(Callbacks.end())) {}
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000569
Eugene Zelenko06becf82017-11-01 23:09:49 +0000570 ~OverloadedMatcherDescriptor() override = default;
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000571
Craig Toppere335f252015-10-04 04:53:55 +0000572 VariantMatcher create(SourceRange NameRange,
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000573 ArrayRef<ParserValue> Args,
574 Diagnostics *Error) const override {
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000575 std::vector<VariantMatcher> Constructed;
576 Diagnostics::OverloadContext Ctx(Error);
Ahmed Charlesbfb62322014-03-09 12:24:23 +0000577 for (const auto &O : Overloads) {
578 VariantMatcher SubMatcher = O->create(NameRange, Args, Error);
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000579 if (!SubMatcher.isNull()) {
580 Constructed.push_back(SubMatcher);
581 }
582 }
583
584 if (Constructed.empty()) return VariantMatcher(); // No overload matched.
585 // We ignore the errors if any matcher succeeded.
586 Ctx.revertErrors();
587 if (Constructed.size() > 1) {
588 // More than one constructed. It is ambiguous.
589 Error->addError(NameRange, Error->ET_RegistryAmbiguousOverload);
590 return VariantMatcher();
591 }
592 return Constructed[0];
593 }
594
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000595 bool isVariadic() const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000596 bool Overload0Variadic = Overloads[0]->isVariadic();
597#ifndef NDEBUG
Ahmed Charlesbfb62322014-03-09 12:24:23 +0000598 for (const auto &O : Overloads) {
599 assert(Overload0Variadic == O->isVariadic());
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000600 }
601#endif
602 return Overload0Variadic;
603 }
604
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000605 unsigned getNumArgs() const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000606 unsigned Overload0NumArgs = Overloads[0]->getNumArgs();
607#ifndef NDEBUG
Ahmed Charlesbfb62322014-03-09 12:24:23 +0000608 for (const auto &O : Overloads) {
609 assert(Overload0NumArgs == O->getNumArgs());
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000610 }
611#endif
612 return Overload0NumArgs;
613 }
614
615 void getArgKinds(ast_type_traits::ASTNodeKind ThisKind, unsigned ArgNo,
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000616 std::vector<ArgKind> &Kinds) const override {
Ahmed Charlesbfb62322014-03-09 12:24:23 +0000617 for (const auto &O : Overloads) {
618 if (O->isConvertibleTo(ThisKind))
619 O->getArgKinds(ThisKind, ArgNo, Kinds);
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000620 }
621 }
622
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000623 bool isConvertibleTo(
624 ast_type_traits::ASTNodeKind Kind, unsigned *Specificity,
625 ast_type_traits::ASTNodeKind *LeastDerivedKind) const override {
Ahmed Charlesbfb62322014-03-09 12:24:23 +0000626 for (const auto &O : Overloads) {
627 if (O->isConvertibleTo(Kind, Specificity, LeastDerivedKind))
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000628 return true;
629 }
630 return false;
631 }
632
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000633private:
Ahmed Charlesbfb62322014-03-09 12:24:23 +0000634 std::vector<std::unique_ptr<MatcherDescriptor>> Overloads;
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000635};
636
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000637/// Variadic operator marshaller function.
Peter Collingbournef43e6942013-11-23 01:34:36 +0000638class VariadicOperatorMatcherDescriptor : public MatcherDescriptor {
Samuel Benzaquen4adca622013-08-28 18:42:04 +0000639public:
Eugene Zelenko06becf82017-11-01 23:09:49 +0000640 using VarOp = DynTypedMatcher::VariadicOperator;
641
Peter Collingbournef43e6942013-11-23 01:34:36 +0000642 VariadicOperatorMatcherDescriptor(unsigned MinCount, unsigned MaxCount,
Samuel Benzaquen2c15e8c2014-11-20 15:45:53 +0000643 VarOp Op, StringRef MatcherName)
644 : MinCount(MinCount), MaxCount(MaxCount), Op(Op),
Samuel Benzaquen4d058742013-11-22 14:41:48 +0000645 MatcherName(MatcherName) {}
Samuel Benzaquen4adca622013-08-28 18:42:04 +0000646
Craig Toppere335f252015-10-04 04:53:55 +0000647 VariantMatcher create(SourceRange NameRange,
Alexander Kornienko34eb2072015-04-11 02:00:23 +0000648 ArrayRef<ParserValue> Args,
649 Diagnostics *Error) const override {
Samuel Benzaquen4d058742013-11-22 14:41:48 +0000650 if (Args.size() < MinCount || MaxCount < Args.size()) {
651 const std::string MaxStr =
Eugene Zelenko06becf82017-11-01 23:09:49 +0000652 (MaxCount == std::numeric_limits<unsigned>::max() ? ""
653 : Twine(MaxCount))
654 .str();
Samuel Benzaquen4d058742013-11-22 14:41:48 +0000655 Error->addError(NameRange, Error->ET_RegistryWrongArgCount)
656 << ("(" + Twine(MinCount) + ", " + MaxStr + ")") << Args.size();
657 return VariantMatcher();
658 }
659
Samuel Benzaquen4adca622013-08-28 18:42:04 +0000660 std::vector<VariantMatcher> InnerArgs;
661 for (size_t i = 0, e = Args.size(); i != e; ++i) {
662 const ParserValue &Arg = Args[i];
663 const VariantValue &Value = Arg.Value;
664 if (!Value.isMatcher()) {
665 Error->addError(Arg.Range, Error->ET_RegistryWrongArgType)
666 << (i + 1) << "Matcher<>" << Value.getTypeAsString();
667 return VariantMatcher();
668 }
669 InnerArgs.push_back(Value.getMatcher());
670 }
Samuel Benzaquen2c15e8c2014-11-20 15:45:53 +0000671 return VariantMatcher::VariadicOperatorMatcher(Op, std::move(InnerArgs));
Samuel Benzaquen4adca622013-08-28 18:42:04 +0000672 }
673
Fariborz Jahanian5afc8692014-10-01 16:56:40 +0000674 bool isVariadic() const override { return true; }
675 unsigned getNumArgs() const override { return 0; }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000676
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000677 void getArgKinds(ast_type_traits::ASTNodeKind ThisKind, unsigned ArgNo,
Fariborz Jahanian5afc8692014-10-01 16:56:40 +0000678 std::vector<ArgKind> &Kinds) const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000679 Kinds.push_back(ThisKind);
680 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000681
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000682 bool isConvertibleTo(ast_type_traits::ASTNodeKind Kind, unsigned *Specificity,
Fariborz Jahanian5afc8692014-10-01 16:56:40 +0000683 ast_type_traits::ASTNodeKind *LeastDerivedKind) const override {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000684 if (Specificity)
685 *Specificity = 1;
686 if (LeastDerivedKind)
687 *LeastDerivedKind = Kind;
688 return true;
689 }
Eugene Zelenko06becf82017-11-01 23:09:49 +0000690
Craig Topperbbc6d622014-03-02 10:02:43 +0000691 bool isPolymorphic() const override { return true; }
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000692
Samuel Benzaquen4adca622013-08-28 18:42:04 +0000693private:
Samuel Benzaquen4d058742013-11-22 14:41:48 +0000694 const unsigned MinCount;
695 const unsigned MaxCount;
Samuel Benzaquen2c15e8c2014-11-20 15:45:53 +0000696 const VarOp Op;
Samuel Benzaquen4adca622013-08-28 18:42:04 +0000697 const StringRef MatcherName;
698};
699
Manuel Klimek24db0f02013-05-14 09:13:00 +0000700/// Helper functions to select the appropriate marshaller functions.
Samuel Benzaquenb5dd69f2013-06-11 18:51:07 +0000701/// They detect the number of arguments, arguments types and return type.
Manuel Klimek24db0f02013-05-14 09:13:00 +0000702
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000703/// 0-arg overload
Manuel Klimek24db0f02013-05-14 09:13:00 +0000704template <typename ReturnType>
Justin Lebar82380d82016-10-10 16:26:40 +0000705std::unique_ptr<MatcherDescriptor>
706makeMatcherAutoMarshall(ReturnType (*Func)(), StringRef MatcherName) {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000707 std::vector<ast_type_traits::ASTNodeKind> RetTypes;
708 BuildReturnTypeVector<ReturnType>::build(RetTypes);
Justin Lebar82380d82016-10-10 16:26:40 +0000709 return llvm::make_unique<FixedArgCountMatcherDescriptor>(
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000710 matcherMarshall0<ReturnType>, reinterpret_cast<void (*)()>(Func),
Craig Topper00bbdcf2014-06-28 23:22:23 +0000711 MatcherName, RetTypes, None);
Manuel Klimek24db0f02013-05-14 09:13:00 +0000712}
713
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000714/// 1-arg overload
Manuel Klimek24db0f02013-05-14 09:13:00 +0000715template <typename ReturnType, typename ArgType1>
Justin Lebar82380d82016-10-10 16:26:40 +0000716std::unique_ptr<MatcherDescriptor>
717makeMatcherAutoMarshall(ReturnType (*Func)(ArgType1), StringRef MatcherName) {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000718 std::vector<ast_type_traits::ASTNodeKind> RetTypes;
719 BuildReturnTypeVector<ReturnType>::build(RetTypes);
720 ArgKind AK = ArgTypeTraits<ArgType1>::getKind();
Justin Lebar82380d82016-10-10 16:26:40 +0000721 return llvm::make_unique<FixedArgCountMatcherDescriptor>(
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000722 matcherMarshall1<ReturnType, ArgType1>,
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000723 reinterpret_cast<void (*)()>(Func), MatcherName, RetTypes, AK);
Manuel Klimek24db0f02013-05-14 09:13:00 +0000724}
725
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000726/// 2-arg overload
Samuel Benzaquenc31b3522013-06-04 15:46:22 +0000727template <typename ReturnType, typename ArgType1, typename ArgType2>
Justin Lebar82380d82016-10-10 16:26:40 +0000728std::unique_ptr<MatcherDescriptor>
729makeMatcherAutoMarshall(ReturnType (*Func)(ArgType1, ArgType2),
730 StringRef MatcherName) {
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000731 std::vector<ast_type_traits::ASTNodeKind> RetTypes;
732 BuildReturnTypeVector<ReturnType>::build(RetTypes);
733 ArgKind AKs[] = { ArgTypeTraits<ArgType1>::getKind(),
734 ArgTypeTraits<ArgType2>::getKind() };
Justin Lebar82380d82016-10-10 16:26:40 +0000735 return llvm::make_unique<FixedArgCountMatcherDescriptor>(
Samuel Benzaquen998cda232013-08-30 15:09:52 +0000736 matcherMarshall2<ReturnType, ArgType1, ArgType2>,
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000737 reinterpret_cast<void (*)()>(Func), MatcherName, RetTypes, AKs);
Samuel Benzaquenc31b3522013-06-04 15:46:22 +0000738}
739
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000740/// Variadic overload.
Samuel Benzaquen79656e12013-07-15 19:25:06 +0000741template <typename ResultT, typename ArgT,
742 ResultT (*Func)(ArrayRef<const ArgT *>)>
Justin Lebar82380d82016-10-10 16:26:40 +0000743std::unique_ptr<MatcherDescriptor> makeMatcherAutoMarshall(
Samuel Benzaquenc1384c12016-03-25 16:29:30 +0000744 ast_matchers::internal::VariadicFunction<ResultT, ArgT, Func> VarFunc,
745 StringRef MatcherName) {
Justin Lebar82380d82016-10-10 16:26:40 +0000746 return llvm::make_unique<VariadicFuncMatcherDescriptor>(VarFunc, MatcherName);
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000747}
748
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000749/// Overload for VariadicDynCastAllOfMatchers.
Peter Collingbourned32e28c2014-01-23 22:48:38 +0000750///
751/// Not strictly necessary, but DynCastAllOfMatcherDescriptor gives us better
752/// completion results for that type of matcher.
753template <typename BaseT, typename DerivedT>
Justin Lebar82380d82016-10-10 16:26:40 +0000754std::unique_ptr<MatcherDescriptor> makeMatcherAutoMarshall(
755 ast_matchers::internal::VariadicDynCastAllOfMatcher<BaseT, DerivedT>
756 VarFunc,
757 StringRef MatcherName) {
758 return llvm::make_unique<DynCastAllOfMatcherDescriptor>(VarFunc, MatcherName);
Manuel Klimek24db0f02013-05-14 09:13:00 +0000759}
760
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000761/// Argument adaptative overload.
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000762template <template <typename ToArg, typename FromArg> class ArgumentAdapterT,
763 typename FromTypes, typename ToTypes>
Justin Lebar82380d82016-10-10 16:26:40 +0000764std::unique_ptr<MatcherDescriptor> makeMatcherAutoMarshall(
765 ast_matchers::internal::ArgumentAdaptingMatcherFunc<ArgumentAdapterT,
766 FromTypes, ToTypes>,
767 StringRef MatcherName) {
768 std::vector<std::unique_ptr<MatcherDescriptor>> Overloads;
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000769 AdaptativeOverloadCollector<ArgumentAdapterT, FromTypes, ToTypes>(MatcherName,
770 Overloads);
Justin Lebar82380d82016-10-10 16:26:40 +0000771 return llvm::make_unique<OverloadedMatcherDescriptor>(Overloads);
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000772}
773
774template <template <typename ToArg, typename FromArg> class ArgumentAdapterT,
775 typename FromTypes, typename ToTypes>
776template <typename FromTypeList>
Peter Collingbournef43e6942013-11-23 01:34:36 +0000777inline void AdaptativeOverloadCollector<ArgumentAdapterT, FromTypes,
778 ToTypes>::collect(FromTypeList) {
Samuel Benzaquenbd7d8872013-08-16 16:19:42 +0000779 Out.push_back(makeMatcherAutoMarshall(
780 &AdaptativeFunc::template create<typename FromTypeList::head>, Name));
781 collect(typename FromTypeList::tail());
782}
783
Adrian Prantl9fc8faf2018-05-09 01:00:01 +0000784/// Variadic operator overload.
Samuel Benzaquen4d058742013-11-22 14:41:48 +0000785template <unsigned MinCount, unsigned MaxCount>
Justin Lebar82380d82016-10-10 16:26:40 +0000786std::unique_ptr<MatcherDescriptor> makeMatcherAutoMarshall(
787 ast_matchers::internal::VariadicOperatorMatcherFunc<MinCount, MaxCount>
788 Func,
789 StringRef MatcherName) {
790 return llvm::make_unique<VariadicOperatorMatcherDescriptor>(
791 MinCount, MaxCount, Func.Op, MatcherName);
Samuel Benzaquen4adca622013-08-28 18:42:04 +0000792}
793
Eugene Zelenko1660a5d2016-01-26 19:01:06 +0000794} // namespace internal
795} // namespace dynamic
796} // namespace ast_matchers
797} // namespace clang
Manuel Klimek24db0f02013-05-14 09:13:00 +0000798
Eugene Zelenko1660a5d2016-01-26 19:01:06 +0000799#endif // LLVM_CLANG_AST_MATCHERS_DYNAMIC_MARSHALLERS_H