Skip to content

Commit a5c743c

Browse files
authored
Merge pull request #713 from github/michaelrfairhurst/implement-misra-c-noreturn-rule-package
Implement Misra-c Noreturn rule package.
2 parents 021fabb + 71b4c25 commit a5c743c

19 files changed

+475
-7
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
| test.c:9:16:9:31 | test_noreturn_f2 | The function test_noreturn_f2 declared with attribute _Noreturn returns a value. |
2+
| test.c:34:16:34:31 | test_noreturn_f5 | The function test_noreturn_f5 declared with attribute _Noreturn returns a value. |
3+
| test.c:49:32:49:47 | test_noreturn_f7 | The function test_noreturn_f7 declared with attribute _Noreturn returns a value. |
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
// GENERATED FILE - DO NOT MODIFY
2+
import codingstandards.cpp.rules.functionnoreturnattributecondition.FunctionNoReturnAttributeCondition
3+
4+
class TestFileQuery extends FunctionNoReturnAttributeConditionSharedQuery, TestQuery { }
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,88 @@
1+
#include "setjmp.h"
2+
#include "stdlib.h"
3+
#include "threads.h"
4+
5+
_Noreturn void test_noreturn_f1(int i) { // COMPLIANT
6+
abort();
7+
}
8+
9+
_Noreturn void test_noreturn_f2(int i) { // NON_COMPLIANT
10+
if (i > 0) {
11+
abort();
12+
}
13+
if (i < 0) {
14+
abort();
15+
}
16+
}
17+
18+
_Noreturn void test_noreturn_f3(int i) { // COMPLIANT
19+
if (i > 0) {
20+
abort();
21+
}
22+
exit(1);
23+
}
24+
25+
void test_noreturn_f4(int i) { // COMPLIANT
26+
if (i > 0) {
27+
abort();
28+
}
29+
if (i < 0) {
30+
abort();
31+
}
32+
}
33+
34+
_Noreturn void test_noreturn_f5(int i) { // NON_COMPLIANT
35+
if (i > 0) {
36+
abort();
37+
}
38+
}
39+
40+
_Noreturn void test_noreturn_f6(int i) { // COMPLIANT
41+
if (i > 0) {
42+
abort();
43+
}
44+
while (1) {
45+
i = 5;
46+
}
47+
}
48+
49+
__attribute__((noreturn)) void test_noreturn_f7(int i) { // NON_COMPLIANT
50+
if (i > 0) {
51+
abort();
52+
}
53+
}
54+
55+
__attribute__((noreturn)) void test_noreturn_f8(int i) { // COMPLIANT
56+
abort();
57+
}
58+
59+
_Noreturn void test_noreturn_f9(int i) { // COMPLIANT
60+
test_noreturn_f1(i);
61+
}
62+
63+
_Noreturn void test_noreturn_f10(int i) { // COMPLIANT
64+
switch (i) {
65+
case 0:
66+
abort();
67+
break;
68+
case 1:
69+
exit(0);
70+
break;
71+
case 2:
72+
_Exit(0);
73+
break;
74+
case 3:
75+
quick_exit(0);
76+
break;
77+
case 4:
78+
thrd_exit(0);
79+
break;
80+
default:
81+
jmp_buf jb;
82+
longjmp(jb, 0);
83+
}
84+
}
85+
86+
_Noreturn void test_noreturn_f11(int i) { // COMPLIANT
87+
return test_noreturn_f11(i);
88+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
/**
2+
* @id c/misra/non-void-return-type-of-noreturn-function
3+
* @name RULE-17-10: A function declared with _noreturn shall have a return type of void
4+
* @description Function declared with _noreturn will by definition not return a value, and should
5+
* be declared to return void.
6+
* @kind problem
7+
* @precision very-high
8+
* @problem.severity recommendation
9+
* @tags external/misra/id/rule-17-10
10+
* correctness
11+
* external/misra/obligation/required
12+
*/
13+
14+
import cpp
15+
import codingstandards.c.misra
16+
import codingstandards.cpp.Noreturn
17+
18+
from NoreturnFunction f, Type returnType
19+
where
20+
not isExcluded(f, NoReturnPackage::nonVoidReturnTypeOfNoreturnFunctionQuery()) and
21+
returnType = f.getType() and
22+
not returnType instanceof VoidType and
23+
not f.isCompilerGenerated()
24+
select f,
25+
"The function " + f.getName() + " is declared _noreturn but has a return type of " +
26+
returnType.toString() + "."
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
/**
2+
* @id c/misra/function-with-no-returning-branch-should-be-noreturn
3+
* @name RULE-17-11: A function without a branch that returns shall be declared with _Noreturn
4+
* @description Functions which cannot return should be declared with _Noreturn.
5+
* @kind problem
6+
* @precision very-high
7+
* @problem.severity recommendation
8+
* @tags external/misra/id/rule-17-11
9+
* correctness
10+
* external/misra/obligation/advisory
11+
*/
12+
13+
import cpp
14+
import codingstandards.c.misra
15+
import codingstandards.cpp.Noreturn
16+
17+
from Function f
18+
where
19+
not isExcluded(f, NoReturnPackage::functionWithNoReturningBranchShouldBeNoreturnQuery()) and
20+
not f instanceof NoreturnFunction and
21+
not mayReturn(f) and
22+
f.hasDefinition() and
23+
not f.getName() = "main" and // Allowed exception; _Noreturn main() is undefined behavior.
24+
// Harden against c++ cases.
25+
not f.isFromUninstantiatedTemplate(_) and
26+
not f.isDeleted() and
27+
not f.isCompilerGenerated()
28+
select f, "The function " + f.getName() + " cannot return and should be declared as _Noreturn."
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
/**
2+
* @id c/misra/return-statement-in-noreturn-function
3+
* @name RULE-17-9: Verify that a function declared with _Noreturn does not return
4+
* @description Returning inside a function declared with _Noreturn is undefined behavior.
5+
* @kind problem
6+
* @precision very-high
7+
* @problem.severity error
8+
* @tags external/misra/id/rule-17-9
9+
* correctness
10+
* external/misra/obligation/mandatory
11+
*/
12+
13+
import cpp
14+
import codingstandards.c.misra
15+
import codingstandards.cpp.rules.functionnoreturnattributecondition.FunctionNoReturnAttributeCondition
16+
17+
class ReturnStatementInNoreturnFunctionQuery extends FunctionNoReturnAttributeConditionSharedQuery {
18+
ReturnStatementInNoreturnFunctionQuery() {
19+
this = NoReturnPackage::returnStatementInNoreturnFunctionQuery()
20+
}
21+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
| test.c:6:15:6:16 | f4 | The function f4 is declared _noreturn but has a return type of int. |
2+
| test.c:19:15:19:16 | f8 | The function f8 is declared _noreturn but has a return type of int. |
3+
| test.c:24:17:24:18 | f9 | The function f9 is declared _noreturn but has a return type of void *. |
4+
| test.c:26:31:26:33 | f10 | The function f10 is declared _noreturn but has a return type of int. |
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
rules/RULE-17-10/NonVoidReturnTypeOfNoreturnFunction.ql

c/misra/test/rules/RULE-17-10/test.c

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
#include "stdlib.h"
2+
3+
void f1(); // COMPLIANT
4+
int f2(); // COMPLIANT
5+
_Noreturn void f3(); // COMPLIANT
6+
_Noreturn int f4(); // NON-COMPLIANT
7+
8+
void f5() { // COMPLIANT
9+
}
10+
11+
int f6() { // COMPLIANT
12+
return 0;
13+
}
14+
15+
_Noreturn void f7() { // COMPLIANT
16+
abort();
17+
}
18+
19+
_Noreturn int f8() { // NON-COMPLIANT
20+
abort();
21+
return 0;
22+
}
23+
24+
_Noreturn void *f9(); // NON-COMPLIANT
25+
26+
__attribute__((noreturn)) int f10(); // NON-COMPLIANT
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
| test.c:7:6:7:21 | test_noreturn_f2 | The function test_noreturn_f2 cannot return and should be declared as _Noreturn. |
2+
| test.c:18:6:18:21 | test_noreturn_f4 | The function test_noreturn_f4 cannot return and should be declared as _Noreturn. |
3+
| test.c:47:6:47:21 | test_noreturn_f8 | The function test_noreturn_f8 cannot return and should be declared as _Noreturn. |
4+
| test.c:63:6:63:22 | test_noreturn_f10 | The function test_noreturn_f10 cannot return and should be declared as _Noreturn. |
5+
| test.c:97:6:97:22 | test_noreturn_f15 | The function test_noreturn_f15 cannot return and should be declared as _Noreturn. |
6+
| test.c:101:6:101:22 | test_noreturn_f16 | The function test_noreturn_f16 cannot return and should be declared as _Noreturn. |
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
rules/RULE-17-11/FunctionWithNoReturningBranchShouldBeNoreturn.ql

c/misra/test/rules/RULE-17-11/test.c

+104
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
1+
#include "stdlib.h"
2+
3+
_Noreturn void test_noreturn_f1(int i) { // COMPLIANT
4+
abort();
5+
}
6+
7+
void test_noreturn_f2(int i) { // NON_COMPLIANT
8+
abort();
9+
}
10+
11+
_Noreturn void test_noreturn_f3(int i) { // COMPLIANT
12+
if (i > 0) {
13+
abort();
14+
}
15+
exit(1);
16+
}
17+
18+
void test_noreturn_f4(int i) { // NON_COMPLIANT
19+
if (i > 0) {
20+
abort();
21+
}
22+
exit(1);
23+
}
24+
25+
void test_noreturn_f5(int i) { // COMPLIANT
26+
if (i > 0) {
27+
return;
28+
}
29+
exit(1);
30+
}
31+
32+
void test_noreturn_f6(int i) { // COMPLIANT
33+
if (i > 0) {
34+
abort();
35+
}
36+
if (i < 0) {
37+
abort();
38+
}
39+
}
40+
41+
void test_noreturn_f7(int i) { // COMPLIANT
42+
if (i > 0) {
43+
abort();
44+
}
45+
}
46+
47+
void test_noreturn_f8(int i) { // NON_COMPLIANT
48+
if (i > 0) {
49+
abort();
50+
} else {
51+
abort();
52+
}
53+
}
54+
55+
_Noreturn void test_noreturn_f9(int i) { // COMPLIANT
56+
if (i > 0) {
57+
abort();
58+
} else {
59+
abort();
60+
}
61+
}
62+
63+
void test_noreturn_f10(int i) { // NON_COMPLIANT
64+
if (i > 0) {
65+
abort();
66+
}
67+
while (1) {
68+
i = 5;
69+
}
70+
}
71+
72+
_Noreturn void test_noreturn_f11(int i) { // COMPLIANT
73+
if (i > 0) {
74+
abort();
75+
}
76+
while (1) {
77+
i = 5;
78+
}
79+
}
80+
81+
void test_noreturn_f12(); // COMPLIANT
82+
83+
__attribute__((noreturn)) void test_noreturn_f13(int i) { // COMPLIANT
84+
abort();
85+
}
86+
87+
// Allowed by exception. It is undefined behavior for main() to be declared with
88+
// noreturn.
89+
int main(char **argv, int argc) { // COMPLIANT
90+
abort();
91+
}
92+
93+
_Noreturn void test_noreturn_f14(int i) { // COMPLIANT
94+
test_noreturn_f1(i);
95+
}
96+
97+
void test_noreturn_f15(int i) { // NON_COMPLIANT
98+
test_noreturn_f1(i);
99+
}
100+
101+
void test_noreturn_f16(int i) { // NON_COMPLIANT
102+
// Infinite tail recursion
103+
test_noreturn_f16(i);
104+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
c/common/test/rules/functionnoreturnattributecondition/FunctionNoReturnAttributeCondition.ql
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
- `A7-6-1`, `MSC53-CPP`, `RULE-9-6-4` - `FunctionNoReturnAttbrituteCondition.qll`
2+
- Analysis expanded from functions with "noreturn" attribute, now includes the "noreturn" specifier as well to handle new c rules. No difference in C++ results expected.
3+
- Exclude compiler generated functions from being reported.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
import cpp
2+
3+
/**
4+
* A function marked with _Noreturn or __attribute((noreturn))
5+
*/
6+
class NoreturnFunction extends Function {
7+
NoreturnFunction() {
8+
this.getASpecifier().getName() = "noreturn" or
9+
this.getAnAttribute().getName() = "noreturn"
10+
}
11+
}
12+
13+
/**
14+
* A function that may complete normally, and/or contains an explicit reachable
15+
* return statement.
16+
*/
17+
predicate mayReturn(Function function) {
18+
exists(ReturnStmt s |
19+
function = s.getEnclosingFunction() and
20+
s.getBasicBlock().isReachable()
21+
)
22+
}

0 commit comments

Comments
 (0)