Dhruvaraj Subhashchandran | 35714cd | 2024-05-16 05:54:38 -0500 | [diff] [blame] | 1 | #!/bin/bash |
| 2 | # shellcheck disable=SC2034 # Variable is used elsewhere |
| 3 | |
| 4 | help=$(cat << EOF |
| 5 | opdreport creates an archive consisting of the following: |
| 6 | * host dump files and header applied on top of it |
| 7 | The type parameter controls the content of the data. The generated |
| 8 | archive is stored in the user specified location. |
| 9 | |
| 10 | usage: opdreport [OPTION] |
| 11 | |
| 12 | Options: |
| 13 | -n, --name <name> Name to be used for the archive. |
| 14 | Default name format |
| 15 | SYSDUMP.<serial number>.<dump_id>.<time> |
| 16 | Optional parameter. |
| 17 | -d, --dir <directory> Archive directory to copy the compressed report. |
| 18 | Default output directory is current working |
| 19 | directory. Optional parameter. |
| 20 | -i, --dumpid <id> Dump identifier to associate with the archive. |
| 21 | Identifiers include numeric characters. |
| 22 | Default dump identifier is 0 |
| 23 | -s, --size <size> Maximum allowed size (in KB) of the archive. |
| 24 | Report will be truncated if size exceeds |
| 25 | this limit. Default size is unlimited. |
| 26 | -f, --failingunit The id of the failed unit |
| 27 | -e, --eid Error log associated with the failure |
| 28 | -t, --type Type of the dump to be collected |
| 29 | 1 - Hardware dump |
| 30 | 3 - Performance dump |
| 31 | 5 - Hostboot dump |
| 32 | 10 - SBE Dump |
| 33 | -h, --help Display this help and exit. |
| 34 | EOF |
| 35 | ) |
| 36 | |
| 37 | # Constants |
| 38 | readonly OP_DUMP="opdump" |
| 39 | readonly DREPORT_SOURCE="/usr/share/dreport.d" |
| 40 | readonly TRUE=1 |
| 41 | readonly FALSE=0 |
| 42 | readonly TIME_STAMP="date -u" |
| 43 | readonly UNLIMITED="unlimited" |
| 44 | readonly DREPORT_INCLUDE="$DREPORT_SOURCE/include.d" |
| 45 | readonly INVENTORY_MANAGER='xyz.openbmc_project.Inventory.Manager' |
| 46 | readonly INVENTORY_PATH='/xyz/openbmc_project/inventory/system' |
| 47 | readonly INVENTORY_ASSET_INT='xyz.openbmc_project.Inventory.Decorator.Asset' |
| 48 | readonly INVENTORY_BMC_BOARD='/xyz/openbmc_project/inventory/system/chassis/motherboard' |
| 49 | readonly HEADER_EXTENSION="$DREPORT_INCLUDE/gendumpheader" |
| 50 | readonly FILE_SCRIPT="$DREPORT_SOURCE/include.d/gendumpinfo" |
| 51 | |
| 52 | # Error Codes |
| 53 | readonly SUCCESS=0 |
| 54 | readonly INTERNAL_FAILURE=1 |
| 55 | readonly RESOURCE_UNAVAILABLE=2 |
| 56 | |
| 57 | # Variables |
| 58 | declare -x dump_type="$OP_DUMP" |
| 59 | declare -x dump_sbe_type=100 |
| 60 | declare -x size_dump="" |
| 61 | declare -x elog_id="00000000" |
| 62 | declare -x EPOCHTIME |
| 63 | EPOCHTIME=$(date +"%s") |
| 64 | declare -x name="" |
| 65 | declare -x dump_dir="/tmp" |
| 66 | declare -x dump_id="00000000" |
| 67 | declare -x dump_size="unlimited" |
| 68 | declare -x content_path="" |
| 69 | declare -x name_dir="" |
| 70 | declare -x serialNo="0000000" |
| 71 | declare -x dDay |
| 72 | dDay=$(date -d @"$EPOCHTIME" +'%Y%m%d%H%M%S') |
| 73 | declare -x dump_content_type="" |
| 74 | declare -x FILE="" |
| 75 | |
| 76 | # @brief Get serial number property from inventory |
| 77 | function fetch_serial_number() { |
| 78 | serialNo=$(busctl get-property "$INVENTORY_MANAGER" "$INVENTORY_PATH" \ |
| 79 | "$INVENTORY_ASSET_INT" SerialNumber | cut -d " " -f 2 | \ |
| 80 | sed 's/^"\(.*\)"$/\1/') |
| 81 | |
| 82 | if [ -z "$serialNo" ]; then |
| 83 | serialNo="0000000" |
| 84 | fi |
| 85 | } |
| 86 | |
| 87 | # @brief Check the validity of user inputs and initialize global variables |
| 88 | function initialize() { |
| 89 | # shellcheck disable=SC2154 # name comes from elsewhere |
| 90 | if [ -z "$name" ]; then |
| 91 | name="SYSDUMP" |
| 92 | fi |
| 93 | fetch_serial_number |
| 94 | # shellcheck disable=SC2154 # dump_id comes from elsewhere |
| 95 | name="${name}.${serialNo}.${dump_id}.${dDay}" |
| 96 | |
| 97 | if [ -z "$dump_sbe_type" ]; then |
| 98 | echo "Error: Dump type is not provided." |
| 99 | return "$RESOURCE_UNAVAILABLE" |
| 100 | fi |
| 101 | |
| 102 | if [ -z "$dump_dir" ]; then |
| 103 | dump_dir=$PWD |
| 104 | fi |
| 105 | |
| 106 | if [[ "$dump_size" =~ ^[0-9]+$ ]]; then |
| 107 | dump_size=$((dump_size * 1024)) |
| 108 | else |
| 109 | dump_size=$UNLIMITED |
| 110 | fi |
| 111 | |
| 112 | return "$SUCCESS" |
| 113 | } |
| 114 | |
| 115 | # @brief Collect the dump |
| 116 | function collect() { |
| 117 | content_path="/tmp/dump_${dump_id}_${EPOCHTIME}" |
| 118 | dump_outpath="$content_path/plat_dump" |
| 119 | if ! mkdir -p "$dump_outpath"; then |
| 120 | echo "Could not create the destination directory $dump_outpath" |
| 121 | return "$INTERNAL_FAILURE" |
| 122 | fi |
| 123 | |
| 124 | dump-collect --type "$dump_sbe_type" --id "0x$dump_id" \ |
| 125 | --failingunit "$failing_unit" --path "$dump_outpath" |
| 126 | } |
| 127 | |
| 128 | # @brief Package the dump and transfer to dump location |
| 129 | function package() { |
| 130 | FILE="/tmp/dumpheader_${dump_id}_${EPOCHTIME}" |
| 131 | if ! mkdir -p "$dump_dir"; then |
| 132 | echo "Could not create the destination directory $dump_dir" |
| 133 | dump_dir="/tmp" |
| 134 | fi |
| 135 | |
| 136 | cd "$content_path" || exit "$INTERNAL_FAILURE" |
| 137 | |
| 138 | dump_content_type=${id:0:2} |
| 139 | "$FILE_SCRIPT" |
| 140 | elog_id=$eid |
| 141 | |
| 142 | if ! tar -cvzf "$name" plat_dump/*Sbe* info.yaml; then |
| 143 | echo "$($TIME_STAMP)" "Could not create the compressed tar file" |
| 144 | return "$INTERNAL_FAILURE" |
| 145 | fi |
| 146 | |
| 147 | size_dump=$(stat -c %s "$name") |
| 148 | |
| 149 | if [ "$dump_size" != "$UNLIMITED" ] && \ |
| 150 | [ "$size_dump" -gt "$dump_size" ]; then |
| 151 | rm "$name" |
| 152 | return "$RESOURCE_UNAVAILABLE" |
| 153 | fi |
| 154 | |
| 155 | echo "Adding Dump Header: $HEADER_EXTENSION" |
| 156 | "$HEADER_EXTENSION" |
| 157 | |
| 158 | if ! tee -a "$FILE" < "$name" > /dev/null; then |
| 159 | echo "$($TIME_STAMP)" "Could not create the compressed file" |
| 160 | rm -rf "$name" "$FILE" |
| 161 | return "$INTERNAL_FAILURE" |
| 162 | fi |
| 163 | |
| 164 | if ! mv "$FILE" "$name"; then |
| 165 | echo "$($TIME_STAMP)" "Could not create the compressed file" |
| 166 | rm -rf "$name" "$FILE" |
| 167 | return "$INTERNAL_FAILURE" |
| 168 | fi |
| 169 | |
| 170 | mv "$name" "$dump_dir" |
| 171 | |
| 172 | rm -rf "$content_path" |
| 173 | rm -rf "$FILE" "$name" |
| 174 | |
| 175 | return "$SUCCESS" |
| 176 | } |
| 177 | |
| 178 | # @brief Initiate BMC dump |
| 179 | function initiate_bmc_dump() { |
| 180 | bmcDumpPath=$(busctl call xyz.openbmc_project.Dump.Manager \ |
| 181 | /xyz/openbmc_project/dump/bmc \ |
| 182 | xyz.openbmc_project.Dump.Create CreateDump a\{sv\} 0) |
| 183 | result=$? |
| 184 | if [[ $result -ne $SUCCESS ]]; then |
| 185 | echo "Error in creating BMC dump associated with system dump" |
| 186 | else |
| 187 | echo "BMC dump initiated $bmcDumpPath" |
| 188 | fi |
| 189 | } |
| 190 | |
| 191 | # @brief Main function |
| 192 | function main() { |
| 193 | initialize |
| 194 | result=$? |
| 195 | if [[ $result -ne $SUCCESS ]]; then |
| 196 | echo "$($TIME_STAMP)" "Error: Failed to initialize, Exiting" |
| 197 | return "$INTERNAL_FAILURE" |
| 198 | fi |
| 199 | |
| 200 | collect |
| 201 | result=$? |
| 202 | if [[ $result -ne $SUCCESS ]]; then |
| 203 | echo "$($TIME_STAMP)" "Error: Failed to collect dump, Exiting" |
| 204 | return "$INTERNAL_FAILURE" |
| 205 | fi |
| 206 | |
| 207 | package |
| 208 | result=$? |
| 209 | if [[ $result -ne $SUCCESS ]]; then |
| 210 | echo "$($TIME_STAMP)" "Error: Failed to package, Exiting" |
| 211 | return "$INTERNAL_FAILURE" |
| 212 | else |
| 213 | echo "$($TIME_STAMP)" "Successfully completed" |
| 214 | fi |
| 215 | |
| 216 | initiate_bmc_dump |
| 217 | return "$SUCCESS" |
| 218 | } |
| 219 | |
| 220 | if ! TEMP=$(getopt -o n:d:i:s:t:e:f:h \ |
| 221 | --long name:,dir:,dumpid:,size:,type:,eid:,failingunit:,help \ |
| 222 | -- "$@"); then |
| 223 | echo "Error: Invalid options" |
| 224 | exit 1 |
| 225 | fi |
| 226 | |
| 227 | eval set -- "$TEMP" |
| 228 | |
| 229 | while [[ $# -gt 1 ]]; do |
| 230 | key="$1" |
| 231 | case $key in |
| 232 | -n|--name) |
| 233 | name=$2 |
| 234 | shift 2 ;; |
| 235 | -d|--dir) |
| 236 | dump_dir=$2 |
| 237 | shift 2 ;; |
| 238 | -i|--dumpid) |
| 239 | dump_id=$2 |
| 240 | shift 2 ;; |
| 241 | -s|--size) |
| 242 | dump_size=$2 |
| 243 | shift 2 ;; |
| 244 | -f|--failingunit) |
| 245 | failing_unit=$2 |
| 246 | shift 2 ;; |
| 247 | -e|--eid) |
| 248 | eid=$2 |
| 249 | shift 2 ;; |
| 250 | -t|--type) |
| 251 | dump_sbe_type=$2 |
| 252 | shift 2 ;; |
| 253 | -h|--help) |
| 254 | echo "$help" |
| 255 | exit ;; |
| 256 | *) # unknown option |
| 257 | echo "Unknown argument: $1" |
| 258 | echo "$help" |
| 259 | exit 1 ;; |
| 260 | esac |
| 261 | done |
| 262 | |
| 263 | main |
| 264 | exit $? |