1 /* valacastexpression.vala
3 * Copyright (C) 2006-2011 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>
25 * Represents a type cast in the source code.
27 public class Vala
.CastExpression
: Expression
{
29 * The expression to be cast.
31 public Expression inner
{
37 _inner
.parent_node
= this
;
44 public DataType type_reference
{
45 get { return _data_type
; }
48 _data_type
.parent_node
= this
;
53 * Checked casts return NULL instead of raising an error.
55 public bool is_silent_cast
{ get; private set; }
57 public bool is_non_null_cast
{ get; private set; }
59 private Expression _inner
;
61 private DataType _data_type
;
64 * Creates a new cast expression.
66 * @param inner expression to be cast
67 * @param type_reference target type
68 * @return newly created cast expression
70 public CastExpression (Expression inner
, DataType type_reference
, SourceReference source_reference
) {
71 this
.type_reference
= type_reference
;
72 this
.source_reference
= source_reference
;
73 this
.is_silent_cast
= false;
74 this
.is_non_null_cast
= false;
78 public CastExpression
.silent (Expression inner
, DataType type_reference
, SourceReference source_reference
) {
79 this
.type_reference
= type_reference
;
80 this
.source_reference
= source_reference
;
81 this
.is_silent_cast
= true;
82 this
.is_non_null_cast
= false;
86 public CastExpression
.non_null (Expression inner
, SourceReference source_reference
) {
88 this
.is_non_null_cast
= true;
89 this
.source_reference
= source_reference
;
92 public override void accept (CodeVisitor visitor
) {
93 visitor
.visit_cast_expression (this
);
95 visitor
.visit_expression (this
);
98 public override void accept_children (CodeVisitor visitor
) {
99 inner
.accept (visitor
);
100 if (!is_non_null_cast
) {
101 type_reference
.accept (visitor
);
105 public override void replace_expression (Expression old_node
, Expression new_node
) {
106 if (inner
== old_node
) {
111 public override bool is_pure () {
112 return inner
.is_pure ();
115 public override bool is_accessible (Symbol sym
) {
116 return inner
.is_accessible (sym
);
119 public override void replace_type (DataType old_type
, DataType new_type
) {
120 if (type_reference
== old_type
) {
121 type_reference
= new_type
;
125 public override bool check (CodeContext context
) {
132 if (!inner
.check (context
)) {
137 if (inner
.value_type
== null) {
138 Report
.error (source_reference
, "Invalid cast expression");
143 if (is_non_null_cast
) {
145 type_reference
= inner
.value_type
.copy ();
146 type_reference
.nullable
= false;
149 type_reference
.check (context
);
151 // FIXME: check whether cast is allowed
153 if (type_reference is DelegateType
&& inner
.value_type is MethodType
) {
154 if (target_type
!= null) {
155 inner
.value_type
.value_owned
= target_type
.value_owned
;
157 inner
.value_type
.value_owned
= true;
161 value_type
= type_reference
;
162 value_type
.value_owned
= inner
.value_type
.value_owned
;
163 value_type
.floating_reference
= inner
.value_type
.floating_reference
;
165 if (is_silent_cast
) {
166 value_type
.nullable
= true;
169 if (is_gvariant (context
, inner
.value_type
) && !is_gvariant (context
, value_type
)) {
170 // GVariant unboxing returns owned value
171 value_type
.value_owned
= true;
174 inner
.target_type
= inner
.value_type
.copy ();
179 bool is_gvariant (CodeContext context
, DataType type
) {
180 return type
.data_type
!= null && type
.data_type
.is_subtype_of (context
.analyzer
.gvariant_type
.data_type
);
183 public override void emit (CodeGenerator codegen
) {
184 inner
.emit (codegen
);
186 codegen
.visit_cast_expression (this
);
188 codegen
.visit_expression (this
);
191 public override void get_defined_variables (Collection
<Variable
> collection
) {
192 inner
.get_defined_variables (collection
);
195 public override void get_used_variables (Collection
<Variable
> collection
) {
196 inner
.get_used_variables (collection
);
199 public override bool is_constant () {
200 return inner
.is_constant ();