-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmemory_manage_data.cpp
265 lines (216 loc) · 8.12 KB
/
memory_manage_data.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
//Copyright 2019 Hust.
//Author: HYL
//Descriptor:
/*
*/
#include <regex>
#include "debug_info.h"
#include "tool_fun.h"
#include "memory_manage_data.h"
using namespace::std;
class DebugInfo;
DataAreaManage* DataAreaManage::data_manage_instance = NULL;
long int DataAreaManage::begin_addr_dataArea = 15872;
long int DataAreaManage::end_addr_dataArea = 16384;
int DataAreaManage::available_data_area_addr = 512;
int DataAreaManage::have_used_data_area_addr = 0;
unordered_map<string, DataStoreInfo> \
DataAreaManage::data_area_addr_map;
DataAreaManage::DataAreaManage(){
}
void
DataAreaManage::InitialRomRange(vector<string> &rom_range_vec, \
string &core_name){
int vec_size = rom_range_vec.size();
for(int i = 0; i < vec_size; i++){
string range_elem = rom_range_vec[i];
auto comma_index = range_elem.find('-');
string range_begin_str = range_elem.substr(0, comma_index);
string range_end_str = range_elem.substr(comma_index + 1);
int range_begin_num = ChangeHexToDec(range_begin_str);
int range_end_num = ChangeHexToDec(range_end_str);
DataAreaManage::available_data_area_addr = \
range_end_num - range_begin_num;
DataAreaManage::begin_addr_dataArea = range_begin_num;
DataAreaManage::end_addr_dataArea = range_end_num;
if(available_data_area_addr <= 0) {
throw \
"Error: InitialRomRange() available_data_area_addr invaild";
}
}
}
//var_name is the variable name, var_type is the type of the variable
//elem_num is used for the array.Because we just record the first
//elem addr of the array as it's whole addr. elem_num record how many
//elem the array have. if the vriable is not array, the value of the
//elem num will be 1.
DataStoreInfo
DataAreaManage::CoreAllocateDataAreaFun(string var_name, \
string var_type, \
int elem_num){
DataStoreInfo core_info;
//if elem_num equal -1. it represent the var_name is truct type
//variable.then each elem in this var_name will have different
//type. so we should treat it with different way.
if(elem_num == -1) {
//get all elem type about this struct
const vector<string>& type_vec = \
this->GetRecordVarDecalare(var_type);
//DataStroeInfo core_info
core_info.virtual_addr = var_name;
core_info.data_type = var_type;
core_info.store_where_place = 2;
//how many reg that needed
core_info.belong_which_fun_name = this->GetBelongWhatCallName();
//addr debug info
DebugInfo debug_info_obj = DebugInfo();
debug_info_obj.CreateAAddrDebugRecord(var_name, var_type);
for(int i = 0; i < type_vec.size(); i++){
debug_info_obj.AddAddrDebugInfoToRecord(var_name, \
"element: " + to_string(i));
debug_info_obj.AddAddrDebugInfoToRecord(var_name, \
"type: " + type_vec[i]);//yzk 2019.4.28 添加了struct中的元素类型信息
int reg_num = this->HowBigType(type_vec[i]);
//get reg and make record
for(int j = 0; j < reg_num; j++){
if(available_data_area_addr <= 0 || \
reg_num > available_data_area_addr){
throw "No enough data area to allocate!";
}
int data_area_addr_int = have_used_data_area_addr + \
begin_addr_dataArea;
string data_area_addr_str = ChangeDecToHexDataArea(data_area_addr_int);
core_info.actual_addr.push_back(data_area_addr_str);
//add data area debug info into record
debug_info_obj.AddAddrDebugInfoToRecord(var_name, \
data_area_addr_str);
//updata data area addr status
++have_used_data_area_addr;
--available_data_area_addr;
}
}
return core_info;
}
//the var_name is single variable or array
else {
//check the type that decide how many data area need
int data_area_need_num = this->HowBigType(var_type);
int total_addr_used_for_var = data_area_need_num * elem_num;
//check whether still have enough data area to allocate
if(available_data_area_addr <= 0 || \
total_addr_used_for_var > available_data_area_addr){
throw "No enough data area to allocate!";
}
core_info.virtual_addr = var_name;
core_info.data_type = var_type;
core_info.store_where_place = 2;
//how many reg.
core_info.actual_addr_num = data_area_need_num;
//record addr debug infomation
DebugInfo debug_info_obj = DebugInfo();
debug_info_obj.CreateAAddrDebugRecord(var_name, var_type);
//the outer statement for is for array,
//the inter for statement if for register of a variable
for(int i = 0; i < elem_num; i++){
//get data area addr and make record
for(int j = 0; j < data_area_need_num; j++){
int data_area_addr_int = have_used_data_area_addr + \
begin_addr_dataArea;
string data_area_addr_str = ChangeDecToHexDataArea(data_area_addr_int);
core_info.actual_addr.push_back(data_area_addr_str);
//add data area debug info into record
debug_info_obj.AddAddrDebugInfoToRecord(var_name, \
data_area_addr_str);
//updata data area addr status
++have_used_data_area_addr;
--available_data_area_addr;
}
}
return core_info;
}
return core_info;
}//end
//addr_num is for array. we just record the first elem addr of a
//array when we translate array
void DataAreaManage::AllocateDataAreaToVal(string var_name, string var_type, int elem_num){
DataStoreInfo core_info = this->CoreAllocateDataAreaFun(var_name, \
var_type, \
elem_num);
this->SetVarStoreWhichMap(var_name, 4);
this->CreateDataAreaMapForVal(var_name, core_info);
}
//for two dimension array
void DataAreaManage::AllocateDataAreaToVal_2DArray(string var_name,\
string var_type, \
int elem_num, \
int first_size, \
int second_size){
DataStoreInfo core_info = this->CoreAllocateDataAreaFun(var_name, \
var_type, \
elem_num);
this->SetVarStoreWhichMap(var_name, 4);
this->CreateDataAreaMapForVal(var_name, core_info);
DebugInfo debug_info_obj = DebugInfo();
debug_info_obj.AddAddrDebugInfoToRecord_2DArray(var_name,\
first_size,\
second_size);
}
void DataAreaManage::CreateDataAreaMapForVal(string var_name, DataStoreInfo core_info){
//store the core_info into data_area_addr_map
data_area_addr_map.insert(make_pair(var_name, core_info));
}
//get addr from data_area_addr_map via var_name
//remenber that if var_name is a single variable, then the elem_index
//equal 0. since the index begin from 0.
//but we still don't store single variable in data area
vector<string> DataAreaManage::GetActualAddrFromVal(string var_name, \
int elem_index){
if(data_area_addr_map.find(var_name) == data_area_addr_map.end()){
throw "GetActualAddrFromVal(): No this variable in "\
"data area map";
}
DataStoreInfo core_info = data_area_addr_map.find(var_name)->second;
vector<string> addr = core_info.actual_addr;
string type = core_info.data_type;
vector<string> res;
//this type is struct type, we will get the elem_index elem
//in var_name
regex which_type_struct("%struct.*");
if(regex_match(type, which_type_struct)){
const vector<string> &struct_elem_type_vec = \
this->GetRecordVarDecalare(type);
int elem_base = 0;
for(int i = 0; i < elem_index; i++){
elem_base += this->HowBigType(struct_elem_type_vec[i]);
}
int reg_num = this->HowBigType(struct_elem_type_vec[elem_index]);
for(int i = 0; i < reg_num; i++){
res.push_back(addr[elem_base + i]);
}
return res;
}
//this type is array or single variable
else {
int type_num = this->HowBigType(type);
for(int i = 0; i < type_num; i++){
res.push_back(addr[elem_index * type_num + i]);
}
return res;
}
return res;
}
vector<string> \
DataAreaManage::GetAllActualAddrFromVal(string var_name){
if(data_area_addr_map.find(var_name) == data_area_addr_map.end()){
throw "Error: memory_manage_data.cpp GetAllActualAddrFromVal()";
}
DataStoreInfo core_info = data_area_addr_map.find(var_name)->second;
return core_info.actual_addr;
}
string DataAreaManage::IntegrateVectorAddr(vector<string>& actual_addr){
int addr_num = actual_addr.size();
}
void DataAreaManage::GetValueFromGivenDataAreaAddr(string var_name, int addr_num){
}
vector<string> DataAreaManage::SplitStringAddrToVector(string var_name){
}