CertC++-INT50¶
Do not cast to an out-of-range enumeration value
Required inputs: IR, StaticSemanticAnalysis
Enumerations in C++ come in two forms: scoped enumerations in which the underlying type is fixed and unscoped enumerations in which the underlying type may or may not be fixed. The range of values that can be represented by either form of enumeration may include enumerator values not specified by the enumeration itself. The range of valid enumeration values for an enumeration type is defined by the C++ Standard, [dcl.enum], in paragraph 8 [ ISO/IEC 14882-2020]:
For an enumeration whose underlying type is fixed, the values of the enumeration are the values of the underlying type. Otherwise, the values of the enumeration are the values representable by a hypothetical integer type with minimal width M such that all enumerators can be represented. The width of the smallest bit-field large enough to hold all the values of the enumeration type is M. It is possible to define an enumeration that has values not defined by any of its enumerators. If the enumerator-list is empty, the values of the enumeration are as if the enumeration had a single enumerator with value 0.
The C++ Standard, [expr.static.cast], paragraph 10, states the following:
A value of integral or enumeration type can be explicitly converted to a complete enumeration type. If the enumeration type has a fixed underlying type, the value is first converted to that type by integral conversion, if necessary, and then to the enumeration type. If the enumeration type does not have a fixed underlying type, the value is unchanged if the original value is within the range of the enumeration values (9.7.1), and otherwise, the behavior is undefined. A value of floating-point type can also be explicitly converted to an enumeration type. The resulting value is the same as converting the original value to the underlying type of the enumeration (7.3.10), and subsequently to the enumeration type.
To avoid operating on unspecified values, the arithmetic value being cast must be within the range of values the enumeration can represent. When dynamically checking for out-of-range values, checking must be performed before the cast expression.
Noncompliant Code Example (Bounds Checking)
This noncompliant code example attempts to check whether a given value is
within the range of acceptable enumeration values. However, it is doing so
after casting to the enumeration type, which may not be able to represent the
given integer value. On a two's complement system, the valid range of values
that can be represented by
EnumType are [0..3], so if a value outside of that range were
passed to
f(), the cast to
EnumType would result in an unspecified value, and using that
value within the
if statement results in
unspecified
behavior.
enum EnumType {
First,
Second,
Third
};
void f(int intVar) {
EnumType enumVar = static_cast<EnumType>(intVar);
if (enumVar < First || enumVar > Third) {
// Handle error
}
}
Compliant Solution (Bounds Checking)
This compliant solution checks that the value can be represented by the enumeration type before performing the conversion to guarantee the conversion does not result in an unspecified value. It does this by restricting the converted value to one for which there is a specific enumerator value.
enum EnumType {
First,
Second,
Third
};
void f(int intVar) {
if (intVar < First || intVar > Third) {
// Handle error
}
EnumType enumVar = static_cast<EnumType>(intVar);
}
Compliant Solution (Scoped Enumeration)
This compliant solution uses a scoped enumeration, which has a fixed underlying
int type by default, allowing any value from the parameter to be
converted into a valid enumeration value. It does not restrict the converted
value to one for which there is a specific enumerator value, but it could do so
as shown in the previous compliant solution.
enum class EnumType {
First,
Second,
Third
};
void f(int intVar) {
EnumType enumVar = static_cast<EnumType>(intVar);
}
Compliant Solution (Fixed Unscoped Enumeration)
Similar to the previous compliant solution, this compliant solution uses an
unscoped enumeration but provides a fixed underlying
int type allowing any value from the parameter to be converted to
a valid enumeration value.
enum EnumType : int {
First,
Second,
Third
};
void f(int intVar) {
EnumType enumVar = static_cast<EnumType>(intVar);
}
Although similar to the previous compliant solution, this compliant solution
differs from the noncompliant code example in the way the enumerator values are
expressed in code and which implicit conversions are allowed. The previous
compliant solution requires a nested name specifier to identify the enumerator
(for example,
EnumType::First) and will not implicitly convert the enumerator
value to
int. As with the noncompliant code example, this compliant
solution does not allow a nested name specifier and will implicitly convert the
enumerator value to
int.
Risk Assessment
It is possible for unspecified values to result in a buffer overflow, leading to the execution of arbitrary code by an attacker. However, because enumerators are rarely used for indexing into arrays or other forms of pointer arithmetic, it is more likely that this scenario will result in data integrity violations rather than arbitrary code execution.
| Rule | Severity | Likelihood | Remediation Cost | Priority | Level |
|---|---|---|---|---|---|
| INT50-CPP | Medium | Unlikely | Medium | P4 | L3 |
Bibliography
| [ Becker 2009] | Section 7.2, "Enumeration Declarations" |
| [ ISO/IEC 14882-2020] | Subclause 5.2.9, "Static Cast" Subclause 7.2, "Enumeration Declarations" |
Possible Messages
Key |
Text |
Severity |
Disabled |
|---|---|---|---|
asgn_interval_not_an_enumerator |
Possible assignment of value which does not correspond to an enumerator (values computed by the analysis are [{int1};{int2}], they contain e.g. {int0} which does not correspond to an enumerator) |
None |
False |
asgn_interval_out_of_enum_range |
Possible assignment of value outside the enumerators (values computed by the analysis are [{int0};{int1}]) |
None |
False |
asgn_not_an_enumerator |
Assignment of value {int0} which does not correspond to an enumerator |
None |
False |
asgn_out_of_enum_range |
Assignment of value {int0} outside the range of the enumerators |
None |
False |
conversion_creating_bad_enum_value |
Expression does not correspond to an enumerator in {} |
None |
False |
Options¶
This rule shares the following common options: exclude_in_macros, exclude_messages_in_system_headers, excludes, extend_exclude_to_macro_invocations, includes, justification_checker, languages, post_processing, provider, report_at, severity
The following places define options that affect this rule: Stylechecks, Analysis-GlobalOptions
any_enum¶
any_enum : bool = False
enum_kinds¶
enum_kinds : EnumKinds = 'Unscoped_Unbased'
with_predicate_analysis¶
with_predicate_analysis : bool = True
Option Types¶
These types are used by options listed above:
EnumKinds¶
An enumeration.Any
Scoped
Unscoped_Unbased