1 /* valaarraycreationexpression.vala
3 * Copyright (C) 2006-2009 Jürg Billeter
4 * Copyright (C) 2006-2008 Raffaele Sandrini
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21 * Raffaele Sandrini <raffaele@sandrini.ch>
22 * Jürg Billeter <j@bitron.ch>
29 * Represents an array creation expression e.g. "new int[] {1,2,3}".
31 public class Vala
.ArrayCreationExpression
: Expression
{
33 * The type of the elements of the array.
35 public DataType element_type
{
36 get { return _element_type
; }
38 _element_type
= value
;
39 _element_type
.parent_node
= this
;
44 * The rank of the array.
46 public int rank
{ get; set; }
49 * The size for each dimension ascending from left to right.
51 private Gee
.List
<Expression
> sizes
= new ArrayList
<Expression
> ();
54 * The root array initializer list.
56 public InitializerList? initializer_list
{
57 get { return _initializer_list
; }
59 _initializer_list
= value
;
60 if (_initializer_list
!= null) {
61 _initializer_list
.parent_node
= this
;
66 private DataType _element_type
;
67 private InitializerList? _initializer_list
;
70 * Add a size expression.
72 public void append_size (Expression size
) {
77 * Get the sizes for all dimensions ascending from left to right.
79 public Gee
.List
<Expression
> get_sizes () {
80 return new ReadOnlyList
<Expression
> (sizes
);
83 public ArrayCreationExpression (DataType element_type
, int rank
, InitializerList? initializer_list
, SourceReference source_reference
) {
84 this
.element_type
= element_type
;
86 this
.initializer_list
= initializer_list
;
87 this
.source_reference
= source_reference
;
90 public override void accept_children (CodeVisitor visitor
) {
91 if (element_type
!= null) {
92 element_type
.accept (visitor
);
95 foreach (Expression e
in sizes
) {
99 if (initializer_list
!= null) {
100 initializer_list
.accept (visitor
);
104 public override void accept (CodeVisitor visitor
) {
105 visitor
.visit_array_creation_expression (this
);
107 visitor
.visit_expression (this
);
110 public override bool is_pure () {
114 public override void replace_type (DataType old_type
, DataType new_type
) {
115 if (element_type
== old_type
) {
116 element_type
= new_type
;
120 private int create_sizes_from_initializer_list (SemanticAnalyzer analyzer
, InitializerList il
, int rank
, Gee
.List
<Literal
> sl
) {
121 if (sl
.size
== (this
.rank
- rank
)) {
122 // only add size if this is the first initializer list of the current dimension
123 var init
= new
IntegerLiteral (il
.size
.to_string (), il
.source_reference
);
124 init
.check (analyzer
);
129 foreach (Expression e
in il
.get_initializers ()) {
130 if (e is InitializerList
) {
134 Report
.error (e
.source_reference
, "Expected array element, got array initializer list");
137 int size
= create_sizes_from_initializer_list (analyzer
, (InitializerList
) e
, rank
- 1, sl
);
141 if (subsize
>= 0 && subsize
!= size
) {
143 Report
.error (il
.source_reference
, "Expected initializer list of size %d, got size %d".printf (subsize
, size
));
152 Report
.error (e
.source_reference
, "Expected array initializer list, got array element");
160 public override bool check (SemanticAnalyzer analyzer
) {
167 Gee
.List
<Expression
> sizes
= get_sizes ();
168 var initlist
= initializer_list
;
170 if (element_type
!= null) {
171 element_type
.check (analyzer
);
174 foreach (Expression e
in sizes
) {
178 var calc_sizes
= new ArrayList
<Literal
> ();
179 if (initlist
!= null) {
180 initlist
.target_type
= new
ArrayType (element_type
, rank
, source_reference
);
182 initlist
.check (analyzer
);
184 var ret
= create_sizes_from_initializer_list (analyzer
, initlist
, rank
, calc_sizes
);
190 if (sizes
.size
> 0) {
191 /* check for errors in the size list */
192 foreach (Expression e
in sizes
) {
193 if (e
.value_type
== null) {
194 /* return on previous error */
196 } else if (!e
.value_type
.compatible (analyzer
.uint64_type
)) {
198 Report
.error (e
.source_reference
, "Expression of integer type expected");
202 if (initlist
== null) {
204 /* this is an internal error because it is already handeld by the parser */
205 Report
.error (source_reference
, "internal error: initializer list expected");
207 foreach (Expression size
in calc_sizes
) {
217 /* check for wrong elements inside the initializer */
218 if (initializer_list
!= null && initializer_list
.value_type
== null) {
222 /* try to construct the type of the array */
223 if (element_type
== null) {
225 Report
.error (source_reference
, "Cannot determine the element type of the created array");
229 element_type
.value_owned
= true;
231 value_type
= new
ArrayType (element_type
, rank
, source_reference
);
232 value_type
.value_owned
= true;