forked from openbmc/openpower-vpd-parser
-
Notifications
You must be signed in to change notification settings - Fork 21
/
Copy pathvpd_tool.cpp
264 lines (229 loc) · 8.79 KB
/
vpd_tool.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
#include "vpd_tool_impl.hpp"
#include <CLI/CLI.hpp>
#include <filesystem>
#include <fstream>
#include <iostream>
using namespace CLI;
using namespace std;
namespace fs = std::filesystem;
using namespace openpower::vpd;
using json = nlohmann::json;
int main(int argc, char** argv)
{
int rc = 0;
App app{"VPD Command line tool to dump the inventory and to read and "
"update the keywords"};
string objectPath{};
string recordName{};
string keyword{};
string val{};
uint32_t offset = 0;
auto object =
app.add_option("--object, -O", objectPath, "Enter the Object Path");
auto record =
app.add_option("--record, -R", recordName, "Enter the Record Name");
auto kw = app.add_option("--keyword, -K", keyword, "Enter the Keyword");
auto valOption = app.add_option(
"--value, -V", val,
"Enter the value. The value to be updated should be either in ascii or "
"in hex. ascii eg: 01234; hex eg: 0x30313233");
app.add_option("--seek, -s", offset,
"User can provide VPD offset using this option. Default "
"offset value is 0. Using --seek is optional and is valid "
"only while using --Hardware/-H option.");
auto dumpObjFlag =
app.add_flag("--dumpObject, -o",
"Dump the given object from the inventory. { "
"vpd-tool-exe --dumpObject/-o --object/-O object-name }")
->needs(object);
auto dumpInvFlag = app.add_flag(
"--dumpInventory, -i", "Dump all the inventory objects. { vpd-tool-exe "
"--dumpInventory/-i }");
auto readFlag =
app.add_flag("--readKeyword, -r",
"Read the data of the given keyword. { "
"vpd-tool-exe --readKeyword/-r --object/-O "
"\"object-name\" --record/-R \"record-name\" --keyword/-K "
"\"keyword-name\" }")
->needs(object)
->needs(record)
->needs(kw);
auto writeFlag =
app.add_flag(
"--writeKeyword, -w, --updateKeyword, -u",
"Update the value. { vpd-tool-exe "
"--writeKeyword/-w/--updateKeyword/-u "
"--object/-O object-name --record/-R record-name --keyword/-K "
"keyword-name --value/-V (or) --file }. Value can be given "
"directly via console using --value or via file using --file")
->needs(object)
->needs(record)
->needs(kw);
auto fileOption = app.add_option(
"--file", val,
"Enter the file name with its absolute path. This option can be used "
"in read and write operations. When used in read, the read value will "
"be saved to this file and when used in write, the value to be written "
"will be taken from this file.");
auto forceResetFlag =
app.add_flag("--forceReset, -f, -F",
"Force Collect for Hardware. CAUTION: Developer Only "
"Option. { vpd-tool-exe --forceReset/-f/-F }");
auto Hardware = app.add_flag(
"--Hardware, -H",
"This is a supplementary flag to read/write directly from/to hardware. "
"User should provide valid hardware/eeprom path (and not dbus object "
"path) in the -O/--object path. CAUTION: Developer Only Option");
auto fixSystemVPDFlag = app.add_flag(
"--fixSystemVPD", "Use this option to interactively fix critical "
"system VPD keywords {vpd-tool-exe --fixSystemVPD}");
auto mfgClean =
app.add_flag("--mfgClean", "Flag to clean and reset specific keywords "
"on system VPD to its default value.");
auto confirm =
app.add_flag("--yes", "Using this flag with --mfgClean option, assumes "
"yes to proceed without confirmation.");
CLI11_PARSE(app, argc, argv);
ifstream inventoryJson(INVENTORY_JSON_SYM_LINK);
auto jsObject = json::parse(inventoryJson);
try
{
if (*object && objectPath.empty())
{
throw runtime_error("Given path is empty.");
}
if (*record && (recordName.size() != 4))
{
throw runtime_error("Record " + recordName + " not supported.");
}
if ((*kw) && (keyword.size() != 2))
{
throw runtime_error("Keyword " + keyword + " not supported.");
}
if (*Hardware)
{
if (!fs::exists(objectPath)) // if dbus object path is given or
// invalid eeprom path is given
{
string errorMsg = "Invalid EEPROM path : ";
errorMsg += objectPath;
errorMsg +=
". The given EEPROM path doesn't exist. Provide valid "
"EEPROM path when -H flag is used. Refer help option. ";
throw runtime_error(errorMsg);
}
}
if (*writeFlag)
{
if (isReadOnlyEEPROM(objectPath, jsObject))
{
throw runtime_error("Read only EEPROM. Update not allowed.");
}
if ((!*fileOption) && (!*valOption))
{
throw runtime_error("Please provide the data that needs to be "
"updated. Use --value/--file to "
"input data. Refer --help.");
}
if ((*fileOption) && (!fs::exists(val)))
{
throw runtime_error("Please provide a valid file with absolute "
"path in --file.");
}
}
if (*dumpObjFlag)
{
VpdTool vpdToolObj(move(objectPath));
vpdToolObj.dumpObject(jsObject);
}
else if (*dumpInvFlag)
{
VpdTool vpdToolObj;
vpdToolObj.dumpInventory(jsObject);
}
else if (*readFlag && !*Hardware)
{
VpdTool vpdToolObj(move(objectPath), move(recordName),
move(keyword), move(val));
vpdToolObj.readKeyword();
}
else if (*writeFlag && !*Hardware)
{
VpdTool vpdToolObj(move(objectPath), move(recordName),
move(keyword), move(val));
rc = vpdToolObj.updateKeyword();
}
else if (*forceResetFlag)
{
// Force reset the BMC only if the CEC is powered OFF.
if (getPowerState() ==
"xyz.openbmc_project.State.Chassis.PowerState.Off")
{
VpdTool vpdToolObj;
vpdToolObj.forceReset(jsObject);
}
else
{
std::cerr << "The chassis power state is not Off. Force reset "
"operation is not allowed.";
return -1;
}
}
else if (*writeFlag && *Hardware)
{
VpdTool vpdToolObj(move(objectPath), move(recordName),
move(keyword), move(val));
rc = vpdToolObj.updateHardware(offset);
}
else if (*readFlag && *Hardware)
{
VpdTool vpdToolObj(move(objectPath), move(recordName),
move(keyword), move(val));
vpdToolObj.readKwFromHw(offset);
}
else if (*fixSystemVPDFlag)
{
std::string backupEepromPath;
std::string backupInvPath;
findBackupVPDPaths(backupEepromPath, backupInvPath, jsObject);
VpdTool vpdToolObj;
if (backupEepromPath.empty())
{
rc = vpdToolObj.fixSystemVPD();
}
else
{
rc = vpdToolObj.fixSystemBackupVPD(backupEepromPath,
backupInvPath);
}
}
else if (*mfgClean)
{
if (!*confirm)
{
std::string confirmation{};
std::cout << "\nThis option resets some of the system VPD "
"keywords to their default values. Do you really "
"wish to proceed further?[yes/no]: ";
std::cin >> confirmation;
if (confirmation != "yes")
{
return 0;
}
}
VpdTool vpdToolObj;
rc = vpdToolObj.cleanSystemVPD();
}
else
{
throw runtime_error("One of the valid options is required. Refer "
"--help for list of options.");
}
}
catch (const exception& e)
{
cerr << e.what();
rc = -1;
}
return rc;
}