PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 1 | #include "vpd_tool_impl.hpp" |
| 2 | |
| 3 | #include <CLI/CLI.hpp> |
Patrick Williams | c78d887 | 2023-05-10 07:50:56 -0500 | [diff] [blame] | 4 | |
PriyangaRamasamy | c0a534f | 2020-08-24 21:29:18 +0530 | [diff] [blame] | 5 | #include <filesystem> |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 6 | #include <fstream> |
| 7 | #include <iostream> |
| 8 | |
| 9 | using namespace CLI; |
| 10 | using namespace std; |
PriyangaRamasamy | c0a534f | 2020-08-24 21:29:18 +0530 | [diff] [blame] | 11 | namespace fs = std::filesystem; |
| 12 | using namespace openpower::vpd; |
| 13 | using json = nlohmann::json; |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 14 | |
| 15 | int main(int argc, char** argv) |
| 16 | { |
PriyangaRamasamy | d09d2ec | 2020-03-12 14:11:50 +0530 | [diff] [blame] | 17 | int rc = 0; |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 18 | App app{"VPD Command line tool to dump the inventory and to read and " |
| 19 | "update the keywords"}; |
| 20 | |
| 21 | string objectPath{}; |
PriyangaRamasamy | 02d4d4e | 2020-02-24 14:54:45 +0530 | [diff] [blame] | 22 | string recordName{}; |
| 23 | string keyword{}; |
PriyangaRamasamy | d09d2ec | 2020-03-12 14:11:50 +0530 | [diff] [blame] | 24 | string val{}; |
Priyanga Ramasamy | c99a0b0 | 2022-06-08 14:53:39 -0500 | [diff] [blame] | 25 | uint32_t offset = 0; |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 26 | |
Patrick Williams | c78d887 | 2023-05-10 07:50:56 -0500 | [diff] [blame] | 27 | auto object = app.add_option("--object, -O", objectPath, |
| 28 | "Enter the Object Path"); |
| 29 | auto record = app.add_option("--record, -R", recordName, |
| 30 | "Enter the Record Name"); |
PriyangaRamasamy | 02d4d4e | 2020-02-24 14:54:45 +0530 | [diff] [blame] | 31 | auto kw = app.add_option("--keyword, -K", keyword, "Enter the Keyword"); |
PriyangaRamasamy | d09d2ec | 2020-03-12 14:11:50 +0530 | [diff] [blame] | 32 | auto valOption = app.add_option( |
| 33 | "--value, -V", val, |
| 34 | "Enter the value. The value to be updated should be either in ascii or " |
| 35 | "in hex. ascii eg: 01234; hex eg: 0x30313233"); |
Priyanga Ramasamy | c99a0b0 | 2022-06-08 14:53:39 -0500 | [diff] [blame] | 36 | app.add_option("--seek, -s", offset, |
| 37 | "User can provide VPD offset using this option. Default " |
Priyanga Ramasamy | 5629fbc | 2023-03-01 08:17:19 -0600 | [diff] [blame] | 38 | "offset value is 0. Using --seek is optional and is valid " |
Priyanga Ramasamy | c99a0b0 | 2022-06-08 14:53:39 -0500 | [diff] [blame] | 39 | "only while using --Hardware/-H option."); |
| 40 | |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 41 | auto dumpObjFlag = |
| 42 | app.add_flag("--dumpObject, -o", |
| 43 | "Dump the given object from the inventory. { " |
| 44 | "vpd-tool-exe --dumpObject/-o --object/-O object-name }") |
| 45 | ->needs(object); |
| 46 | |
| 47 | auto dumpInvFlag = app.add_flag( |
| 48 | "--dumpInventory, -i", "Dump all the inventory objects. { vpd-tool-exe " |
| 49 | "--dumpInventory/-i }"); |
| 50 | |
PriyangaRamasamy | 02d4d4e | 2020-02-24 14:54:45 +0530 | [diff] [blame] | 51 | auto readFlag = |
| 52 | app.add_flag("--readKeyword, -r", |
| 53 | "Read the data of the given keyword. { " |
| 54 | "vpd-tool-exe --readKeyword/-r --object/-O " |
| 55 | "\"object-name\" --record/-R \"record-name\" --keyword/-K " |
| 56 | "\"keyword-name\" }") |
| 57 | ->needs(object) |
| 58 | ->needs(record) |
| 59 | ->needs(kw); |
| 60 | |
PriyangaRamasamy | d09d2ec | 2020-03-12 14:11:50 +0530 | [diff] [blame] | 61 | auto writeFlag = |
| 62 | app.add_flag( |
| 63 | "--writeKeyword, -w, --updateKeyword, -u", |
| 64 | "Update the value. { vpd-tool-exe " |
| 65 | "--writeKeyword/-w/--updateKeyword/-u " |
| 66 | "--object/-O object-name --record/-R record-name --keyword/-K " |
Priyanga Ramasamy | 5629fbc | 2023-03-01 08:17:19 -0600 | [diff] [blame] | 67 | "keyword-name --value/-V (or) --file }. Value can be given " |
| 68 | "directly via console using --value or via file using --file") |
PriyangaRamasamy | 83ea53f | 2021-05-13 05:55:21 -0500 | [diff] [blame] | 69 | ->needs(object) |
PriyangaRamasamy | d09d2ec | 2020-03-12 14:11:50 +0530 | [diff] [blame] | 70 | ->needs(record) |
Priyanga Ramasamy | 5629fbc | 2023-03-01 08:17:19 -0600 | [diff] [blame] | 71 | ->needs(kw); |
| 72 | |
| 73 | auto fileOption = app.add_option( |
| 74 | "--file", val, |
| 75 | "Enter the file name with its absolute path. This option can be used " |
| 76 | "in read and write operations. When used in read, the read value will " |
| 77 | "be saved to this file and when used in write, the value to be written " |
| 78 | "will be taken from this file."); |
PriyangaRamasamy | d09d2ec | 2020-03-12 14:11:50 +0530 | [diff] [blame] | 79 | |
Priyanga Ramasamy | c99a0b0 | 2022-06-08 14:53:39 -0500 | [diff] [blame] | 80 | auto forceResetFlag = |
| 81 | app.add_flag("--forceReset, -f, -F", |
| 82 | "Force Collect for Hardware. CAUTION: Developer Only " |
| 83 | "Option. { vpd-tool-exe --forceReset/-f/-F }"); |
| 84 | |
PriyangaRamasamy | c0a534f | 2020-08-24 21:29:18 +0530 | [diff] [blame] | 85 | auto Hardware = app.add_flag( |
| 86 | "--Hardware, -H", |
Priyanga Ramasamy | c99a0b0 | 2022-06-08 14:53:39 -0500 | [diff] [blame] | 87 | "This is a supplementary flag to read/write directly from/to hardware. " |
| 88 | "User should provide valid hardware/eeprom path (and not dbus object " |
| 89 | "path) in the -O/--object path. CAUTION: Developer Only Option"); |
PriyangaRamasamy | 0407b17 | 2020-03-31 13:57:18 +0530 | [diff] [blame] | 90 | |
Priyanga Ramasamy | 43ffcf7 | 2022-06-08 14:10:11 -0500 | [diff] [blame] | 91 | auto fixSystemVPDFlag = app.add_flag( |
| 92 | "--fixSystemVPD", "Use this option to interactively fix critical " |
| 93 | "system VPD keywords {vpd-tool-exe --fixSystemVPD}"); |
| 94 | |
Patrick Williams | c78d887 | 2023-05-10 07:50:56 -0500 | [diff] [blame] | 95 | auto mfgClean = app.add_flag("--mfgClean", |
| 96 | "Flag to clean and reset specific keywords " |
| 97 | "on system VPD to its default value."); |
Priyanga Ramasamy | 124ae6c | 2022-10-18 12:46:14 -0500 | [diff] [blame] | 98 | |
| 99 | auto confirm = |
| 100 | app.add_flag("--yes", "Using this flag with --mfgClean option, assumes " |
| 101 | "yes to proceed without confirmation."); |
| 102 | |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 103 | CLI11_PARSE(app, argc, argv); |
| 104 | |
Santosh Puranik | 0246a4d | 2020-11-04 16:57:39 +0530 | [diff] [blame] | 105 | ifstream inventoryJson(INVENTORY_JSON_SYM_LINK); |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 106 | auto jsObject = json::parse(inventoryJson); |
| 107 | |
| 108 | try |
| 109 | { |
Priyanga Ramasamy | 5629fbc | 2023-03-01 08:17:19 -0600 | [diff] [blame] | 110 | if ((*kw) && (keyword.size() != 2)) |
| 111 | { |
| 112 | throw runtime_error("Keyword " + keyword + " not supported."); |
| 113 | } |
| 114 | |
PriyangaRamasamy | c0a534f | 2020-08-24 21:29:18 +0530 | [diff] [blame] | 115 | if (*Hardware) |
| 116 | { |
PriyangaRamasamy | 83ea53f | 2021-05-13 05:55:21 -0500 | [diff] [blame] | 117 | if (!fs::exists(objectPath)) // if dbus object path is given or |
| 118 | // invalid eeprom path is given |
PriyangaRamasamy | c0a534f | 2020-08-24 21:29:18 +0530 | [diff] [blame] | 119 | { |
PriyangaRamasamy | 83ea53f | 2021-05-13 05:55:21 -0500 | [diff] [blame] | 120 | string errorMsg = "Invalid EEPROM path : "; |
| 121 | errorMsg += objectPath; |
| 122 | errorMsg += |
| 123 | ". The given EEPROM path doesn't exist. Provide valid " |
| 124 | "EEPROM path when -H flag is used. Refer help option. "; |
| 125 | throw runtime_error(errorMsg); |
PriyangaRamasamy | c0a534f | 2020-08-24 21:29:18 +0530 | [diff] [blame] | 126 | } |
| 127 | } |
Priyanga Ramasamy | 5629fbc | 2023-03-01 08:17:19 -0600 | [diff] [blame] | 128 | |
| 129 | if (*writeFlag) |
| 130 | { |
| 131 | if ((!*fileOption) && (!*valOption)) |
| 132 | { |
| 133 | throw runtime_error("Please provide the data that needs to be " |
| 134 | "updated. Use --value/--file to " |
| 135 | "input data. Refer --help."); |
| 136 | } |
| 137 | |
| 138 | if ((*fileOption) && (!fs::exists(val))) |
| 139 | { |
| 140 | throw runtime_error("Please provide a valid file with absolute " |
| 141 | "path in --file."); |
| 142 | } |
| 143 | } |
| 144 | |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 145 | if (*dumpObjFlag) |
| 146 | { |
| 147 | VpdTool vpdToolObj(move(objectPath)); |
| 148 | vpdToolObj.dumpObject(jsObject); |
| 149 | } |
| 150 | |
| 151 | else if (*dumpInvFlag) |
| 152 | { |
| 153 | VpdTool vpdToolObj; |
| 154 | vpdToolObj.dumpInventory(jsObject); |
| 155 | } |
| 156 | |
Priyanga Ramasamy | 3803131 | 2021-10-07 16:39:13 -0500 | [diff] [blame] | 157 | else if (*readFlag && !*Hardware) |
PriyangaRamasamy | 02d4d4e | 2020-02-24 14:54:45 +0530 | [diff] [blame] | 158 | { |
| 159 | VpdTool vpdToolObj(move(objectPath), move(recordName), |
Priyanga Ramasamy | 5629fbc | 2023-03-01 08:17:19 -0600 | [diff] [blame] | 160 | move(keyword), move(val)); |
PriyangaRamasamy | 02d4d4e | 2020-02-24 14:54:45 +0530 | [diff] [blame] | 161 | vpdToolObj.readKeyword(); |
| 162 | } |
| 163 | |
PriyangaRamasamy | c0a534f | 2020-08-24 21:29:18 +0530 | [diff] [blame] | 164 | else if (*writeFlag && !*Hardware) |
PriyangaRamasamy | d09d2ec | 2020-03-12 14:11:50 +0530 | [diff] [blame] | 165 | { |
PriyangaRamasamy | 83ea53f | 2021-05-13 05:55:21 -0500 | [diff] [blame] | 166 | VpdTool vpdToolObj(move(objectPath), move(recordName), |
| 167 | move(keyword), move(val)); |
PriyangaRamasamy | d09d2ec | 2020-03-12 14:11:50 +0530 | [diff] [blame] | 168 | rc = vpdToolObj.updateKeyword(); |
| 169 | } |
| 170 | |
PriyangaRamasamy | 0407b17 | 2020-03-31 13:57:18 +0530 | [diff] [blame] | 171 | else if (*forceResetFlag) |
| 172 | { |
Priyanga Ramasamy | 335873f | 2022-05-18 01:31:54 -0500 | [diff] [blame] | 173 | // Force reset the BMC only if the CEC is powered OFF. |
| 174 | if (getPowerState() == |
| 175 | "xyz.openbmc_project.State.Chassis.PowerState.Off") |
| 176 | { |
| 177 | VpdTool vpdToolObj; |
| 178 | vpdToolObj.forceReset(jsObject); |
| 179 | } |
| 180 | else |
| 181 | { |
| 182 | std::cerr << "The chassis power state is not Off. Force reset " |
| 183 | "operation is not allowed."; |
| 184 | return -1; |
| 185 | } |
PriyangaRamasamy | 0407b17 | 2020-03-31 13:57:18 +0530 | [diff] [blame] | 186 | } |
| 187 | |
PriyangaRamasamy | c0a534f | 2020-08-24 21:29:18 +0530 | [diff] [blame] | 188 | else if (*writeFlag && *Hardware) |
| 189 | { |
PriyangaRamasamy | 83ea53f | 2021-05-13 05:55:21 -0500 | [diff] [blame] | 190 | VpdTool vpdToolObj(move(objectPath), move(recordName), |
| 191 | move(keyword), move(val)); |
Priyanga Ramasamy | c99a0b0 | 2022-06-08 14:53:39 -0500 | [diff] [blame] | 192 | rc = vpdToolObj.updateHardware(offset); |
PriyangaRamasamy | c0a534f | 2020-08-24 21:29:18 +0530 | [diff] [blame] | 193 | } |
Priyanga Ramasamy | 3803131 | 2021-10-07 16:39:13 -0500 | [diff] [blame] | 194 | else if (*readFlag && *Hardware) |
| 195 | { |
| 196 | VpdTool vpdToolObj(move(objectPath), move(recordName), |
Priyanga Ramasamy | 5629fbc | 2023-03-01 08:17:19 -0600 | [diff] [blame] | 197 | move(keyword), move(val)); |
Priyanga Ramasamy | c99a0b0 | 2022-06-08 14:53:39 -0500 | [diff] [blame] | 198 | vpdToolObj.readKwFromHw(offset); |
Priyanga Ramasamy | 3803131 | 2021-10-07 16:39:13 -0500 | [diff] [blame] | 199 | } |
Priyanga Ramasamy | 43ffcf7 | 2022-06-08 14:10:11 -0500 | [diff] [blame] | 200 | else if (*fixSystemVPDFlag) |
| 201 | { |
| 202 | VpdTool vpdToolObj; |
| 203 | rc = vpdToolObj.fixSystemVPD(); |
| 204 | } |
Priyanga Ramasamy | 124ae6c | 2022-10-18 12:46:14 -0500 | [diff] [blame] | 205 | else if (*mfgClean) |
| 206 | { |
| 207 | if (!*confirm) |
| 208 | { |
| 209 | std::string confirmation{}; |
| 210 | std::cout << "\nThis option resets some of the system VPD " |
| 211 | "keywords to their default values. Do you really " |
| 212 | "wish to proceed further?[yes/no]: "; |
| 213 | std::cin >> confirmation; |
| 214 | |
| 215 | if (confirmation != "yes") |
| 216 | { |
| 217 | return 0; |
| 218 | } |
| 219 | } |
| 220 | VpdTool vpdToolObj; |
| 221 | rc = vpdToolObj.cleanSystemVPD(); |
| 222 | } |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 223 | else |
| 224 | { |
| 225 | throw runtime_error("One of the valid options is required. Refer " |
| 226 | "--help for list of options."); |
| 227 | } |
| 228 | } |
| 229 | |
Patrick Williams | 8e15b93 | 2021-10-06 13:04:22 -0500 | [diff] [blame] | 230 | catch (const exception& e) |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 231 | { |
| 232 | cerr << e.what(); |
Priyanga Ramasamy | c99a0b0 | 2022-06-08 14:53:39 -0500 | [diff] [blame] | 233 | |
| 234 | if (*Hardware) |
| 235 | { |
Priyanga Ramasamy | 5629fbc | 2023-03-01 08:17:19 -0600 | [diff] [blame] | 236 | std::cerr << "Did you provide a valid offset? By default VPD " |
| 237 | "offset is taken as 0. To input offset, use --seek. " |
| 238 | "Refer vpd-tool help." |
| 239 | << std::endl; |
Priyanga Ramasamy | c99a0b0 | 2022-06-08 14:53:39 -0500 | [diff] [blame] | 240 | } |
Santosh Puranik | 6c7a84e | 2022-03-09 13:42:18 +0530 | [diff] [blame] | 241 | rc = -1; |
PriyangaRamasamy | 1f0b1e6 | 2020-02-20 20:48:25 +0530 | [diff] [blame] | 242 | } |
| 243 | |
PriyangaRamasamy | d09d2ec | 2020-03-12 14:11:50 +0530 | [diff] [blame] | 244 | return rc; |
Patrick Williams | c78d887 | 2023-05-10 07:50:56 -0500 | [diff] [blame] | 245 | } |