|
14 | 14 |
|
15 | 15 | import cpp
|
16 | 16 | import codingstandards.c.cert
|
| 17 | +import codingstandards.c.Objects |
17 | 18 | import codingstandards.cpp.Concurrency
|
18 | 19 | import semmle.code.cpp.dataflow.DataFlow
|
19 | 20 | import semmle.code.cpp.commons.Alloc
|
20 | 21 |
|
21 |
| -from C11ThreadCreateCall tcc, StackVariable sv, Expr arg, Expr acc |
| 22 | +from C11ThreadCreateCall tcc, Expr arg |
22 | 23 | where
|
23 | 24 | not isExcluded(tcc, Concurrency4Package::appropriateThreadObjectStorageDurationsQuery()) and
|
24 | 25 | tcc.getArgument(2) = arg and
|
25 |
| - sv.getAnAccess() = acc and |
26 |
| - // a stack variable that is given as an argument to a thread |
27 |
| - TaintTracking::localTaint(DataFlow::exprNode(acc), DataFlow::exprNode(arg)) and |
28 |
| - // or isn't one of the allowed usage patterns |
29 |
| - not exists(Expr mfc | |
30 |
| - isAllocationExpr(mfc) and |
31 |
| - sv.getAnAssignedValue() = mfc and |
32 |
| - acc.getAPredecessor*() = mfc |
33 |
| - ) and |
34 |
| - not exists(TSSGetFunctionCall tsg, TSSSetFunctionCall tss, DataFlow::Node src | |
35 |
| - sv.getAnAssignedValue() = tsg and |
36 |
| - acc.getAPredecessor*() = tsg and |
37 |
| - // there should be dataflow from somewhere (the same somewhere) |
38 |
| - // into each of the first arguments |
39 |
| - DataFlow::localFlow(src, DataFlow::exprNode(tsg.getArgument(0))) and |
40 |
| - DataFlow::localFlow(src, DataFlow::exprNode(tss.getArgument(0))) |
| 26 | + ( |
| 27 | + exists(ObjectIdentity obj, Expr acc | |
| 28 | + obj.getASubobjectAccess() = acc and |
| 29 | + obj.getStorageDuration().isAutomatic() and |
| 30 | + exists(DataFlow::Node addrNode | |
| 31 | + ( |
| 32 | + addrNode = DataFlow::exprNode(any(AddressOfExpr e | e.getOperand() = acc)) |
| 33 | + or |
| 34 | + addrNode = DataFlow::exprNode(acc) and |
| 35 | + exists(ArrayToPointerConversion c | c.getExpr() = acc) |
| 36 | + ) and |
| 37 | + TaintTracking::localTaint(addrNode, DataFlow::exprNode(arg)) |
| 38 | + ) |
| 39 | + ) |
| 40 | + or |
| 41 | + // TODO: This case is handling threadlocals in a useful way that's not intended to be covered |
| 42 | + // by the rule. See issue #801. The actual rule should expect no tss_t objects is used, and |
| 43 | + // this check that this is initialized doesn't seem to belong here. However, it is a useful |
| 44 | + // check in and of itself, so we should figure out if this is part of an optional rule we |
| 45 | + // haven't yet implemented and move this behavior there. |
| 46 | + exists(TSSGetFunctionCall tsg | |
| 47 | + TaintTracking::localTaint(DataFlow::exprNode(tsg), DataFlow::exprNode(arg)) and |
| 48 | + not exists(TSSSetFunctionCall tss, DataFlow::Node src | |
| 49 | + // there should be dataflow from somewhere (the same somewhere) |
| 50 | + // into each of the first arguments |
| 51 | + DataFlow::localFlow(src, DataFlow::exprNode(tsg.getArgument(0))) and |
| 52 | + DataFlow::localFlow(src, DataFlow::exprNode(tss.getArgument(0))) |
| 53 | + ) |
| 54 | + ) |
41 | 55 | )
|
42 | 56 | select tcc, "$@ not declared with appropriate storage duration", arg, "Shared object"
|
0 commit comments