-
Notifications
You must be signed in to change notification settings - Fork 125
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Delay the differentiation process until the end of TU.
Before this patch clad attaches itself as a first consumer and applies AD before code generation. However, that is limited since clang sends every top-level declaration to codegen which limits the amount of flexibility clad has. For example, we have to instantiate all pending templates at every HandleTopLevelDecl calls; we cannot really differentiate virtual functions whose classes have sent their key function to CodeGen; and in general we perform actions which are semantically useful for the end of the translation unit. This patch makes clad a single consumer of clang which dispatches to the others. That's done by delaying all calls to the consumers until the end of the TU where clad can replay the exact sequence of calls to the other consumers as if they were directly connected to the frontend. Fixes #248
- Loading branch information
1 parent
e930a4a
commit a730695
Showing
8 changed files
with
478 additions
and
79 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
#ifndef CLAD_DIFFERENTIATOR_SINS_H | ||
#define CLAD_DIFFERENTIATOR_SINS_H | ||
|
||
#include <type_traits> | ||
|
||
/// Standard-protected facility allowing access into private members in C++. | ||
/// Use with caution! | ||
// NOLINTBEGIN(cppcoreguidelines-macro-usage) | ||
#define CONCATE_(X, Y) X##Y | ||
#define CONCATE(X, Y) CONCATE_(X, Y) | ||
#define ALLOW_ACCESS(CLASS, MEMBER, ...) \ | ||
template <typename Only, __VA_ARGS__ CLASS::*Member> \ | ||
struct CONCATE(MEMBER, __LINE__) { \ | ||
friend __VA_ARGS__ CLASS::*Access(Only*) { return Member; } \ | ||
}; \ | ||
template <typename> struct Only_##MEMBER; \ | ||
template <> struct Only_##MEMBER<CLASS> { \ | ||
friend __VA_ARGS__ CLASS::*Access(Only_##MEMBER<CLASS>*); \ | ||
}; \ | ||
template struct CONCATE(MEMBER, \ | ||
__LINE__)<Only_##MEMBER<CLASS>, &CLASS::MEMBER> | ||
|
||
#define ACCESS(OBJECT, MEMBER) \ | ||
(OBJECT).*Access((Only_##MEMBER< \ | ||
std::remove_reference<decltype(OBJECT)>::type>*)nullptr) | ||
|
||
// NOLINTEND(cppcoreguidelines-macro-usage) | ||
|
||
#endif // CLAD_DIFFERENTIATOR_SINS_H |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,81 @@ | ||
// RUN: %cladclang %s -I%S/../../include -oClangConsumers.out \ | ||
// RUN: -fms-compatibility -DMS_COMPAT -std=c++14 -fmodules \ | ||
// RUN: -Xclang -print-stats 2>&1 | FileCheck %s | ||
// CHECK-NOT: {{.*error|warning|note:.*}} | ||
// | ||
// RUN: clang -xc -Xclang -add-plugin -Xclang clad -Xclang -load \ | ||
// RUN: -Xclang %cladlib %s -I%S/../../include -oClangConsumers.out \ | ||
// RUN: -Xclang -debug-info-kind=limited -Xclang -triple -Xclang bpf-linux-gnu \ | ||
// RUN: -S -emit-llvm -Xclang -target-cpu -Xclang generic \ | ||
// RUN: -Xclang -print-stats 2>&1 | \ | ||
// RUN: FileCheck -check-prefix=CHECK_C %s | ||
// CHECK_C-NOT: {{.*error|warning|note:.*}} | ||
// XFAIL: clang-7, clang-8, clang-9, target={{i586.*}}, target=arm64-apple-{{.*}} | ||
// | ||
// RUN: clang -xobjective-c -Xclang -add-plugin -Xclang clad -Xclang -load \ | ||
// RUN: -Xclang %cladlib %s -I%S/../../include -oClangConsumers.out \ | ||
// RUN: -Xclang -print-stats 2>&1 | \ | ||
// RUN: FileCheck -check-prefix=CHECK_OBJC %s | ||
// CHECK_OBJC-NOT: {{.*error|warning|note:.*}} | ||
|
||
#ifdef __cplusplus | ||
|
||
#pragma clang module build N | ||
module N {} | ||
#pragma clang module contents | ||
#pragma clang module begin N | ||
struct f { void operator()() const {} }; | ||
template <typename T> auto vtemplate = f{}; | ||
#pragma clang module end | ||
#pragma clang module endbuild | ||
|
||
#pragma clang module import N | ||
|
||
#ifdef MS_COMPAT | ||
class __single_inheritance IncSingle; | ||
#endif // MS_COMPAT | ||
|
||
struct V { virtual int f(); }; | ||
int V::f() { return 1; } | ||
template <typename T> T f() { return T(); } | ||
int i = f<int>(); | ||
|
||
// Check if shouldSkipFunctionBody is called. | ||
// RUN: %cladclang -I%S/../../include -fsyntax-only -fmodules \ | ||
// RUN: -Xclang -code-completion-at=%s:%(line-1):1 %s -o - | \ | ||
// RUN: FileCheck -check-prefix=CHECK-CODECOMP %s | ||
// CHECK-CODECOMP: COMPLETION | ||
|
||
// CHECK: HandleImplicitImportDecl | ||
// CHECK: AssignInheritanceModel | ||
// CHECK: HandleTopLevelDecl | ||
// CHECK: HandleCXXImplicitFunctionInstantiation | ||
// CHECK: HandleInterestingDecl | ||
// CHECK: HandleVTable | ||
// CHECK: HandleCXXStaticMemberVarInstantiation | ||
|
||
#endif // __cplusplus | ||
|
||
#ifdef __STDC_VERSION__ // C mode | ||
int i; | ||
|
||
extern char ch; | ||
int test(void) { return ch; } | ||
char ch = 1; | ||
|
||
// CHECK_C: CompleteTentativeDefinition | ||
// CHECK_C: CompleteExternalDeclaration | ||
#endif // __STDC_VERSION__ | ||
|
||
#ifdef __OBJC__ | ||
@interface I | ||
void f(); | ||
@end | ||
// CHECK_OBJC: HandleTopLevelDeclInObjCContainer | ||
#endif // __OBJC__ | ||
|
||
int main() { | ||
#ifdef __cplusplus | ||
vtemplate<int>(); | ||
#endif // __cplusplus | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.