-
Notifications
You must be signed in to change notification settings - Fork 139
/
Copy pathSymbol_IR.cpp
224 lines (202 loc) · 7.05 KB
/
Symbol_IR.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
/*******************************************************************************
*
* (c) Copyright IBM Corp. 2015, 2016
*
* This program and the accompanying materials are made available
* under the terms of the Eclipse Public License v1.0 and
* Apache License v2.0 which accompanies this distribution.
*
* The Eclipse Public License is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* The Apache License v2.0 is available at
* http://www.opensource.org/licenses/apache2.0.php
*
* Contributors:
* Multiple authors (IBM Corp.) - initial implementation and documentation
*******************************************************************************/
#include "ddr/ir/Symbol_IR.hpp"
#include <assert.h>
#include <algorithm>
#include <map>
#include <stdlib.h>
#include <string.h>
#include <vector>
#include "ddr/config.hpp"
#include "ddr/ir/ClassType.hpp"
#include "ddr/ir/ClassUDT.hpp"
#include "ddr/ir/EnumUDT.hpp"
#include "ddr/ir/EnumUDT.hpp"
#include "ddr/ir/Field.hpp"
#include "ddr/ir/NamespaceUDT.hpp"
#include "ddr/ir/NamespaceUDT.hpp"
#include "ddr/ir/TypedefUDT.hpp"
#include "ddr/ir/TypedefUDT.hpp"
#include "ddr/ir/UnionUDT.hpp"
using std::map;
Symbol_IR::~Symbol_IR()
{
for (size_t i = 0; i < _types.size(); i++) {
if (NULL == _types[i]) {
ERRMSG("Null member, cannot free");
} else {
delete(_types[i]);
}
}
_types.clear();
}
DDR_RC
Symbol_IR::applyOverrideList(OMRPortLibrary *portLibrary, const char *overrideFiles)
{
DDR_RC rc = DDR_RC_OK;
OMRPORT_ACCESS_FROM_OMRPORT(portLibrary);
/* Read list of type override files from specified file. */
intptr_t fd = omrfile_open(overrideFiles, EsOpenRead, 0);
if (0 > fd) {
ERRMSG("Failure attempting to open %s\nExiting...\n", overrideFiles);
rc = DDR_RC_ERROR;
} else {
char *buff = NULL;
int64_t offset = omrfile_seek(fd, 0, EsSeekEnd);
if (-1 != offset) {
buff = (char *)malloc(offset + 1);
memset(buff, 0, offset + 1);
omrfile_seek(fd, 0, EsSeekSet);
/* Read each line as a file name, then open that file to get type overrides. */
if (0 < omrfile_read(fd, buff, offset)) {
char *line = buff;
char *nextLine = strchr(line, '\n');
nextLine[0] = '\0';
while ((NULL != nextLine) && (DDR_RC_OK == rc)) {
nextLine[0] = '\0';
rc = applyOverrides(portLibrary, line);
line = nextLine + 1;
nextLine = strchr(line, '\n');
}
}
free(buff);
}
}
return rc;
}
DDR_RC
Symbol_IR::applyOverrides(OMRPortLibrary *portLibrary, const char *overrideFile)
{
DDR_RC rc = DDR_RC_OK;
OMRPORT_ACCESS_FROM_OMRPORT(portLibrary);
/* Read list of type overrides from specified file. */
vector<FieldOverride> overrideList;
intptr_t fd = omrfile_open(overrideFile, EsOpenRead, 0);
if (0 > fd) {
ERRMSG("Failure attempting to open %s\nExiting...\n", overrideFile);
rc = DDR_RC_ERROR;
} else {
char *buff = NULL;
int64_t offset = omrfile_seek(fd, 0, EsSeekEnd);
if (-1 != offset) {
buff = (char *)malloc(offset + 1);
memset(buff, 0, offset + 1);
omrfile_seek(fd, 0, EsSeekSet);
/* Read each line, expecting format of "[typeoverride/fieldoverride].[StructureName].[fieldName]=[overrideType]" */
if (0 < omrfile_read(fd, buff, offset)) {
char *nextLine = strtok(buff, "\n");
while (NULL != nextLine) {
string line = nextLine;
/* Remove comment portion of the line, if present. */
size_t commentPosition = line.find("#");
if (string::npos != commentPosition) {
line = line.substr(0, commentPosition);
}
line.erase(line.find_last_not_of(" \t") + 1);
/* Ignore the prefix "ddrblob.", if present. */
string prefix = "ddrblob.";
if (string::npos != line.find(prefix)) {
line = line.substr(prefix.length(), line.length());
}
/* Check if the override is a typeoverride or a fieldoverride. */
string typeoverride = "typeoverride.";
string fieldoverride = "fieldoverride.";
bool isTypeOverride = false;
if (string::npos != line.find(typeoverride)) {
isTypeOverride = true;
line = line.substr(typeoverride.length(), line.length());
} else if (string::npos != line.find(fieldoverride)) {
line = line.substr(fieldoverride.length(), line.length());
}
/* Remove any pointer or array specifiers in the line and use those the field has. */
line.erase(remove(line.begin(), line.end(), '*'), line.end());
line.erase(remove(line.begin(), line.end(), '['), line.end());
line.erase(remove(line.begin(), line.end(), ']'), line.end());
/* Correctly formatted lines must have exactly 1 dot and 1 equals. */
size_t dotCount = count(line.begin(), line.end(), '.');
size_t equalCount = count(line.begin(), line.end(), '=');
if ((1 == dotCount) && (1 == equalCount)) {
/* Get the structure name, field name, and override name. Add to list to process. */
size_t dotPosition = line.find(".");
size_t equalsPosition = line.find("=", dotPosition);
FieldOverride to = {
line.substr(0, dotPosition),
line.substr(dotPosition + 1, equalsPosition - dotPosition - 1),
line.substr(equalsPosition + 1, line.length() - equalsPosition - 1),
isTypeOverride
};
overrideList.push_back(to);
if (DDR_RC_OK != rc) {
break;
}
}
nextLine = strtok(NULL, "\n");
}
}
free(buff);
}
}
/* Create a map of type name to vector of types to check all types
* by name for type overrides.
*/
map<string, vector<Type *> > typeNames;
for (vector<Type *>::iterator it = _types.begin(); it != _types.end(); it += 1) {
Type *type = *it;
typeNames[type->_name].push_back(type);
}
/* Apply the type overrides. */
for (vector<FieldOverride>::iterator it = overrideList.begin(); it != overrideList.end(); it += 1) {
FieldOverride type = *it;
/* Check if the structure to override exists. */
if (typeNames.find(type.structName) != typeNames.end()) {
Type *replacementType = NULL;
if (type.isTypeOverride) {
/* If the type for the override exists in the IR, use it. Otherwise, create it. */
if (typeNames.find(type.overrideName) == typeNames.end()) {
replacementType = new Type(0);
replacementType->_name = type.overrideName;
typeNames[replacementType->_name].push_back(replacementType);
} else {
replacementType = typeNames[type.overrideName].front();
}
}
/* Iterate over the types with a matching name for the override. */
vector<Type *> *typesWithName = &typeNames[type.structName];
for (vector<Type *>::iterator it2 = typesWithName->begin(); it2 != typesWithName->end(); it2 += 1) {
(*it2)->renameFieldsAndMacros(type, replacementType);
}
}
}
return rc;
}
/* Compute the field offsets for all types in the IR from the sizes of the fields. */
void
Symbol_IR::computeOffsets()
{
/* For each Type in the ir, compute the field offsets from the size of each field. */
for (size_t i = 0; i < _types.size(); i++) {
_types[i]->computeFieldOffsets();
}
}
void
Symbol_IR::removeDuplicates()
{
for (vector<Type *>::iterator it = _types.begin(); it != _types.end(); ++it) {
(*it)->checkDuplicate(this);
}
}