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>
26 * Represents a type member with a constant value.
28 public class Vala
.Constant
: Symbol
, Lockable
{
30 * The data type of this constant.
32 public DataType type_reference
{
33 get { return _data_type
; }
36 _data_type
.parent_node
= this
;
41 * The value of this constant.
43 public Expression? value
{
44 get { return _value
; }
48 _value
.parent_node
= this
;
53 private bool lock_used
= false;
55 private DataType _data_type
;
57 private Expression _value
;
60 * Creates a new constant.
62 * @param name constant name
63 * @param type_reference constant type
64 * @param value constant value
65 * @param source_reference reference to source code
66 * @return newly created constant
68 public Constant (string name
, DataType? type_reference
, Expression? value
, SourceReference? source_reference
, Comment? comment
= null) {
69 base (name
, source_reference
, comment
);
70 if (type_reference
!= null) {
71 this
.type_reference
= type_reference
;
76 public override void accept (CodeVisitor visitor
) {
77 visitor
.visit_constant (this
);
80 public override void accept_children (CodeVisitor visitor
) {
81 type_reference
.accept (visitor
);
84 value
.accept (visitor
);
88 public bool get_lock_used () {
92 public void set_lock_used (bool used
) {
96 public override void replace_expression (Expression old_node
, Expression new_node
) {
97 if (value
== old_node
) {
102 public override void replace_type (DataType old_type
, DataType new_type
) {
103 if (type_reference
== old_type
) {
104 type_reference
= new_type
;
108 public override bool check (CodeContext context
) {
115 var old_source_file
= context
.analyzer
.current_source_file
;
116 var old_symbol
= context
.analyzer
.current_symbol
;
118 if (source_reference
!= null) {
119 context
.analyzer
.current_source_file
= source_reference
.file
;
121 if (!(parent_symbol is Block
)) {
122 // non-local constant
123 context
.analyzer
.current_symbol
= this
;
126 type_reference
.check (context
);
128 if (!check_const_type (type_reference
, context
)) {
130 Report
.error (source_reference
, "`%s' not supported as type for constants".printf (type_reference
.to_string ()));
137 Report
.error (source_reference
, "A const field requires a value to be provided");
139 value
.target_type
= type_reference
;
141 if (!value
.check (context
)) {
146 if (!value
.value_type
.compatible (type_reference
)) {
148 Report
.error (source_reference
, "Cannot convert from `%s' to `%s'".printf (value
.value_type
.to_string (), type_reference
.to_string ()));
152 // support translated string constants for efficiency / convenience
153 // even though the expression is not a compile-time constant
154 var call
= value as MethodCall
;
156 var method_type
= call
.call
.value_type as MethodType
;
157 if (method_type
!= null && method_type
.method_symbol
.get_full_name () == "GLib._") {
158 // first argument is string
159 var literal
= call
.get_argument_list ().get (0) as StringLiteral
;
160 if (literal
!= null) {
162 literal
.translate
= true;
167 if (!value
.is_constant ()) {
169 Report
.error (value
.source_reference
, "Value must be constant");
176 Report
.error (source_reference
, "External constants cannot use values");
180 if (!external_package
&& !hides
&& get_hidden_member () != null) {
181 Report
.warning (source_reference
, "%s hides inherited constant `%s'. Use the `new' keyword if hiding was intentional".printf (get_full_name (), get_hidden_member ().get_full_name ()));
184 context
.analyzer
.current_source_file
= old_source_file
;
185 context
.analyzer
.current_symbol
= old_symbol
;
192 bool check_const_type (DataType type
, CodeContext context
) {
193 if (type is ValueType
) {
195 } else if (type is ArrayType
) {
196 var array_type
= type as ArrayType
;
197 return check_const_type (array_type
.element_type
, context
);
198 } else if (type
.data_type
.is_subtype_of (context
.analyzer
.string_type
.data_type
)) {