blob: 4617bf8d71b5e2f283f620f91a539525e4e8c184 [file] [log] [blame]
David Reissea2cba82009-03-30 21:35:00 +00001/*
2 * Licensed to the Apache Software Foundation (ASF) under one
3 * or more contributor license agreements. See the NOTICE file
4 * distributed with this work for additional information
5 * regarding copyright ownership. The ASF licenses this file
6 * to you under the Apache License, Version 2.0 (the
7 * "License"); you may not use this file except in compliance
8 * with the License. You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing,
13 * software distributed under the License is distributed on an
14 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15 * KIND, either express or implied. See the License for the
16 * specific language governing permissions and limitations
17 * under the License.
18 */
Mark Sleee9ce01c2007-05-16 02:29:53 +000019
Mark Sleef0712dc2006-10-25 19:03:57 +000020#ifndef T_SCOPE_H
21#define T_SCOPE_H
22
23#include <map>
24#include <string>
David Reissaca320d2010-08-31 16:51:26 +000025#include <sstream>
Mark Sleef0712dc2006-10-25 19:03:57 +000026
27#include "t_type.h"
28#include "t_service.h"
Bryan Duxbury2d804702009-12-18 19:41:11 +000029#include "t_const.h"
30#include "t_const_value.h"
31#include "t_base_type.h"
32#include "t_map.h"
33#include "t_list.h"
Mark Sleef0712dc2006-10-25 19:03:57 +000034
35/**
36 * This represents a variable scope used for looking up predefined types and
37 * services. Typically, a scope is associated with a t_program. Scopes are not
38 * used to determine code generation, but rather to resolve identifiers at
39 * parse time.
40 *
Mark Sleef0712dc2006-10-25 19:03:57 +000041 */
42class t_scope {
43 public:
44 t_scope() {}
45
46 void add_type(std::string name, t_type* type) {
47 types_[name] = type;
48 }
49
50 t_type* get_type(std::string name) {
51 return types_[name];
52 }
53
54 void add_service(std::string name, t_service* service) {
55 services_[name] = service;
56 }
57
58 t_service* get_service(std::string name) {
59 return services_[name];
60 }
61
Mark Sleed0767c52007-07-27 22:14:41 +000062 void add_constant(std::string name, t_const* constant) {
Bryan Duxbury8fc413f2010-08-05 23:23:04 +000063 if (constants_.find(name) != constants_.end()) {
64 throw "Enum " + name + " is already defined!";
65 } else {
66 constants_[name] = constant;
67 }
68
Mark Sleed0767c52007-07-27 22:14:41 +000069 }
70
71 t_const* get_constant(std::string name) {
72 return constants_[name];
73 }
74
Mark Sleef0712dc2006-10-25 19:03:57 +000075 void print() {
76 std::map<std::string, t_type*>::iterator iter;
77 for (iter = types_.begin(); iter != types_.end(); ++iter) {
78 printf("%s => %s\n",
79 iter->first.c_str(),
80 iter->second->get_name().c_str());
81 }
82 }
83
Bryan Duxbury2d804702009-12-18 19:41:11 +000084 void resolve_const_value(t_const_value* const_val, t_type* ttype) {
85 if (ttype->is_map()) {
86 const std::map<t_const_value*, t_const_value*>& map = const_val->get_map();
87 std::map<t_const_value*, t_const_value*>::const_iterator v_iter;
88 for (v_iter = map.begin(); v_iter != map.end(); ++v_iter) {
89 resolve_const_value(v_iter->first, ((t_map*)ttype)->get_key_type());
90 resolve_const_value(v_iter->second, ((t_map*)ttype)->get_val_type());
91 }
92 } else if (ttype->is_list() || ttype->is_set()) {
93 const std::vector<t_const_value*>& val = const_val->get_list();
94 std::vector<t_const_value*>::const_iterator v_iter;
95 for (v_iter = val.begin(); v_iter != val.end(); ++v_iter) {
96 resolve_const_value((*v_iter), ((t_list*)ttype)->get_elem_type());
97 }
98 } else if (ttype->is_struct()) {
99 t_struct* tstruct = (t_struct*)ttype;
100 const std::map<t_const_value*, t_const_value*>& map = const_val->get_map();
101 std::map<t_const_value*, t_const_value*>::const_iterator v_iter;
102 for (v_iter = map.begin(); v_iter != map.end(); ++v_iter) {
103 t_field* field = tstruct->get_field_by_name(v_iter->first->get_string());
Bryan Duxbury656683c2010-06-23 16:57:51 +0000104 if (field == NULL) {
105 throw "No field named \"" + v_iter->first->get_string() + "\" was found in struct of type \"" + tstruct->get_name() + "\"";
106 }
Bryan Duxbury2d804702009-12-18 19:41:11 +0000107 resolve_const_value(v_iter->second, field->get_type());
108 }
109 } else if (const_val->get_type() == t_const_value::CV_IDENTIFIER) {
110 if (ttype->is_enum()) {
111 const_val->set_enum((t_enum*)ttype);
112 } else {
113 t_const* constant = get_constant(const_val->get_identifier());
114 if (constant == NULL) {
115 throw "No enum value or constant found named \"" + const_val->get_identifier() + "\"!";
116 }
117
David Reiss41d30582010-10-05 16:39:29 +0000118 // Resolve typedefs to the underlying type
119 t_type* const_type = constant->get_type()->get_true_type();
120
121 if (const_type->is_base_type()) {
122 switch (((t_base_type*)const_type)->get_base()) {
Bryan Duxbury2d804702009-12-18 19:41:11 +0000123 case t_base_type::TYPE_I16:
124 case t_base_type::TYPE_I32:
125 case t_base_type::TYPE_I64:
126 case t_base_type::TYPE_BOOL:
127 case t_base_type::TYPE_BYTE:
128 const_val->set_integer(constant->get_value()->get_integer());
129 break;
130 case t_base_type::TYPE_STRING:
131 const_val->set_string(constant->get_value()->get_string());
132 break;
133 case t_base_type::TYPE_DOUBLE:
134 const_val->set_double(constant->get_value()->get_double());
135 break;
136 case t_base_type::TYPE_VOID:
137 throw "Constants cannot be of type VOID";
138 }
David Reiss41d30582010-10-05 16:39:29 +0000139 } else if (const_type->is_map()) {
Bryan Duxbury2d804702009-12-18 19:41:11 +0000140 const std::map<t_const_value*, t_const_value*>& map = constant->get_value()->get_map();
141 std::map<t_const_value*, t_const_value*>::const_iterator v_iter;
142
143 const_val->set_map();
144 for (v_iter = map.begin(); v_iter != map.end(); ++v_iter) {
145 const_val->add_map(v_iter->first, v_iter->second);
146 }
David Reiss41d30582010-10-05 16:39:29 +0000147 } else if (const_type->is_list()) {
Bryan Duxbury2d804702009-12-18 19:41:11 +0000148 const std::vector<t_const_value*>& val = constant->get_value()->get_list();
149 std::vector<t_const_value*>::const_iterator v_iter;
150
151 const_val->set_list();
152 for (v_iter = val.begin(); v_iter != val.end(); ++v_iter) {
153 const_val->add_list(*v_iter);
154 }
155 }
156 }
157 } else if (ttype->is_enum()) {
158 // enum constant with non-identifier value. set the enum and find the
159 // value's name.
160 t_enum* tenum = (t_enum*)ttype;
161 t_enum_value* enum_value = tenum->get_constant_by_value(const_val->get_integer());
162 if (enum_value == NULL) {
David Reissaca320d2010-08-31 16:51:26 +0000163 std::ostringstream valstm;
164 valstm << const_val->get_integer();
165 throw "Couldn't find a named value in enum " + tenum->get_name() + " for value " + valstm.str();
Bryan Duxbury2d804702009-12-18 19:41:11 +0000166 }
Bryan Duxburycb0218f2010-09-12 15:22:21 +0000167 const_val->set_identifier(tenum->get_name() + "." + enum_value->get_name());
Bryan Duxbury2d804702009-12-18 19:41:11 +0000168 const_val->set_enum(tenum);
169 }
170 }
171
Mark Sleef0712dc2006-10-25 19:03:57 +0000172 private:
David Reiss0c90f6f2008-02-06 22:18:40 +0000173
Mark Sleef0712dc2006-10-25 19:03:57 +0000174 // Map of names to types
175 std::map<std::string, t_type*> types_;
176
Mark Sleed0767c52007-07-27 22:14:41 +0000177 // Map of names to constants
178 std::map<std::string, t_const*> constants_;
179
Mark Sleef0712dc2006-10-25 19:03:57 +0000180 // Map of names to services
David Reiss0c90f6f2008-02-06 22:18:40 +0000181 std::map<std::string, t_service*> services_;
182
Mark Sleef0712dc2006-10-25 19:03:57 +0000183};
184
185#endif