tahonermann added inline comments.

================
Comment at: clang/lib/Sema/SemaTemplateInstantiate.cpp:2460-2475
-    FunctionDecl *OwningFunc = cast<FunctionDecl>(OldParm->getDeclContext());
-    if (OwningFunc->isInLocalScopeForInstantiation()) {
-      // Instantiate default arguments for methods of local classes (DR1484)
-      // and non-defining declarations.
-      Sema::ContextRAII SavedContext(*this, OwningFunc);
-      LocalInstantiationScope Local(*this, true);
-      ExprResult NewArg = SubstExpr(Arg, TemplateArgs);
----------------
This functionality is now incorporated into `Sema::SubstDefaultArguments()` and 
postponed until after construction of enclosing declaration contexts.


================
Comment at: clang/lib/Sema/SemaTemplateInstantiate.cpp:2592-2613
+  if (ForCallExpr) {
+    // Check the expression as an initializer for the parameter.
+    InitializedEntity Entity
+      = InitializedEntity::InitializeParameter(Context, Param);
+    InitializationKind Kind = InitializationKind::CreateCopy(
+        Param->getLocation(),
+        /*FIXME:EqualLoc*/ PatternExpr->getBeginLoc());
----------------
I find this code exceedingly troubling, but I haven't been able to figure out 
how to unify it. Conceptually, instantiation of a default argument should 
produce the same result (the instantiated default argument is cached after 
all!) whether it is implicitly always instantiated (as in a declaration in 
local scope or a friend function definition) or instantiated on demand for a 
call expression. However, my attempts to unify this code have all resulted in 
test failures that I have, so far, been unable to explain. Note that much of 
the code in the `ForCallExpr` case is replicated in the 
`ConvertParamDefaultArgument()` function. I think this should be investigated 
further and may be a reason to postpone approving this change.


================
Comment at: clang/lib/Sema/SemaTemplateInstantiateDecl.cpp:2138
+  else if (D->isLocalExternDecl()) {
+    LexicalDC = SemaRef.CurContext;
+  }
----------------
This seems highly questionable to me. I suspect there is a better way to 
identify the correct context, but I have not so far found it elsewhere. Note 
that the context needed is for the enclosing function template specialization 
(not the enclosing template).


================
Comment at: clang/lib/Sema/SemaTemplateInstantiateDecl.cpp:2296-2311
+    for (ParmVarDecl *PVD : Function->parameters()) {
+      if (!PVD->hasDefaultArg())
+        continue;
+      if (SemaRef.SubstDefaultArgument(D->getInnerLocStart(), PVD, 
TemplateArgs)) {
+        // If substitution fails, the default argument is set to a
+        // RecoveryExpr that wraps the uninstantiated default argument so
+        // that downstream diagnostics are omitted.
----------------
This code is effectively replicated in three distinct locations. It is probably 
worth moving it into a helper function somewhere.


================
Comment at: clang/lib/Sema/SemaTemplateInstantiateDecl.cpp:4565
-                          /*DiscardedValue*/ false);
-  if (Result.isInvalid())
     return true;
----------------
The above removed code has been incorporated into 
`Sema::SubstDefaultArgument()`.


Repository:
  rG LLVM Github Monorepo

CHANGES SINCE LAST ACTION
  https://reviews.llvm.org/D133500/new/

https://reviews.llvm.org/D133500

_______________________________________________
cfe-commits mailing list
cfe-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits

Reply via email to