//===--- ASTLambda.h - Lambda Helper Functions --------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// /// /// \file /// This file provides some common utility functions for processing /// Lambda related AST Constructs. /// //===----------------------------------------------------------------------===// #ifndef LLVM_CLANG_AST_ASTLAMBDA_H #define LLVM_CLANG_AST_ASTLAMBDA_H #include "clang/AST/DeclCXX.h" #include "clang/AST/DeclTemplate.h" namespace clang { inline StringRef getLambdaStaticInvokerName() { return "__invoke"; } // This function returns true if M is a specialization, a template, // or a non-generic lambda call operator. inline bool isLambdaCallOperator(const CXXMethodDecl *MD) { const CXXRecordDecl *LambdaClass = MD->getParent(); if (!LambdaClass || !LambdaClass->isLambda()) return false; return MD->getOverloadedOperator() == OO_Call; } inline bool isLambdaCallOperator(const DeclContext *DC) { if (!DC || !isa(DC)) return false; return isLambdaCallOperator(cast(DC)); } inline bool isGenericLambdaCallOperatorSpecialization(const CXXMethodDecl *MD) { if (!MD) return false; const CXXRecordDecl *LambdaClass = MD->getParent(); if (LambdaClass && LambdaClass->isGenericLambda()) return isLambdaCallOperator(MD) && MD->isFunctionTemplateSpecialization(); return false; } inline bool isLambdaConversionOperator(CXXConversionDecl *C) { return C ? C->getParent()->isLambda() : false; } inline bool isLambdaConversionOperator(Decl *D) { if (!D) return false; if (CXXConversionDecl *Conv = dyn_cast(D)) return isLambdaConversionOperator(Conv); if (FunctionTemplateDecl *F = dyn_cast(D)) if (CXXConversionDecl *Conv = dyn_cast_or_null(F->getTemplatedDecl())) return isLambdaConversionOperator(Conv); return false; } inline bool isGenericLambdaCallOperatorSpecialization(DeclContext *DC) { return isGenericLambdaCallOperatorSpecialization( dyn_cast(DC)); } inline bool isGenericLambdaCallOperatorOrStaticInvokerSpecialization( const DeclContext *DC) { const auto *MD = dyn_cast(DC); if (!MD) return false; const CXXRecordDecl *LambdaClass = MD->getParent(); if (LambdaClass && LambdaClass->isGenericLambda()) return (isLambdaCallOperator(MD) || MD->isLambdaStaticInvoker()) && MD->isFunctionTemplateSpecialization(); return false; } // This returns the parent DeclContext ensuring that the correct // parent DeclContext is returned for Lambdas inline DeclContext *getLambdaAwareParentOfDeclContext(DeclContext *DC) { if (isLambdaCallOperator(DC)) return DC->getParent()->getParent(); else return DC->getParent(); } } // clang #endif