1 /* valaswitchstatement.vala
3 * Copyright (C) 2006-2010 Jürg Billeter
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20 * Jürg Billeter <j@bitron.ch>
26 * Represents a switch selection statement in the source code.
28 public class Vala
.SwitchStatement
: CodeNode
, Statement
{
30 * Specifies the switch expression.
32 public Expression expression
{
38 _expression
.parent_node
= this
;
42 private Expression _expression
;
43 private List
<SwitchSection
> sections
= new ArrayList
<SwitchSection
> ();
46 * Creates a new switch statement.
48 * @param expression switch expression
49 * @param source_reference reference to source code
50 * @return newly created switch statement
52 public SwitchStatement (Expression expression
, SourceReference? source_reference
) {
53 this
.source_reference
= source_reference
;
54 this
.expression
= expression
;
58 * Appends the specified section to the list of switch sections.
60 * @param section a switch section
62 public void add_section (SwitchSection section
) {
63 section
.parent_node
= this
;
64 sections
.add (section
);
68 * Returns a copy of the list of switch sections.
70 * @return section list
72 public List
<SwitchSection
> get_sections () {
76 public override void accept (CodeVisitor visitor
) {
77 visitor
.visit_switch_statement (this
);
80 public override void accept_children (CodeVisitor visitor
) {
81 expression
.accept (visitor
);
83 visitor
.visit_end_full_expression (expression
);
85 foreach (SwitchSection section
in sections
) {
86 section
.accept (visitor
);
90 public override void replace_expression (Expression old_node
, Expression new_node
) {
91 if (expression
== old_node
) {
92 expression
= new_node
;
96 public override bool check (SemanticAnalyzer analyzer
) {
103 if (!expression
.check (analyzer
)) {
108 if (expression
.value_type
== null ||
109 (!(expression
.value_type is IntegerType
) &&
110 !(expression
.value_type is EnumValueType
) &&
111 !expression
.value_type
.compatible (analyzer
.string_type
))) {
112 Report
.error (expression
.source_reference
, "Integer or string expression expected");
117 // ensure that possibly owned (string) expression stays alive
118 expression
.target_type
= expression
.value_type
.copy ();
120 var labelset
= new HashSet
<string> (str_hash
, str_equal
);
121 foreach (SwitchSection section
in sections
) {
122 section
.check (analyzer
);
124 // check for duplicate literal case labels
125 // FIXME: make it work for all constant expressions
126 foreach (SwitchLabel label
in section
.get_labels ()) {
127 string? value
= null;
128 if (label
.expression is StringLiteral
) {
129 value
= ((StringLiteral
)label
.expression
).eval ();
130 } else if (label
.expression is Literal
) {
131 value
= ((Literal
)label
.expression
).to_string ();
134 if (value
!= null && !labelset
.add (value
)) {
136 Report
.error (label
.expression
.source_reference
, "Switch statement already contains this label");
139 add_error_types (section
.get_error_types ());
145 public override void emit (CodeGenerator codegen
) {
146 expression
.emit (codegen
);
148 codegen
.visit_end_full_expression (expression
);
150 codegen
.visit_switch_statement (this
);