Implementation of checker "different.LabelInsideSwitch" from potential checkers list (https://clang-analyzer.llvm.org/potential_checkers.html#different)
Diff Detail
- Repository
- rC Clang
Event Timeline
Hello Alexey,
Thank you for the patch. I have made a preliminary review and will add some other reviewers. You can find my comments inline.
lib/StaticAnalyzer/Checkers/LabelInsideSwitchChecker.cpp | ||
---|---|---|
1 | License header and checker descriptions are missed here. You can take a look at any other checker to find how it should be filled. | |
10 | Spaces in template arguments should be removed. | |
12 | LLVM Coding Standard requires variables to start with capital letter: 'Mgr'. Please change such names here and below. | |
15 | We usually pass StringRef by value, not by reference. Same below. | |
16 | LLVM Coding Standard requires method names to start with small letter. | |
33 | Obvious comment. I think it is better to remove it. | |
47 |
| |
59 | raw_svector_ostream is always synchronized with the string it prints to so we can just pass the message string instead of calling .str(). | |
65 | Seems like you are trying to implement some kind of "similarity" metric. Consider using StringRef::edit_distance() instead. I guess, in your case it will give you much more freedom to vary different match parameters. For example, it can consider replacements. | |
84 |
| |
110 | This line exceeds 80-char limit. Please split it. | |
115 | You can write just `void ento::registerLabelInsideSwitchChecker(CheckerManager &Mgr) { | |
test/Analysis/label-inside-switch.c | ||
1 | Not sure if we need to enable 'core' while testing path-insensitive checkers. Artem? | |
5 | Space after switch. Same below |
A few comments.
lib/StaticAnalyzer/Checkers/LabelInsideSwitchChecker.cpp | ||
---|---|---|
19 | Maybe I miss something, but do not we return StringRef to temporary string going out of scope here? Same below. | |
24 | Do you consider using ASTMatchers like in NumberObjectConversionChecker instead of manually traversing the AST? | |
59 | You could use S->getSourceRange() instead of Sr, as llvm::ArrayRef in EmitBasicReport() could be constructed even from the 0 consecutively elements. | |
80 | Maybe so? |
- Now implemented via MatchFinder
- Added missing License header
- Pass all StringRefs by value
- Method names now start from small letter
- Using StringRef::edit_distance instead of custom "similarity" metric
- Various other formating fixes
lib/StaticAnalyzer/Checkers/LabelInsideSwitchChecker.cpp | ||
---|---|---|
24 | Haven't seen it before but surely will consider to use it. Looks like I should use StatementMatcher, shouldn't I? | |
24 | One more thing. After reviewing implementation of NumberObjectConversionChecker I'm not sure if it's more clear to use Matcher + Callback there. | |
59 | Is there way to getSourceRange() which doesn't include following comment (if it exists)? | |
65 | Looks like it can be replaced by StringRef::edit_distance | |
80 | SizeDelta is abs(StrSize - ExpSize) from logical point of view. But if StrSize < ExpSize subtraction ExpSize from StrSize will underflow. Anyway I'm going to replace whole function with existing implementation StringRef::edit_distance. | |
88 | Looks like I have to use forEachDescendant instead of hasDescendant. Please, comment! | |
94 | Typo: Of course should be "sense" instead | |
115 | Actually I can't, get error: /llvm/tools/clang/lib/StaticAnalyzer/Checkers/LabelInsideSwitchChecker.cpp:103:68: error: 'void clang::ento::registerLabelInsideSwitchChecker(clang::ento::CheckerManager&)' should have been declared inside 'clang::ento' void ento::registerLabelInsideSwitchChecker(CheckerManager &Mgr) { |
Hello Alexey,
Thank you for the update. The code looks much cleaner now.
lib/StaticAnalyzer/Checkers/LabelInsideSwitchChecker.cpp | ||
---|---|---|
2 | Check for label inside switch | |
12 | misprinting | |
20 | We don't use CheckerContext in the code below. Looks like this include is redundant or should be replaced by more relevant include files. | |
88 |
switch (x) } case 1: { {{ label: }} // ignored } } Is that intentional? | |
115 | This looks strange. I can remember that I have met this issue but I cannot remember how I resolved it. |
lib/StaticAnalyzer/Checkers/LabelInsideSwitchChecker.cpp | ||
---|---|---|
88 |
switch (x) { case 1: someCodeHere; cas: // missprint ... // other cases }
switch (x) { cas: blabla; // this label is not a child of any case stmt case 3: ... ... // etc On the other hand I would like to avoid matching deeply-nested labels inside switch compound stmt for reducing false-positives probability. Actually I'm not sure in my assumption. |
Thanks for looking into this!
This checker is in the 'core' package, which means (when moved out of alpha) it will be enabled by default.
- Do you think that this checker should be enabled by default for all users of the analyzer?
- If users do actually want to use labels in their switch statements, how should they suppress the diagnostics from the checker?
- What is the benefit of adding this check in the static analyzer vs. in clang-tidy?
(My own sense is that the check for labels that are close to "default" could be on by default but that warning on *any* label inside a switch is more stylistic. I think users should have to opt in to that check.)
I think so
- If users do actually want to use labels in their switch statements, how should they suppress the diagnostics from the checker?
Good point, is there recommended way to implement options for checker? Where can I find any reference example?
- What is the benefit of adding this check in the static analyzer vs. in clang-tidy?
(My own sense is that the check for labels that are close to "default" could be on by default but that warning on *any* label inside a switch is more stylistic. I think users should have to opt in to that check.)
It makes sense. So, I can make generic case when we found any label in swichStmt opt-in (default=off) and left cases when it looks like typo in 'case' or 'default' keywords enabled by default.
Thanks!
License header and checker descriptions are missed here. You can take a look at any other checker to find how it should be filled.