MISRA rule 9.2 The initializer for an aggregate or union shall be enclosed in braces (#2899)
This commit is contained in:
parent
dcd7e3a24f
commit
4d3f76b63c
246
addons/misra.py
246
addons/misra.py
|
@ -1495,6 +1495,250 @@ class MisraChecker:
|
|||
if token.str == 'restrict':
|
||||
self.reportError(token, 8, 14)
|
||||
|
||||
def misra_9_2(self, data):
|
||||
# Holds information about a struct or union's element definition.
|
||||
class ElementDef:
|
||||
def __init__(self, elementType, name, valueType = None, dimensions = None):
|
||||
self.elementType = elementType
|
||||
self.name = name
|
||||
self.valueType = valueType
|
||||
self.dimensions = dimensions
|
||||
|
||||
# Return an array containing the size of each dimension of an array declaration,
|
||||
# or coordinates of a designator in an array initializer,
|
||||
# and the name token's valueType, if it exist.
|
||||
#
|
||||
# In the examples below, the ^ indicates the initial token passed to the function.
|
||||
#
|
||||
# Ex: int arr[1][2][3] = .....
|
||||
# ^
|
||||
# returns: [1,2,3], valueType
|
||||
#
|
||||
# Ex: int arr[3][4] = { [1][2] = 5 }
|
||||
# ^
|
||||
# returns [1,2], None
|
||||
def getArrayDimensionsAndValueType(token):
|
||||
dimensions = []
|
||||
while token and token.str == '[':
|
||||
if token.astOperand2 != None:
|
||||
dimensions.insert(0, token.astOperand2.getKnownIntValue())
|
||||
token = token.astOperand1
|
||||
elif token.astOperand1 != None:
|
||||
dimensions.insert(0, token.astOperand1.getKnownIntValue())
|
||||
break
|
||||
else:
|
||||
dimensions = None
|
||||
break
|
||||
|
||||
valueType = token.valueType if token else None
|
||||
|
||||
return dimensions, valueType
|
||||
|
||||
# Returns a list of the struct elements as StructElementDef in the order they are declared.
|
||||
def getRecordElements(valueType):
|
||||
if not valueType or not valueType.typeScope:
|
||||
return []
|
||||
|
||||
elements = []
|
||||
for variable in valueType.typeScope.varlist:
|
||||
if variable.isArray:
|
||||
dimensions, arrayValueType = getArrayDimensionsAndValueType(variable.nameToken.astParent)
|
||||
elements.append(ElementDef('array', variable.nameToken.str, arrayValueType, dimensions))
|
||||
elif variable.isClass:
|
||||
elements.append(ElementDef('class', variable.nameToken.str, variable.nameToken.valueType))
|
||||
else:
|
||||
elements.append(ElementDef('element', variable.nameToken.str))
|
||||
|
||||
return elements
|
||||
|
||||
# Checks if the initializer conforms to the dimensions of the array declaration
|
||||
# at a given level.
|
||||
# Parameters:
|
||||
# token: root node of the initializer tree
|
||||
# dimensions: dimension sizes of the array declaration
|
||||
# valueType: the array type
|
||||
def checkArrayInitializer(token, dimensions, valueType):
|
||||
level = 0
|
||||
levelOffsets = [] # Calculated when designators in initializers are used
|
||||
elements = getRecordElements(valueType) if valueType.type == 'record' else None
|
||||
|
||||
isFirstElement = False
|
||||
while token:
|
||||
if token.str == ',':
|
||||
token = token.astOperand1
|
||||
isFirstElement = False
|
||||
continue
|
||||
|
||||
if token.isAssignmentOp and not token.valueType:
|
||||
designator, _ = getArrayDimensionsAndValueType(token.astOperand1)
|
||||
# Calculate level offset based on designator in initializer
|
||||
levelOffsets[-1] = len(designator) - 1
|
||||
token = token.astOperand2
|
||||
isFirstElement = False
|
||||
|
||||
effectiveLevel = sum(levelOffsets) + level
|
||||
|
||||
isStringInitializer = token.isString and effectiveLevel == len(dimensions) - 1
|
||||
isZeroInitializer = (isFirstElement and token.str == '0')
|
||||
if effectiveLevel == len(dimensions) or isZeroInitializer or isStringInitializer:
|
||||
if isZeroInitializer or isStringInitializer:
|
||||
# Zero initializer is ok at any level
|
||||
# String initializer is ok at one level below value level
|
||||
pass
|
||||
else:
|
||||
isFirstElement = False
|
||||
if valueType.type == 'record':
|
||||
if token.isName:
|
||||
if not token.valueType.typeScope == valueType.typeScope:
|
||||
self.reportError(token, 9, 2)
|
||||
return False
|
||||
else:
|
||||
if not checkObjectInitializer(token, elements):
|
||||
return False
|
||||
elif token.str == '{' or token.isString:
|
||||
self.reportError(token, 9, 2)
|
||||
return False
|
||||
|
||||
while token:
|
||||
# Done checking once level is back to 0
|
||||
if level == 0:
|
||||
return True
|
||||
|
||||
if not token.astParent:
|
||||
return True
|
||||
|
||||
if token.astParent.astOperand1 == token and token.astParent.astOperand2:
|
||||
token = token.astParent.astOperand2
|
||||
break
|
||||
else:
|
||||
token = token.astParent
|
||||
if token.str == '{':
|
||||
level = level - 1
|
||||
levelOffsets.pop()
|
||||
effectiveLevel = sum(levelOffsets) + level
|
||||
|
||||
elif token.str == '{' :
|
||||
if not token.astOperand1:
|
||||
# Empty initializer
|
||||
self.reportError(token, 9, 2)
|
||||
return False
|
||||
|
||||
token = token.astOperand1
|
||||
level = level + 1
|
||||
levelOffsets.append(0)
|
||||
isFirstElement = True
|
||||
else:
|
||||
self.reportError(token, 9, 2)
|
||||
return False
|
||||
|
||||
return True
|
||||
|
||||
# Checks if the initializer conforms to the elements of the struct or union
|
||||
# Parameters:
|
||||
# token: root node of the initializer tree
|
||||
# elements: the elements as specified in the declaration
|
||||
def checkObjectInitializer(token, elements):
|
||||
if not token:
|
||||
return True
|
||||
|
||||
# Initializer must start with a curly bracket
|
||||
if not token.str == '{':
|
||||
self.reportError(token, 9, 2)
|
||||
return False
|
||||
|
||||
# Empty initializer is not ok { }
|
||||
if not token.astOperand1:
|
||||
self.reportError(token, 9, 2)
|
||||
return False
|
||||
|
||||
token = token.astOperand1
|
||||
|
||||
# Zero initializer is ok { 0 }
|
||||
if token.str == '0' :
|
||||
return True
|
||||
|
||||
pos = None
|
||||
while(token):
|
||||
if token.str == ',':
|
||||
token = token.astOperand1
|
||||
else:
|
||||
if pos == None:
|
||||
pos = 0
|
||||
|
||||
if token.isAssignmentOp:
|
||||
if token.astOperand1.str == '.':
|
||||
elementName = token.astOperand1.astOperand1.str
|
||||
pos = next((i for i, element in enumerate(elements) if element.name == elementName), len(elements))
|
||||
token = token.astOperand2
|
||||
|
||||
if pos >= len(elements):
|
||||
self.reportError(token, 9, 2)
|
||||
return False
|
||||
|
||||
element = elements[pos]
|
||||
if element.elementType == 'class':
|
||||
if token.isName:
|
||||
if not token.valueType.typeScope == element.valueType.typeScope:
|
||||
self.reportError(token, 9, 2)
|
||||
return False
|
||||
else:
|
||||
subElements = getRecordElements(element.valueType)
|
||||
if not checkObjectInitializer(token, subElements):
|
||||
return False
|
||||
elif element.elementType == 'array':
|
||||
if not checkArrayInitializer(token, element.dimensions, element.valueType):
|
||||
return False
|
||||
elif token.str == '{':
|
||||
self.reportError(token, 9, 2)
|
||||
return False
|
||||
|
||||
# The assignment represents the astOperand
|
||||
if token.astParent.isAssignmentOp:
|
||||
token = token.astParent
|
||||
|
||||
if not token == token.astParent.astOperand2:
|
||||
pos = pos + 1
|
||||
token = token.astParent.astOperand2
|
||||
else:
|
||||
token = None
|
||||
|
||||
return True
|
||||
|
||||
# ------
|
||||
for variable in data.variables:
|
||||
if not variable.nameToken:
|
||||
continue
|
||||
|
||||
nameToken = variable.nameToken
|
||||
|
||||
# Check if declaration and initialization is
|
||||
# split into two separate statements in ast.
|
||||
if nameToken.next and nameToken.next.isSplittedVarDeclEq:
|
||||
nameToken = nameToken.next.next
|
||||
|
||||
# Find declarations with initializer assignment
|
||||
eq = nameToken
|
||||
while not eq.isAssignmentOp and eq.astParent:
|
||||
eq = eq.astParent
|
||||
|
||||
if not eq.isAssignmentOp:
|
||||
continue
|
||||
|
||||
if variable.isArray :
|
||||
dimensions, valueType = getArrayDimensionsAndValueType(eq.astOperand1)
|
||||
if dimensions == None:
|
||||
continue
|
||||
|
||||
checkArrayInitializer(eq.astOperand2, dimensions, valueType)
|
||||
elif variable.isClass:
|
||||
if not nameToken.valueType:
|
||||
continue
|
||||
|
||||
valueType = nameToken.valueType
|
||||
if valueType.type == 'record':
|
||||
elements = getRecordElements(valueType)
|
||||
checkObjectInitializer(eq.astOperand2, elements)
|
||||
|
||||
def misra_9_5(self, rawTokens):
|
||||
for token in rawTokens:
|
||||
if simpleMatch(token, '[ ] = { ['):
|
||||
|
@ -3110,6 +3354,8 @@ class MisraChecker:
|
|||
self.executeCheck(812, self.misra_8_12, cfg)
|
||||
if cfgNumber == 0:
|
||||
self.executeCheck(814, self.misra_8_14, data.rawTokens)
|
||||
self.executeCheck(902, self.misra_9_2, cfg)
|
||||
if cfgNumber == 0:
|
||||
self.executeCheck(905, self.misra_9_5, data.rawTokens)
|
||||
self.executeCheck(1001, self.misra_10_1, cfg)
|
||||
self.executeCheck(1002, self.misra_10_2, cfg)
|
||||
|
|
|
@ -295,6 +295,66 @@ enum misra_8_12_e { misra_e1 = sizeof(int), misra_e2}; // no-crash
|
|||
|
||||
void misra_8_14(char * restrict str) {(void)str;} // 8.14
|
||||
|
||||
void misra_9_2() {
|
||||
int empty_init[2][2] = { }; // 9.2
|
||||
int empty_nested_init[2][2] = { { } }; // 9.2
|
||||
int zero_init_a[5] = { 0 };
|
||||
int zero_init_b[5][2] = { 0 };
|
||||
int zero_init_c[2][2] = { { 1, 2 }, { 0 } };
|
||||
|
||||
const char string_wrong_level_a[12] = { "Hello world" }; // 9.2
|
||||
const char string_wrong_level_b[2][20] = "Hello world"; // 9.2
|
||||
const char string_correct_level_a[] = "Hello world";
|
||||
const char string_correct_level_b[2][12] = { "Hello world" };
|
||||
|
||||
int array_init_incorrect_levels_a[3][2] = { 1, 2, 3, 4, 5, 6 }; // 9.2
|
||||
int array_init_correct_levels_a[3][2] = { { 1, 2 }, { 3, 4 }, { 5, 6 } };
|
||||
int array_init_incorrect_levels_b[6] = { { 1, 2 }, { 3, 4 }, { 5, 6 } }; // 9.2
|
||||
int array_init_correct_levels_b[6] = { 1, 2, 3, 4, 5, 6 };
|
||||
|
||||
int array_incorrect_designator_a[1] = { [0][1] = 1 }; // 9.2
|
||||
int array_incorrect_designator_b[1] = { [0] = { 1, 2 } }; // 9.2
|
||||
int array_incorrect_designator_c[1][2] = { [0] = 1 }; // 9.2
|
||||
int array_incorrect_designator_d[2][2] = { { 1, 2 }, [1][0] = {3, 4} }; // 9.2
|
||||
int array_correct_designator_a[2] = { [0] = 1, 2 };
|
||||
int array_correct_designator_b[2] = { [1] = 2, [0] = 1 };
|
||||
int array_correct_designator_c[2][2] = { { 1, 2 }, [1] = {3, 4} };
|
||||
int array_correct_designator_d[2][2] = { { 1, 2 }, [1][0] = 3, 4};
|
||||
|
||||
typedef struct {
|
||||
int i1;
|
||||
int i2;
|
||||
} struct1;
|
||||
|
||||
typedef struct {
|
||||
char c1;
|
||||
struct1 is1;
|
||||
char c2[4];
|
||||
} struct2;
|
||||
|
||||
int a;
|
||||
|
||||
struct2 struct_empty_init = { }; // 9.2
|
||||
struct2 struct_zero_init = { 0 };
|
||||
struct1 struct_missing_brackets = 1; // 9.2
|
||||
struct2 struct_correct_init = { 1, {2, 3}, {0} };
|
||||
struct1 struct_array_incorrect_levels[2] = { 1, 2, 3, 4 }; // 9.2
|
||||
struct1 struct_array_correct_levels[2] = { {1, 2}, {3, 4} };
|
||||
struct1 struct_correct_designator_a = { .i2 = 2, .i1 = 1 };
|
||||
struct2 struct_correct_designator_b = { .is1 = {2, 3}, { 4 } };
|
||||
struct1 struct_correct_designator_c = { a = 1, 2 }; // 13.1
|
||||
struct2 struct_incorrect_type = { .is1 = struct_correct_designator_b }; // 9.2
|
||||
struct2 struct_correct_type = { .is1 = struct_correct_designator_a };
|
||||
|
||||
struct1 struct_array_incorrect_type[1] = { struct_correct_designator_b }; // 9.2
|
||||
struct1 struct_array_correct_type[1] = { struct_correct_designator_a };
|
||||
|
||||
union misra_9_2_union { // 19.2
|
||||
char c;
|
||||
struct1 i;
|
||||
} u = { 3 }; // 19.2
|
||||
}
|
||||
|
||||
void misra_9_5() {
|
||||
int x[] = {[0]=23}; // 9.5
|
||||
}
|
||||
|
@ -1187,7 +1247,7 @@ void misra_18_8(int x) {
|
|||
int buf1[10];
|
||||
int buf2[sizeof(int)];
|
||||
int vla[x]; // 18.8
|
||||
static const unsigned char arr18_8_1[] = UNDEFINED_ID;
|
||||
static const unsigned char arr18_8_1[] = UNDEFINED_ID; // 9.2
|
||||
static uint32_t enum_test_0[R18_8_ENUM_CONSTANT_0] = {0};
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue