Details
- Reviewers
faisalv aaron.ballman • fraggamuffin rsmith hubert.reinterpretcast - Commits
- rGde49845b05d9: [Concepts] Implement a portion of Concepts TS[dcl.spec.concept]p1 by diagnosing…
rC260074: [Concepts] Implement a portion of Concepts TS[dcl.spec.concept]p1 by
rL260074: [Concepts] Implement a portion of Concepts TS[dcl.spec.concept]p1 by
Diff Detail
Event Timeline
test/CXX/concepts-ts/dcl.dcl/dcl.spec/dcl.concept/p7.cpp | ||
---|---|---|
13 ↗ | (On Diff #36270) | "extern template" is a form of explicit instantiation as well. I suggest testing that for both the function concept and variable concept cases. |
test/CXX/concepts-ts/dcl.dcl/dcl.spec/dcl.concept/p7.cpp | ||
---|---|---|
4 ↗ | (On Diff #36270) | This is not a declaration (never mind definition) of a function template or variable template, but of a specialization. Presumably, this violates [dcl.spec.concept]p1. Perhaps a test for p7 should omit the concept specifier? The same logic may apply to the explicit specialization cases. |
lib/Sema/SemaDecl.cpp | ||
---|---|---|
5902–5915 | Maybe combine these into a single check, and use << (IsPartialSpecialization ? 2 : 1) for the specialization kind. | |
7558 | Missing period. | |
7886 | I'm not convinced this is going to work: if the declaration is invalid because it's declared concept /and/ for another reason, we presumably want to use the same error recovery path as before. | |
test/CXX/concepts-ts/dcl.dcl/dcl.spec/dcl.concept/p7.cpp | ||
1 ↗ | (On Diff #36270) | This file is in the wrong directory; the relevant section is [dcl.spec.concept], not [dcl.concept]. |
4 ↗ | (On Diff #36270) | Right, there are two different ways things can go wrong here:
This patch is only checking for the former case, whereas the rule in [dcl.spec.concept]p7 says that the latter case is ill-formed. So these tests are at least in the wrong file. To check for violations of p7, you'll need to look at whether the template's pattern is marked as being a concept. I'm also not sure where we say that a partial specialization of a variable template cannot be declared concept; that seems to be a defect in the TS wording. |
Moving tests to the correct file.
Modifying the diagnostic id and message so it's more applicable to the checks in this Patch. Update the quoted sections of the standard as well.
Use ternary operator for Partial Specialization check.
Add missing period in comment.
lib/Sema/SemaDecl.cpp | ||
---|---|---|
7886 | Maybe there could be a problem further down if we think there aren't explicit template args and there really are. We could do a check similar to the isFriend check in the same block below like this: else if (isConcept && isFunctionTemplateSpecialization) { HasExplicitTemplateArgs = true; } But it seems like that's essentially the same thing. |
Addressing Richard's other comment regarding the check that the FunctionDecl is not a concept. Also, not marking the specialization as invalid if it's declared with the concept specifier since the downstream parts of Clang should look at the primary template.
include/clang/AST/Decl.h | ||
---|---|---|
1577 | My inclination would have been to add this bit to FunctionTemplateDecl, instead of to every FunctionDecl - since not every kind of FunctionDecl can be a concept ... My first instinct would have been to add an enum to TemplateKind, and then forward isConcept() to check if we are a template and if so, through it, if concept is specified? But I suppose that adds more complexity, and you trade space for speed - For my own edification, could I ask you or Richard to comment on the cons of that approach - and why the current approach is preferred? (i.e. simplicity over complexity or space over time etc.) |
include/clang/AST/Decl.h | ||
---|---|---|
1577 | Sorry for the slow reply. Yeah, I thought it was a little more straightforward having isConcept as a member function of FunctionDecl (and the bit). It also seemed that we'd be okay in terms of size since 17 bits are already being used here and adding one more wouldn't go over a byte boundary. Maybe Richard has other thoughts though? |
include/clang/AST/Decl.h | ||
---|---|---|
1577 | I am not sure what the original justification was for IsTrivial to be a bit here instead of in CXXMethodDecl, but it seems that similar considerations apply. | |
lib/Sema/SemaDecl.cpp | ||
5909 | Should we mention that the partial specialization case is pending additional wording (under Concepts TS Issue 15)? |
lib/Sema/SemaDecl.cpp | ||
---|---|---|
5909 | Hmm, I'd lean toward leaving it as is until the wording is sorted out. I don't feel too strongly either way though. |
- Remove marking a variable concept invalid when specialized since we'll only look at the primary template downstream. This removal let's us use the same recovery path as before when 'concept' is specified on a non-template.
Comment added inline. Otherwise, LGTM.
lib/Sema/SemaDecl.cpp | ||
---|---|---|
7619 | I don't think the declaration should still be marked as a concept in this case. |
Aside from one question, LGTM.
~Aaron
include/clang/Basic/DiagnosticSemaKinds.td | ||
---|---|---|
1991 | Is this an extraneous %? |
lib/Sema/SemaDecl.cpp | ||
---|---|---|
7619 | A consideration: |
lib/Sema/SemaDecl.cpp | ||
---|---|---|
7619 | I think the concept flag logically belongs on the template declaration rather than on the templated declaration; moving it there would make this question irrelevant =) |
lib/Sema/SemaDecl.cpp | ||
---|---|---|
7619 | Sorry for the slow reply. Okay. That makes sense. So, we'd be okay putting the IsConcept bit (and the associated member functions) in TemplateDecl then? You'd be okay with the extra byte used? |
- Move IsConcept bit and associated member functions from FunctionDecl to FunctionTemplateDecl.
- Set the IsConcept flag using getDescribedFunctionTemplate.
include/clang/AST/DeclTemplate.h | ||
---|---|---|
836 ↗ | (On Diff #43469) | This might make more sense on TemplateDecl, since we also want this flag for VarTemplateDecls. In any case, please use some existing spare bit for this rather than making all FunctionTemplateDecls 8 bytes larger by putting it here. |
986 ↗ | (On Diff #43469) | Do we need a setter for this? (Can it change after the decl is constructed?) |
lib/Sema/SemaDecl.cpp | ||
7619 | Yes, I think that's a reasonable way forward. You can change either TemplatedDecl or TemplateParams into a PointerIntPair to avoid making TemplateDecl larger. | |
lib/Sema/SemaTemplate.cpp | ||
7602–7604 | I don't think we need to quote the second sentence here. | |
7607 | No space after :: |
include/clang/AST/DeclTemplate.h | ||
---|---|---|
836 ↗ | (On Diff #43469) | @Nate - since you asked before, if I was to try and do this (now that we know Richard cares strongly about the extra space), I would probably risk complexity (and insult readability) by attempting to smuggle the extra bit into RedeclararableTemplateDecl::CommonBase::InstantiatedFromMember or perhaps within the low bits of the pointer to ReDeclarableTemplateDecl::CommonBase itself (i.e. Common) |
include/clang/AST/DeclTemplate.h | ||
---|---|---|
836 ↗ | (On Diff #43469) | @rsmith - Yeah, I originally thought to put it in TemplateDecl but reconsidered since ClassTemplateDecls would get it as well . However, I guess by making TemplateDecl::TemplatedDecl an IntPointerPair that concern is irrelevant. @faisalv - Thanks for the info and suggestion! This helped with Richard's suggestion (below). FYI - It looks like that IntPointerPair, InstantiatedFromMember, is for explicit specializations, and the pointer to CommonBase (Common) is null for a function concept's primary template declaration. |
986 ↗ | (On Diff #43469) | The setter is used below when we see that concept is specified on the declaration. So, it wouldn't be initialized (or set) to true when the decl is constructed. Or, do you have a different opinion on where to set IsConcept to true such as how IsConstExpr is initialized in FunctionDecls being passed as a param in the constructor? |
lib/Sema/SemaDecl.cpp | ||
7619 | Thanks for the suggestion. I'll go with TemplatedDecl and see what you guys think. |
- Store the IsConcept boolean flag in TemplateDecl by making TemplatedDecl an IntPointerPair, and move the associated member functions into TemplateDecl.
- Remove unnecessary quoted comment.
- Remove an extra space where the diagnostic is used.
include/clang/AST/DeclTemplate.h | ||
---|---|---|
375 ↗ | (On Diff #43670) | Hmm, I think I should actually use that parameter since it's a bug as is (and would still be if I removed the parameter). I'll plan on doing that unless there is another thought about this. |
include/clang/AST/DeclTemplate.h | ||
---|---|---|
375 ↗ | (On Diff #43670) | My interpretation of http://reviews.llvm.org/D13357?id=43469#inline-129592 is that the function should take no parameters (there is no need for a way to set the property to false). Perhaps I misunderstood Richard's intent. |
include/clang/AST/DeclTemplate.h | ||
---|---|---|
377 ↗ | (On Diff #43469) | I can't seem to follow the link properly, but I'm assuming it's supposed to be where Richard is asking about whether we need a setter. I thought he meant whether we needed the setter function at all. Perhaps I misunderstood though. But it seems to be the convention to give a boolean (or some other type of param) to a setter. I'd be fine either way though because as you said, there isn't a need to set the property to false. As a point of reference RedeclarableTemplateDecl::setMemberSpecialization doesn't take any params either. |
Ping.
@rsmith - would you also mind clarifying the comment regarding setConcept(bool IC) at to whether it should exist at all or simply not have any params?
- Fix a couple of comments to reflect the Patch.
- Clang-format the changes in this Patch.
include/clang/AST/DeclTemplate.h | ||
---|---|---|
374 ↗ | (On Diff #46588) | I would prefer to not have a setter at all, but if it's awkward to pass this flag into the constructor, then a setter is fine (and I don't mind whether or not it takes a parameter). We should definitely not take a parameter and ignore it though. |
include/clang/AST/DeclTemplate.h | ||
---|---|---|
374 ↗ | (On Diff #46588) | Ah, gotcha. Yeah, it seemed somewhat awkward to me to pass the flag all the way through. As a note it would be VarTemplateDecl -> RedeclarableTemplateDecl -> TemplateDecl. Also, RedeclarableTemplateDecl also currently currently takes 7 parameters. If you feel strongly, I can move it to the constructor. Otherwise, I'll take the parameter out of the setter as both you and Hubert have pointed out the property will not need to set to false. |
- This update removes the parameter in TemplateDecl::setConcept since there isn't a need to mark a concept false.
Minor comments; otherwise, LGTM.
lib/Sema/SemaDecl.cpp | ||
---|---|---|
5903 | We do not need to quote the second sentence here. The ellipsis in the first sentence should be expanded since the full list is necessary to conclude that the cases being diagnosed here are excluded. | |
7616 | We do not need to quote the second sentence here. The ellipsis in the first sentence should be expanded since the full list is necessary to conclude that the cases being diagnosed here are excluded. |
My inclination would have been to add this bit to FunctionTemplateDecl, instead of to every FunctionDecl - since not every kind of FunctionDecl can be a concept ... My first instinct would have been to add an enum to TemplateKind, and then forward isConcept() to check if we are a template and if so, through it, if concept is specified?
But I suppose that adds more complexity, and you trade space for speed - For my own edification, could I ask you or Richard to comment on the cons of that approach - and why the current approach is preferred? (i.e. simplicity over complexity or space over time etc.)