Section 18.104.22.168 of the standard defines four name spaces for identifiers in C, and therefore we use four instances of the necessary environment modules:
(The unnamed instances implement the name space of ``ordinary identifiers''. An ordinary identifier is any identifier that does not belong to one of the other name spaces.)
Label identifiers obey ALGOL-like scope rules because Section 22.214.171.124 of the standard states explicitly that the scope of a label identifier is the entire function in which it appears.
Section 126.96.36.199 of the standard describes the behavior of tags, and although that description is couched in terms of the sequence in which constructs appear in the program, the overall effect is that tags obey ALGOL-like scope rules.
Member identifier declarations do not interact with any other identifier occurrences in their scopes. Similarly, applied occurrences of member identifiers don't interact with any other identifier occurrences. Thus member identifiers could be considered to obey either ALGOL-like or C-like scope rules. We chose to use ALGOL-like scope rules for compatibility with the ScopeProp module.
Ordinary identifiers obey C-like scope rules, in which the scope begins with the declaration of the identifier and ends at the end of some source language construct.
Section 188.8.131.52 of the standard defines the points at which the scopes of various identifiers terminate, and then states that two identifiers have the same scope if and only if their scopes terminate at the same point. Certain symbols of Section 2's abstract syntax tree correspond to constructs ending at those termination points and encompassing all of the possible points at which the scope of an identifier could begin. Each of these symbols inherits the RangeScope role of the corresponding name space module.
Identifiers appearing within the list of parameter declarations in a function definition have block scope terminating at the } that closes the function body. Although the parameter list and the function body are represented by nodes that have a common parent (function_definition), there are intervening nodes. This requires use of the range sequence module, with function_definition inheriting the RangeSequence role and the parameter list and function body each inheriting the RangeElement role. The definition of the range sequence module requires that RangeElement be inherited by a node belonging to a subtree rooted in a node that inherits RangeSequence.
The parameters node, which represents the parameter list of a function definition, is a component of a declarator, which occurs in contexts other than function definitions. When declarator occurs a context other than a function definition, parameters represents a function prototype scope. There is no practical way to make a syntactic distinction between the use of a declarator in a function definition and the use of a declarator in other contexts.
Since parameters must inherit RangeElement in order to implement the block scope of a function definition, it must always appear in a subtree rooted in a node inheriting RangeSequence. Thus several other nodes that would not normally be considered to represent constructs significant for name analysis inherit that role.