1 | From the refrat (5.5) we have our specification:
|
---|
2 |
|
---|
3 | \section{Initialization} An expression that is used as an
|
---|
4 | \nonterm{initializer} is treated as being cast to the type of the
|
---|
5 | object being initialized. An expression used in an
|
---|
6 | \nonterm{initializer-list} is treated as being cast to the type of
|
---|
7 | the aggregate member that it initializes. In either case the cast
|
---|
8 | must have a single unambiguous
|
---|
9 | interpretation\index{interpretations}.
|
---|
10 |
|
---|
11 | Steps:
|
---|
12 |
|
---|
13 | - add a member function "void Resolver::visit( SynTree::DeclStmt
|
---|
14 | *declStmt )"; for each DeclStmt:
|
---|
15 |
|
---|
16 | - do what you need to do to establish correspondences between
|
---|
17 | expressions in the initializer and pieces of the object to be
|
---|
18 | initialized
|
---|
19 |
|
---|
20 | - for each initializer expression, construct a cast expression that
|
---|
21 | casts the value of the expression to the type of the corresponding
|
---|
22 | sub-object
|
---|
23 |
|
---|
24 | - invoke the resolver recursively on each cast expression; it's an invariant
|
---|
25 | of the resolver that attempting to resolve a cast expression results either
|
---|
26 | in a single resolved expression (corresponding to the unambiguous interpretation
|
---|
27 | referred to above) or a thrown SemanticError.
|
---|
28 |
|
---|
29 | - construct a new initializer from the resolved expressions
|
---|
30 |
|
---|
31 | You'll undoubtedly have to play with the CodeGen stuff a bit; I
|
---|
32 | hacked it to spit out unresolved initializers for file-scope
|
---|
33 | declarations so that real programs would compile. You'll want to make
|
---|
34 | sure that resolved initializers for all declarations are being
|
---|
35 | generated.
|
---|
36 |
|
---|
37 |
|
---|
38 | ------
|
---|
39 |
|
---|
40 | More recent email: (I am quoted; Richard is the responder)
|
---|
41 | > As far as I'm aware, the only way that I could currently get the correct
|
---|
42 | > results from the unification engine is by feeding it an expression that
|
---|
43 | > looks like "?=?( ((struct Y)x.y).a, 10 )", then picking out the pieces that
|
---|
44 | > I need (namely the correct choice for a). Does this seem like a reasonable
|
---|
45 | > approach to solve this problem?
|
---|
46 |
|
---|
47 | No, unfortunately. Initialization isn't being rewritten as assignment,
|
---|
48 | so you shouldn't allow the particular selection of assignment
|
---|
49 | operators that happen to be in scope (and which may include
|
---|
50 | user-defined operators) to guide the type resolution.
|
---|
51 |
|
---|
52 | I don't think there is any way to rewrite an initializer as a single
|
---|
53 | expression and have the resolver just do the right thing. I see the
|
---|
54 | algorithm as:
|
---|
55 |
|
---|
56 | For each alternative interpretation of the designator:
|
---|
57 | Construct an expression that casts the initializer to the type of
|
---|
58 | the designator
|
---|
59 | Construct an AlternativeFinder and use it to find the lowest cost
|
---|
60 | interpretation of the expression
|
---|
61 | Add this interpretation to a list of possibilities
|
---|
62 | Go through the list of possibilities and pick the lowest cost
|
---|
63 |
|
---|
64 | As with many things in the resolver, it's conceptually simple but the
|
---|
65 | implementation may be a bit of a pain. It fits in with functions like
|
---|
66 | findSingleExpression, findIntegralExpression in Resolver.cc, although
|
---|
67 | it will be significantly more complicated than any of the existing
|
---|
68 | ones.
|
---|
69 |
|
---|
70 |
|
---|
71 |
|
---|