1366 lines
39 KiB
Diff
1366 lines
39 KiB
Diff
|
From a56e8bf096c04ea0c76ca026e07402f05ce58ca4 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Kinga=20Ta=C5=84ska?= <kinga.tanska@intel.com>
|
||
|
Date: Tue, 14 Jun 2022 12:11:45 +0200
|
||
|
Subject: [PATCH 1/2] Ledctl - slots management (#94)
|
||
|
|
||
|
* ledctl: add commands to support empty slots blinking
|
||
|
* ledctl: --get-slot implementation for vmd
|
||
|
* ledctl: --list-slots implementation for vmd
|
||
|
* ledctl: --set-slot implementation for vmd
|
||
|
* ledctl: --get-slot implementation for npem
|
||
|
* ledctl: --list-slots implementation for npem
|
||
|
* ledctl: --set-slot implementation for npem
|
||
|
|
||
|
Add new interface which allows to read or modify led state for devices
|
||
|
by controller's slots. It allows to manage empty slots in cases where:
|
||
|
- drive is not connected.
|
||
|
- drive is connected, but non-standard driver is used (uio, vfio).
|
||
|
|
||
|
Slot identifier will used to manage led state. Following commands are added:
|
||
|
--get-slot
|
||
|
--list-slots
|
||
|
--set-slot
|
||
|
Support for VMD and NPEM is added.
|
||
|
|
||
|
Signed-off-by: Kinga Tanska <kinga.tanska@intel.com>
|
||
|
---
|
||
|
doc/ledctl.pod | 36 ++++++
|
||
|
src/Makefile.am | 2 +-
|
||
|
src/block.c | 12 ++
|
||
|
src/block.h | 15 ++-
|
||
|
src/ledctl.c | 329 ++++++++++++++++++++++++++++++++++++++++++++++--
|
||
|
src/npem.c | 145 ++++++++++++++++++---
|
||
|
src/npem.h | 28 +++++
|
||
|
src/pci_slot.c | 102 ++++++++++++++-
|
||
|
src/pci_slot.h | 35 +++++-
|
||
|
src/slot.h | 63 ++++++++++
|
||
|
src/sysfs.c | 1 -
|
||
|
src/sysfs.h | 4 +-
|
||
|
src/utils.c | 49 +++++++-
|
||
|
src/utils.h | 15 ++-
|
||
|
src/vmdssd.c | 63 +++++-----
|
||
|
src/vmdssd.h | 11 +-
|
||
|
16 files changed, 833 insertions(+), 77 deletions(-)
|
||
|
create mode 100644 src/slot.h
|
||
|
|
||
|
diff --git a/doc/ledctl.pod b/doc/ledctl.pod
|
||
|
index 774d04b2b0e2..a0da95e4c734 100644
|
||
|
--- a/doc/ledctl.pod
|
||
|
+++ b/doc/ledctl.pod
|
||
|
@@ -324,6 +324,42 @@ Displays version of ledctl and information about the license and exits.
|
||
|
Prints information (system path and type) of all controllers detected by
|
||
|
ledmon and exits.
|
||
|
|
||
|
+=item B<-P> or B<--list-slots> B<--controller>=I<controller>
|
||
|
+
|
||
|
+Prints all slots for the controller. Slot definition depends on the controller
|
||
|
+and is unique across all controllers of the same type.
|
||
|
+
|
||
|
+Definitions for supported controllers are described below:
|
||
|
+
|
||
|
+=over
|
||
|
+
|
||
|
+=item
|
||
|
+
|
||
|
+VMD - PCI Express Hot Plug Controller Driver slot number
|
||
|
+
|
||
|
+=item
|
||
|
+
|
||
|
+NPEM - PCI Express Downstream Port address
|
||
|
+
|
||
|
+=back
|
||
|
+
|
||
|
+Command returns a list of all slots for the controller with current state and
|
||
|
+attached device name (if any). I<controller> is type of controller
|
||
|
+(vmd, NPEM) that should be scanned here.
|
||
|
+
|
||
|
+=item B<-G> or B<--get-slot> B<--controller>=I<controller> B<--device>=I<device>
|
||
|
+
|
||
|
+Displays slot details of given device. I<device> is devnode of selected drive.
|
||
|
+
|
||
|
+=item B<-G> or B<--get-slot> B<--controller>=I<controller> B<--slot>=I<slot>
|
||
|
+
|
||
|
+Displays details of given slot. I<slot> is unique slot identifier.
|
||
|
+
|
||
|
+=item B<-S> or B<--set-slot> B<--controller>=I<controller> B<--slot>=I<slot> B<--state>=I<IBPI_state>
|
||
|
+
|
||
|
+Changes led state for given slot. I<controller> is type of the controller.
|
||
|
+I<slot> is unique slot identifier. I<IBPI_state> is led pattern.
|
||
|
+
|
||
|
=item B<-x> or B<--listed-only>
|
||
|
|
||
|
With this option ledctl will change state only on devices listed in CLI. The
|
||
|
diff --git a/src/Makefile.am b/src/Makefile.am
|
||
|
index f7786206a510..34311a2699a7 100644
|
||
|
--- a/src/Makefile.am
|
||
|
+++ b/src/Makefile.am
|
||
|
@@ -26,7 +26,7 @@ COMMON_SRCS = ahci.c block.c cntrl.c config_file.c enclosure.c list.c \
|
||
|
vmdssd.h ipmi.h amd.h amd_ipmi.h npem.h
|
||
|
|
||
|
LEDMON_SRCS = ledmon.c pidfile.c $(COMMON_SRCS)
|
||
|
-LEDCTL_SRCS = ledctl.c $(COMMON_SRCS)
|
||
|
+LEDCTL_SRCS = ledctl.c slot.h $(COMMON_SRCS)
|
||
|
|
||
|
|
||
|
sbin_PROGRAMS = ledmon ledctl
|
||
|
diff --git a/src/block.c b/src/block.c
|
||
|
index c61c577cb2c8..3d1815cb11a3 100644
|
||
|
--- a/src/block.c
|
||
|
+++ b/src/block.c
|
||
|
@@ -215,6 +215,18 @@ struct _host_type *block_get_host(struct cntrl_device *cntrl, int host_id)
|
||
|
return hosts;
|
||
|
}
|
||
|
|
||
|
+struct block_device *get_block_device_from_sysfs_path(char *sub_path)
|
||
|
+{
|
||
|
+ struct block_device *device;
|
||
|
+
|
||
|
+ list_for_each(sysfs_get_block_devices(), device) {
|
||
|
+ if (strstr(device->sysfs_path, sub_path))
|
||
|
+ return device;
|
||
|
+ }
|
||
|
+
|
||
|
+ return NULL;
|
||
|
+}
|
||
|
+
|
||
|
/*
|
||
|
* Allocates a new block device structure. See block.h for details.
|
||
|
*/
|
||
|
diff --git a/src/block.h b/src/block.h
|
||
|
index 820f2d72325e..00b9fd73a19d 100644
|
||
|
--- a/src/block.h
|
||
|
+++ b/src/block.h
|
||
|
@@ -1,6 +1,6 @@
|
||
|
/*
|
||
|
* Intel(R) Enclosure LED Utilities
|
||
|
- * Copyright (C) 2009-2018 Intel Corporation.
|
||
|
+ * Copyright (C) 2009-2022 Intel Corporation.
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify it
|
||
|
* under the terms and conditions of the GNU General Public License,
|
||
|
@@ -25,6 +25,7 @@
|
||
|
#include "time.h"
|
||
|
#include "list.h"
|
||
|
#include "raid.h"
|
||
|
+#include "status.h"
|
||
|
|
||
|
struct block_device;
|
||
|
|
||
|
@@ -227,4 +228,16 @@ struct _host_type *block_get_host(struct cntrl_device *cntrl, int host_id);
|
||
|
int block_compare(const struct block_device *bd_old,
|
||
|
const struct block_device *bd_new);
|
||
|
|
||
|
+/**
|
||
|
+ * @brief Finds block device which name contains sub-path.
|
||
|
+ *
|
||
|
+ * This function scans block devices and checks their sysfs path
|
||
|
+ * to find any which contains PCI address specified for device in path.
|
||
|
+ *
|
||
|
+ * @param[in] sub_path Sub path.
|
||
|
+ *
|
||
|
+ * @return first block device containing sub-path if any, otherwise NULL.
|
||
|
+ */
|
||
|
+struct block_device *get_block_device_from_sysfs_path(char *sub_path);
|
||
|
+
|
||
|
#endif /* _BLOCK_H_INCLUDED_ */
|
||
|
diff --git a/src/ledctl.c b/src/ledctl.c
|
||
|
index caedc1a4834d..c20986230739 100644
|
||
|
--- a/src/ledctl.c
|
||
|
+++ b/src/ledctl.c
|
||
|
@@ -42,12 +42,13 @@
|
||
|
#include "cntrl.h"
|
||
|
#include "config.h"
|
||
|
#include "config_file.h"
|
||
|
-#include "ibpi.h"
|
||
|
+#include "slot.h"
|
||
|
#include "list.h"
|
||
|
+#include "npem.h"
|
||
|
+#include "pci_slot.h"
|
||
|
#include "scsi.h"
|
||
|
#include "status.h"
|
||
|
#include "sysfs.h"
|
||
|
-#include "utils.h"
|
||
|
|
||
|
/**
|
||
|
* @brief An IBPI state structure.
|
||
|
@@ -69,6 +70,73 @@ struct ibpi_state {
|
||
|
*/
|
||
|
static struct list ibpi_list;
|
||
|
|
||
|
+/**
|
||
|
+ * @brief Pointer to a get slot function.
|
||
|
+ *
|
||
|
+ * The pointer to a function which will print slot details.
|
||
|
+ *
|
||
|
+ * @param[in] device Name of the device.
|
||
|
+ * @param[in] slot Unique identifier of the slot.
|
||
|
+ * @param[in] res Pointer to slot response.
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+typedef status_t (*get_slot_t) (char *device, char *slot, struct slot_response *res);
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Pointer to a set slot function.
|
||
|
+ *
|
||
|
+ * The pointer to a function which will set slot details.
|
||
|
+ *
|
||
|
+ * @param[in] slot Unique identifier of the slot.
|
||
|
+ * @param[in] state IBPI state based on slot request.
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+typedef status_t (*set_slot_t) (char *slot, enum ibpi_pattern state);
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief slot request parametres
|
||
|
+ *
|
||
|
+ * This structure contains all possible parameters for slot related commands.
|
||
|
+ */
|
||
|
+struct slot_request {
|
||
|
+ /**
|
||
|
+ * Option given in the request.
|
||
|
+ */
|
||
|
+ int chosen_opt;
|
||
|
+
|
||
|
+ /**
|
||
|
+ * Name of the device.
|
||
|
+ */
|
||
|
+ char device[PATH_MAX];
|
||
|
+
|
||
|
+ /**
|
||
|
+ * Unique slot identifier.
|
||
|
+ */
|
||
|
+ char slot[PATH_MAX];
|
||
|
+
|
||
|
+ /**
|
||
|
+ * Type of the controller.
|
||
|
+ */
|
||
|
+ enum cntrl_type cntrl;
|
||
|
+
|
||
|
+ /**
|
||
|
+ * IBPI state.
|
||
|
+ */
|
||
|
+ enum ibpi_pattern state;
|
||
|
+
|
||
|
+ /**
|
||
|
+ * Pointer to the get slot function.
|
||
|
+ */
|
||
|
+ get_slot_t get_slot_fn;
|
||
|
+
|
||
|
+ /**
|
||
|
+ * Pointer to the set slot function.
|
||
|
+ */
|
||
|
+ set_slot_t set_slot_fn;
|
||
|
+};
|
||
|
+
|
||
|
/**
|
||
|
* @brief IBPI pattern names.
|
||
|
*
|
||
|
@@ -76,7 +144,7 @@ static struct list ibpi_list;
|
||
|
* this entries to translate enumeration type values into the string.
|
||
|
*/
|
||
|
const char *ibpi_str[] = {
|
||
|
- [IBPI_PATTERN_UNKNOWN] = "",
|
||
|
+ [IBPI_PATTERN_UNKNOWN] = "UNKNOWN",
|
||
|
[IBPI_PATTERN_NORMAL] = "NORMAL",
|
||
|
[IBPI_PATTERN_ONESHOT_NORMAL] = "",
|
||
|
[IBPI_PATTERN_DEGRADED] = "ICA",
|
||
|
@@ -112,6 +180,13 @@ static int possible_params[] = {
|
||
|
OPT_VERSION,
|
||
|
OPT_LIST_CTRL,
|
||
|
OPT_LISTED_ONLY,
|
||
|
+ OPT_LIST_SLOTS,
|
||
|
+ OPT_GET_SLOT,
|
||
|
+ OPT_SET_SLOT,
|
||
|
+ OPT_CONTROLLER,
|
||
|
+ OPT_DEVICE,
|
||
|
+ OPT_SLOT,
|
||
|
+ OPT_STATE,
|
||
|
OPT_ALL,
|
||
|
OPT_DEBUG,
|
||
|
OPT_ERROR,
|
||
|
@@ -126,6 +201,43 @@ static const int possible_params_size = sizeof(possible_params)
|
||
|
|
||
|
static int listed_only;
|
||
|
|
||
|
+enum cntrl_type get_cntrl_type(const char *cntrl)
|
||
|
+{
|
||
|
+ if (strcasecmp(cntrl, "vmd") == 0)
|
||
|
+ return CNTRL_TYPE_VMD;
|
||
|
+ else if (strcasecmp(cntrl, "npem") == 0)
|
||
|
+ return CNTRL_TYPE_NPEM;
|
||
|
+ return CNTRL_TYPE_UNKNOWN;
|
||
|
+}
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Determines a slot functions based on controller.
|
||
|
+ *
|
||
|
+ * This function determines slot functions based on
|
||
|
+ * controller type.
|
||
|
+ *
|
||
|
+ * @param[in] ctrl_type Controller type.
|
||
|
+ * @param[in] slot_req Pointer to the slot request.
|
||
|
+ *
|
||
|
+ * @return This function does not return a value.
|
||
|
+ */
|
||
|
+static void _get_slot_ctrl_fn(enum cntrl_type ctrl_type, struct slot_request *slot_req)
|
||
|
+{
|
||
|
+ switch (ctrl_type) {
|
||
|
+ case CNTRL_TYPE_VMD:
|
||
|
+ slot_req->get_slot_fn = pci_get_slot;
|
||
|
+ slot_req->set_slot_fn = pci_set_slot;
|
||
|
+ break;
|
||
|
+ case CNTRL_TYPE_NPEM:
|
||
|
+ slot_req->get_slot_fn = npem_get_slot;
|
||
|
+ slot_req->set_slot_fn = npem_set_slot;
|
||
|
+ break;
|
||
|
+ default:
|
||
|
+ log_debug("Slot functions could not be set because the controller type %s does not "
|
||
|
+ "support slots managing.", ctrl_type);
|
||
|
+ }
|
||
|
+}
|
||
|
+
|
||
|
static void ibpi_state_fini(struct ibpi_state *p)
|
||
|
{
|
||
|
list_clear(&p->block_list);
|
||
|
@@ -186,16 +298,25 @@ static void _ledctl_help(void)
|
||
|
progname);
|
||
|
printf("Mandatory arguments for long options are mandatory for short options, too.\n\n");
|
||
|
print_opt("--listed-only", "-x",
|
||
|
- "Ledctl will change state only for given devices.");
|
||
|
+ "Ledctl will change state only for given devices.");
|
||
|
print_opt("--list-controllers", "-L",
|
||
|
- "Displays list of controllers detected by ledmon.");
|
||
|
+ "Displays list of controllers detected by ledmon.");
|
||
|
+ print_opt("--list-slots --controller CONTROLLER", "-P -c CONTROLLER",
|
||
|
+ "List slots under the controller, their led states, slot numbers and "
|
||
|
+ "devnodes connected.");
|
||
|
+ print_opt("--get-slot --controller CONTROLLER --device DEVNODE / --slot SLOT",
|
||
|
+ "-G -c CONTROLLER -d DEVNODE / -p SLOT",
|
||
|
+ "Prints slot information, its led state, slot number and devnode.");
|
||
|
+ print_opt("--set-slot --controller CONTROLLER --slot SLOT --state STATE",
|
||
|
+ "-S -c CONTROLLER -p SLOT -s STATE", "Sets given state for chosen slot "
|
||
|
+ "under the controller.");
|
||
|
print_opt("--log=PATH", "-l PATH",
|
||
|
- "Use local log file instead /var/log/ledctl.log.");
|
||
|
+ "Use local log file instead /var/log/ledctl.log.");
|
||
|
print_opt("--help", "-h", "Displays this help text.");
|
||
|
print_opt("--version", "-v",
|
||
|
- "Displays version and license information.");
|
||
|
+ "Displays version and license information.");
|
||
|
print_opt("--log-level=VALUE", "-l VALUE",
|
||
|
- "Allows user to set ledctl verbose level in logs.");
|
||
|
+ "Allows user to set ledctl verbose level in logs.");
|
||
|
printf("\nPatterns:\n"
|
||
|
"\tCommon patterns are:\n"
|
||
|
"\t\tlocate, locate_off, normal, off, degraded, rebuild,\n" ""
|
||
|
@@ -555,6 +676,156 @@ static status_t _cmdline_parse_non_root(int argc, char *argv[])
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
+/**
|
||
|
+ * @brief Inits slot request structure with initial values.
|
||
|
+ *
|
||
|
+ * @param[in] slot_req structure with slot request
|
||
|
+ *
|
||
|
+ * @return This function does not return a value.
|
||
|
+ */
|
||
|
+static void slot_request_init(struct slot_request *slot_req)
|
||
|
+{
|
||
|
+ memset(slot_req, 0, sizeof(struct slot_request));
|
||
|
+
|
||
|
+ slot_req->chosen_opt = OPT_NULL_ELEMENT;
|
||
|
+ slot_req->state = IBPI_PATTERN_UNKNOWN;
|
||
|
+}
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Inits slot response structure with initial values.
|
||
|
+ *
|
||
|
+ * @param[in] slot_res structure with slot response
|
||
|
+ *
|
||
|
+ * @return This function does not return a value.
|
||
|
+ */
|
||
|
+static void slot_response_init(struct slot_response *slot_res)
|
||
|
+{
|
||
|
+ memset(slot_res, 0, sizeof(struct slot_response));
|
||
|
+
|
||
|
+ slot_res->state = IBPI_PATTERN_UNKNOWN;
|
||
|
+}
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Verifies slot request parameters.
|
||
|
+ *
|
||
|
+ * @param[in] slot_req structure with slot request
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+static status_t slot_verify_request(struct slot_request *slot_req)
|
||
|
+{
|
||
|
+ if (slot_req->cntrl == CNTRL_TYPE_UNKNOWN) {
|
||
|
+ log_error("Invalid controller in the request.");
|
||
|
+ return STATUS_INVALID_CONTROLLER;
|
||
|
+ }
|
||
|
+ if (slot_req->chosen_opt == OPT_SET_SLOT && slot_req->state == IBPI_PATTERN_UNKNOWN) {
|
||
|
+ log_error("Invalid IBPI state in the request.");
|
||
|
+ return STATUS_INVALID_STATE;
|
||
|
+ }
|
||
|
+ if (!slot_req->get_slot_fn && !slot_req->set_slot_fn) {
|
||
|
+ log_error("The controller type %s doesn't support slot functionality.",
|
||
|
+ slot_req->cntrl);
|
||
|
+ return STATUS_INVALID_CONTROLLER;
|
||
|
+ }
|
||
|
+ if (slot_req->device[0] && slot_req->slot[0]) {
|
||
|
+ log_error("Device and slot parameters are exclusive.");
|
||
|
+ return STATUS_DATA_ERROR;
|
||
|
+ }
|
||
|
+
|
||
|
+ return STATUS_SUCCESS;
|
||
|
+}
|
||
|
+
|
||
|
+static status_t get_state_for_slot(char *slot, struct slot_request *slot_req)
|
||
|
+{
|
||
|
+ struct slot_response slot_res;
|
||
|
+ status_t status = STATUS_SUCCESS;
|
||
|
+
|
||
|
+ slot_response_init(&slot_res);
|
||
|
+ status = slot_req->get_slot_fn(NULL, slot, &slot_res);
|
||
|
+ if (status == STATUS_SUCCESS)
|
||
|
+ print_slot_state(&slot_res);
|
||
|
+
|
||
|
+ return status;
|
||
|
+}
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief List slots connected to given controller
|
||
|
+ *
|
||
|
+ * This function scans all available slots connected to given controller
|
||
|
+ * and prints their led states and names of the connected devices (if exist).
|
||
|
+ *
|
||
|
+ * @param[in] slot_req Structure with slot request.
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+static status_t list_slots(struct slot_request *slot_req)
|
||
|
+{
|
||
|
+ status_t status = STATUS_SUCCESS;
|
||
|
+
|
||
|
+ switch (slot_req->cntrl) {
|
||
|
+ case CNTRL_TYPE_VMD:
|
||
|
+ {
|
||
|
+ struct pci_slot *slot;
|
||
|
+
|
||
|
+ list_for_each(sysfs_get_pci_slots(), slot)
|
||
|
+ status = get_state_for_slot(slot->sysfs_path, slot_req);
|
||
|
+ return status;
|
||
|
+ }
|
||
|
+ case CNTRL_TYPE_NPEM:
|
||
|
+ {
|
||
|
+ struct cntrl_device *ctrl_dev;
|
||
|
+
|
||
|
+ list_for_each(sysfs_get_cntrl_devices(), ctrl_dev) {
|
||
|
+ if (ctrl_dev->cntrl_type != CNTRL_TYPE_NPEM)
|
||
|
+ continue;
|
||
|
+ status = get_state_for_slot(ctrl_dev->sysfs_path, slot_req);
|
||
|
+ }
|
||
|
+ return status;
|
||
|
+ }
|
||
|
+ default:
|
||
|
+ return STATUS_NOT_SUPPORTED;
|
||
|
+ }
|
||
|
+}
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Executes proper slot mode function.
|
||
|
+ *
|
||
|
+ * @param[in] slot_req Structure with slot request.
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+status_t slot_execute(struct slot_request *slot_req)
|
||
|
+{
|
||
|
+ struct slot_response slot_res;
|
||
|
+ status_t status = STATUS_SUCCESS;
|
||
|
+
|
||
|
+ slot_response_init(&slot_res);
|
||
|
+
|
||
|
+ switch (slot_req->chosen_opt) {
|
||
|
+ case OPT_LIST_SLOTS:
|
||
|
+ return list_slots(slot_req);
|
||
|
+ case OPT_SET_SLOT:
|
||
|
+ status = slot_req->get_slot_fn(slot_req->device, slot_req->slot, &slot_res);
|
||
|
+ if (slot_res.state == slot_req->state) {
|
||
|
+ log_warning("Led state: %s is already set for the slot.",
|
||
|
+ ibpi2str(slot_req->state));
|
||
|
+ return STATUS_SUCCESS;
|
||
|
+ }
|
||
|
+ if (status != STATUS_SUCCESS)
|
||
|
+ return status;
|
||
|
+ status = slot_req->set_slot_fn(slot_res.slot, slot_req->state);
|
||
|
+ if (status != STATUS_SUCCESS)
|
||
|
+ return status;
|
||
|
+ case OPT_GET_SLOT:
|
||
|
+ status = slot_req->get_slot_fn(slot_req->device, slot_req->slot, &slot_res);
|
||
|
+ if (status == STATUS_SUCCESS)
|
||
|
+ print_slot_state(&slot_res);
|
||
|
+ return status;
|
||
|
+ default:
|
||
|
+ return STATUS_NOT_SUPPORTED;
|
||
|
+ }
|
||
|
+}
|
||
|
+
|
||
|
/**
|
||
|
* @brief Command line parser - options.
|
||
|
*
|
||
|
@@ -567,7 +838,7 @@ static status_t _cmdline_parse_non_root(int argc, char *argv[])
|
||
|
*
|
||
|
* @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
*/
|
||
|
-static status_t _cmdline_parse(int argc, char *argv[])
|
||
|
+static status_t _cmdline_parse(int argc, char *argv[], struct slot_request *req)
|
||
|
{
|
||
|
int opt, opt_index = -1;
|
||
|
status_t status = STATUS_SUCCESS;
|
||
|
@@ -614,6 +885,34 @@ static status_t _cmdline_parse(int argc, char *argv[])
|
||
|
sysfs_reset();
|
||
|
exit(EXIT_SUCCESS);
|
||
|
}
|
||
|
+ case 'G':
|
||
|
+ req->chosen_opt = OPT_GET_SLOT;
|
||
|
+ break;
|
||
|
+ case 'P':
|
||
|
+ req->chosen_opt = OPT_LIST_SLOTS;
|
||
|
+ break;
|
||
|
+ case 'S':
|
||
|
+ req->chosen_opt = OPT_SET_SLOT;
|
||
|
+ break;
|
||
|
+ case 'c':
|
||
|
+ req->cntrl = get_cntrl_type(optarg);
|
||
|
+ _get_slot_ctrl_fn(req->cntrl, req);
|
||
|
+ break;
|
||
|
+ case 's':
|
||
|
+ {
|
||
|
+ struct ibpi_state *state = _ibpi_state_get(optarg);
|
||
|
+
|
||
|
+ if (state)
|
||
|
+ req->state = state->ibpi;
|
||
|
+ free(state);
|
||
|
+ break;
|
||
|
+ }
|
||
|
+ case 'd':
|
||
|
+ strncpy(req->device, optarg, PATH_MAX - 1);
|
||
|
+ break;
|
||
|
+ case 'p':
|
||
|
+ strncpy(req->slot, optarg, PATH_MAX - 1);
|
||
|
+ break;
|
||
|
case ':':
|
||
|
case '?':
|
||
|
default:
|
||
|
@@ -712,6 +1011,7 @@ static status_t _init_ledctl_conf(void)
|
||
|
int main(int argc, char *argv[])
|
||
|
{
|
||
|
status_t status;
|
||
|
+ struct slot_request slot_req;
|
||
|
|
||
|
setup_options(&longopt, &shortopt, possible_params,
|
||
|
possible_params_size);
|
||
|
@@ -732,7 +1032,9 @@ int main(int argc, char *argv[])
|
||
|
return status;
|
||
|
if (on_exit(_ledctl_fini, progname))
|
||
|
exit(STATUS_ONEXIT_ERROR);
|
||
|
- if (_cmdline_parse(argc, argv))
|
||
|
+ slot_request_init(&slot_req);
|
||
|
+ status = _cmdline_parse(argc, argv, &slot_req);
|
||
|
+ if (status != STATUS_SUCCESS)
|
||
|
exit(STATUS_CMDLINE_ERROR);
|
||
|
free(shortopt);
|
||
|
free(longopt);
|
||
|
@@ -746,6 +1048,13 @@ int main(int argc, char *argv[])
|
||
|
list_init(&ibpi_list, (item_free_t)ibpi_state_fini);
|
||
|
sysfs_init();
|
||
|
sysfs_scan();
|
||
|
+ if (slot_req.chosen_opt != OPT_NULL_ELEMENT) {
|
||
|
+ status = slot_verify_request(&slot_req);
|
||
|
+ if (status == STATUS_SUCCESS)
|
||
|
+ return slot_execute(&slot_req);
|
||
|
+ else
|
||
|
+ exit(status);
|
||
|
+ }
|
||
|
status = _cmdline_ibpi_parse(argc, argv);
|
||
|
if (status != STATUS_SUCCESS) {
|
||
|
log_debug("main(): _ibpi_parse() failed (status=%s).",
|
||
|
diff --git a/src/npem.c b/src/npem.c
|
||
|
index 4f836ee8571d..4b08966bd13a 100644
|
||
|
--- a/src/npem.c
|
||
|
+++ b/src/npem.c
|
||
|
@@ -25,7 +25,9 @@
|
||
|
|
||
|
#include "config.h"
|
||
|
#include "cntrl.h"
|
||
|
+#include "list.h"
|
||
|
#include "npem.h"
|
||
|
+#include "sysfs.h"
|
||
|
#include "utils.h"
|
||
|
|
||
|
#define PCI_EXT_CAP_ID_NPEM 0x29 /* Native PCIe Enclosure Management */
|
||
|
@@ -57,19 +59,32 @@
|
||
|
|
||
|
#define PCI_NPEM_STATUS_CC 0x01 /* NPEM Command Completed */
|
||
|
|
||
|
-const int ibpi_to_npem_capability[] = {
|
||
|
- [IBPI_PATTERN_NORMAL] = PCI_NPEM_OK_CAP,
|
||
|
- [IBPI_PATTERN_ONESHOT_NORMAL] = PCI_NPEM_OK_CAP,
|
||
|
- [IBPI_PATTERN_DEGRADED] = PCI_NPEM_CRA_CAP,
|
||
|
- [IBPI_PATTERN_HOTSPARE] = PCI_NPEM_HOT_SPARE_CAP,
|
||
|
- [IBPI_PATTERN_REBUILD] = PCI_NPEM_REBUILD_CAP,
|
||
|
- [IBPI_PATTERN_FAILED_ARRAY] = PCI_NPEM_FA_CAP,
|
||
|
- [IBPI_PATTERN_PFA] = PCI_NPEM_PFA_CAP,
|
||
|
- [IBPI_PATTERN_FAILED_DRIVE] = PCI_NPEM_FAIL_CAP,
|
||
|
- [IBPI_PATTERN_LOCATE] = PCI_NPEM_LOCATE_CAP,
|
||
|
- [IBPI_PATTERN_LOCATE_OFF] = PCI_NPEM_OK_CAP,
|
||
|
+const struct ibpi_value ibpi_to_npem_capability[] = {
|
||
|
+ {IBPI_PATTERN_NORMAL, PCI_NPEM_OK_CAP},
|
||
|
+ {IBPI_PATTERN_ONESHOT_NORMAL, PCI_NPEM_OK_CAP},
|
||
|
+ {IBPI_PATTERN_DEGRADED, PCI_NPEM_CRA_CAP},
|
||
|
+ {IBPI_PATTERN_HOTSPARE, PCI_NPEM_HOT_SPARE_CAP},
|
||
|
+ {IBPI_PATTERN_REBUILD, PCI_NPEM_REBUILD_CAP},
|
||
|
+ {IBPI_PATTERN_FAILED_ARRAY, PCI_NPEM_FA_CAP},
|
||
|
+ {IBPI_PATTERN_PFA, PCI_NPEM_PFA_CAP},
|
||
|
+ {IBPI_PATTERN_FAILED_DRIVE, PCI_NPEM_FAIL_CAP},
|
||
|
+ {IBPI_PATTERN_LOCATE, PCI_NPEM_LOCATE_CAP},
|
||
|
+ {IBPI_PATTERN_LOCATE_OFF, PCI_NPEM_OK_CAP},
|
||
|
+ {IBPI_PATTERN_UNKNOWN}
|
||
|
};
|
||
|
|
||
|
+static enum ibpi_pattern npem_capability_to_ibpi(const u32 reg)
|
||
|
+{
|
||
|
+ const struct ibpi_value *tmp = ibpi_to_npem_capability;
|
||
|
+
|
||
|
+ while (tmp->ibpi != IBPI_PATTERN_UNKNOWN) {
|
||
|
+ if (reg & tmp->value)
|
||
|
+ break;
|
||
|
+ tmp++;
|
||
|
+ }
|
||
|
+ return tmp->ibpi;
|
||
|
+}
|
||
|
+
|
||
|
static struct pci_access *get_pci_access()
|
||
|
{
|
||
|
struct pci_access *pacc;
|
||
|
@@ -131,8 +146,10 @@ int is_npem_capable(const char *path)
|
||
|
struct pci_access *pacc = get_pci_access();
|
||
|
struct pci_dev *pdev;
|
||
|
|
||
|
- if (!pacc)
|
||
|
+ if (!pacc) {
|
||
|
+ log_error("NPEM: Unable to initialize pci access for %s\n", path);
|
||
|
return 0;
|
||
|
+ }
|
||
|
|
||
|
pdev = get_pci_dev(pacc, path);
|
||
|
|
||
|
@@ -215,7 +232,9 @@ int npem_write(struct block_device *device, enum ibpi_pattern ibpi)
|
||
|
}
|
||
|
|
||
|
reg = read_npem_register(pdev, PCI_NPEM_CAP_REG);
|
||
|
- if ((reg & ibpi_to_npem_capability[ibpi]) == 0) {
|
||
|
+ u32 cap = (u32)get_value_for_ibpi(ibpi, ibpi_to_npem_capability);
|
||
|
+
|
||
|
+ if ((reg & cap) == 0) {
|
||
|
log_debug("NPEM: Controller %s doesn't support %s pattern\n",
|
||
|
npem_cntrl->sysfs_path, ibpi_str[ibpi]);
|
||
|
ibpi = IBPI_PATTERN_NORMAL;
|
||
|
@@ -223,7 +242,8 @@ int npem_write(struct block_device *device, enum ibpi_pattern ibpi)
|
||
|
|
||
|
reg = read_npem_register(pdev, PCI_NPEM_CTRL_REG);
|
||
|
val = (reg & PCI_NPEM_RESERVED);
|
||
|
- val = (val | PCI_NPEM_CAP | ibpi_to_npem_capability[ibpi]);
|
||
|
+ cap = (u32)get_value_for_ibpi(ibpi, ibpi_to_npem_capability);
|
||
|
+ val = (val | PCI_NPEM_CAP | cap);
|
||
|
|
||
|
write_npem_register(pdev, PCI_NPEM_CTRL_REG, val);
|
||
|
if (npem_wait_command(pdev)) {
|
||
|
@@ -244,3 +264,100 @@ char *npem_get_path(const char *cntrl_path)
|
||
|
{
|
||
|
return str_dup(cntrl_path);
|
||
|
}
|
||
|
+
|
||
|
+status_t npem_get_slot(char *device, char *slot_path, struct slot_response *slot_res)
|
||
|
+{
|
||
|
+ struct pci_dev *pdev = NULL;
|
||
|
+ struct block_device *block_device = NULL;
|
||
|
+ struct pci_access *pacc = get_pci_access();
|
||
|
+ status_t status = STATUS_SUCCESS;
|
||
|
+ char *path = NULL;
|
||
|
+ u32 reg;
|
||
|
+
|
||
|
+ if (!pacc) {
|
||
|
+ log_error("NPEM: Unable to initialize pci access for %s\n", path);
|
||
|
+ return STATUS_NULL_POINTER;
|
||
|
+ }
|
||
|
+
|
||
|
+ if (device && device[0] != '\0') {
|
||
|
+ block_device = get_block_device_from_sysfs_path(basename(device));
|
||
|
+ if (block_device)
|
||
|
+ path = block_device->cntrl->sysfs_path;
|
||
|
+ } else if (slot_path && slot_path[0] != '\0') {
|
||
|
+ struct cntrl_device *ctrl_dev;
|
||
|
+
|
||
|
+ list_for_each(sysfs_get_cntrl_devices(), ctrl_dev) {
|
||
|
+ if (ctrl_dev->cntrl_type != CNTRL_TYPE_NPEM)
|
||
|
+ continue;
|
||
|
+ if (strcmp(basename(ctrl_dev->sysfs_path), basename(slot_path)) != 0)
|
||
|
+ continue;
|
||
|
+ path = ctrl_dev->sysfs_path;
|
||
|
+ block_device = get_block_device_from_sysfs_path(path);
|
||
|
+ break;
|
||
|
+ }
|
||
|
+ }
|
||
|
+
|
||
|
+ if (path) {
|
||
|
+ pdev = get_pci_dev(pacc, path);
|
||
|
+ } else {
|
||
|
+ log_debug("NPEM: unable to get sysfs path for the controller.");
|
||
|
+ pci_cleanup(pacc);
|
||
|
+ return STATUS_INVALID_PATH;
|
||
|
+ }
|
||
|
+
|
||
|
+ if (!pdev) {
|
||
|
+ log_error("NPEM: Unable to get pci device for %s\n", path);
|
||
|
+ pci_cleanup(pacc);
|
||
|
+ return STATUS_NULL_POINTER;
|
||
|
+ }
|
||
|
+
|
||
|
+ reg = read_npem_register(pdev, PCI_NPEM_CTRL_REG);
|
||
|
+ slot_res->state = npem_capability_to_ibpi(reg);
|
||
|
+ snprintf(slot_res->slot, PATH_MAX, "%s", path);
|
||
|
+
|
||
|
+ if (block_device)
|
||
|
+ snprintf(slot_res->device, PATH_MAX, "/dev/%s", basename(block_device->sysfs_path));
|
||
|
+ else
|
||
|
+ snprintf(slot_res->device, PATH_MAX, "(empty)");
|
||
|
+
|
||
|
+ pci_free_dev(pdev);
|
||
|
+ pci_cleanup(pacc);
|
||
|
+ return status;
|
||
|
+}
|
||
|
+
|
||
|
+status_t npem_set_slot(char *slot_path, enum ibpi_pattern state)
|
||
|
+{
|
||
|
+ struct pci_dev *pdev = NULL;
|
||
|
+ struct pci_access *pacc = get_pci_access();
|
||
|
+ status_t status = STATUS_SUCCESS;
|
||
|
+ u32 val;
|
||
|
+ u32 reg;
|
||
|
+ u32 cap;
|
||
|
+
|
||
|
+ if (!pacc) {
|
||
|
+ log_error("NPEM: Unable to initialize pci access for %s\n", slot_path);
|
||
|
+ return STATUS_NULL_POINTER;
|
||
|
+ }
|
||
|
+
|
||
|
+ pdev = get_pci_dev(pacc, slot_path);
|
||
|
+ if (!pdev) {
|
||
|
+ log_error("NPEM: Unable to get pci device for %s\n", slot_path);
|
||
|
+ pci_cleanup(pacc);
|
||
|
+ return STATUS_NULL_POINTER;
|
||
|
+ }
|
||
|
+
|
||
|
+ reg = read_npem_register(pdev, PCI_NPEM_CTRL_REG);
|
||
|
+ val = (reg & PCI_NPEM_RESERVED);
|
||
|
+ cap = (u32)get_value_for_ibpi(state, ibpi_to_npem_capability);
|
||
|
+ val = (val | PCI_NPEM_CAP | cap);
|
||
|
+
|
||
|
+ write_npem_register(pdev, PCI_NPEM_CTRL_REG, val);
|
||
|
+ if (npem_wait_command(pdev)) {
|
||
|
+ log_error("NPEM: Write timeout for %s\n", slot_path);
|
||
|
+ status = STATUS_FILE_WRITE_ERROR;
|
||
|
+ }
|
||
|
+
|
||
|
+ pci_free_dev(pdev);
|
||
|
+ pci_cleanup(pacc);
|
||
|
+ return status;
|
||
|
+}
|
||
|
diff --git a/src/npem.h b/src/npem.h
|
||
|
index 5fa3f11d0ba3..8f4dd7b96151 100644
|
||
|
--- a/src/npem.h
|
||
|
+++ b/src/npem.h
|
||
|
@@ -21,9 +21,37 @@
|
||
|
#define NPEM_H_INCLUDED_
|
||
|
#include "block.h"
|
||
|
#include "ibpi.h"
|
||
|
+#include "slot.h"
|
||
|
+#include "status.h"
|
||
|
|
||
|
int is_npem_capable(const char *path);
|
||
|
int npem_write(struct block_device *device, enum ibpi_pattern ibpi);
|
||
|
char *npem_get_path(const char *cntrl_path);
|
||
|
|
||
|
+/**
|
||
|
+ * @brief Gets led state for slot.
|
||
|
+ *
|
||
|
+ * This function finds slot connected to given identifier
|
||
|
+ * and fills slot response related to the slot.
|
||
|
+ *
|
||
|
+ * @param[in] device Requested device name.
|
||
|
+ * @param[in] slot_num Requested identifier of the slot.
|
||
|
+ * @param[in] slot_res Pointer to the slot response.
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+status_t npem_get_slot(char *device, char *slot_num, struct slot_response *slot_res);
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Sets led state for slot.
|
||
|
+ *
|
||
|
+ * This function finds slot connected to given number or device name
|
||
|
+ * and set given led state.
|
||
|
+ *
|
||
|
+ * @param[in] slot_num Requested number of the slot.
|
||
|
+ * @param[in] state IBPI state based on slot request.
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+status_t npem_set_slot(char *slot_num, enum ibpi_pattern state);
|
||
|
#endif // NPEM_H_INCLUDED_
|
||
|
diff --git a/src/pci_slot.c b/src/pci_slot.c
|
||
|
index b15e1947c043..1e03dbabe91d 100644
|
||
|
--- a/src/pci_slot.c
|
||
|
+++ b/src/pci_slot.c
|
||
|
@@ -29,7 +29,9 @@
|
||
|
|
||
|
#include "config.h"
|
||
|
#include "pci_slot.h"
|
||
|
+#include "sysfs.h"
|
||
|
#include "utils.h"
|
||
|
+#include "vmdssd.h"
|
||
|
|
||
|
/*
|
||
|
* Allocates memory for PCI hotplug slot structure and initializes fields of
|
||
|
@@ -44,12 +46,6 @@ struct pci_slot *pci_slot_init(const char *path)
|
||
|
return NULL;
|
||
|
result->sysfs_path = str_dup(path);
|
||
|
result->address = get_text(path, "address");
|
||
|
- result->attention = get_int(path, -1, "attention");
|
||
|
-
|
||
|
- if (result->attention == -1) {
|
||
|
- pci_slot_fini(result);
|
||
|
- return NULL;
|
||
|
- }
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
@@ -66,3 +62,97 @@ void pci_slot_fini(struct pci_slot *slot)
|
||
|
free(slot);
|
||
|
}
|
||
|
}
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Finds PCI slot by number of the slot.
|
||
|
+ *
|
||
|
+ * @param[in] slot_number Number of the slot
|
||
|
+ *
|
||
|
+ * @return Struct with pci slot if successful, otherwise the function returns NULL pointer.
|
||
|
+ */
|
||
|
+static struct pci_slot *find_pci_slot_by_number(char *slot_number)
|
||
|
+{
|
||
|
+ struct pci_slot *slot = NULL;
|
||
|
+ char *temp;
|
||
|
+
|
||
|
+ if (slot_number == NULL)
|
||
|
+ return NULL;
|
||
|
+
|
||
|
+ list_for_each(sysfs_get_pci_slots(), slot) {
|
||
|
+ temp = basename(slot->sysfs_path);
|
||
|
+ if (temp && strncmp(temp, slot_number, PATH_MAX) == 0)
|
||
|
+ return slot;
|
||
|
+ }
|
||
|
+ return NULL;
|
||
|
+}
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Sets the slot response.
|
||
|
+ *
|
||
|
+ * @param[in] slot Struct with PCI slot parameters.
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+static status_t set_slot_response(struct pci_slot *slot, struct slot_response *slot_res)
|
||
|
+{
|
||
|
+ struct block_device *bl_device;
|
||
|
+ status_t status = STATUS_SUCCESS;
|
||
|
+ int attention = get_int(slot->sysfs_path, -1, "attention");
|
||
|
+
|
||
|
+ if (attention == -1)
|
||
|
+ return STATUS_INVALID_STATE;
|
||
|
+
|
||
|
+ slot_res->state = get_ibpi_for_value(attention, ibpi_to_attention);
|
||
|
+ snprintf(slot_res->slot, PATH_MAX, "%s", basename(slot->sysfs_path));
|
||
|
+
|
||
|
+ bl_device = get_block_device_from_sysfs_path(slot->address);
|
||
|
+ if (bl_device)
|
||
|
+ snprintf(slot_res->device, PATH_MAX, "/dev/%s", basename(bl_device->sysfs_path));
|
||
|
+ else
|
||
|
+ snprintf(slot_res->device, PATH_MAX, "(empty)");
|
||
|
+
|
||
|
+ return status;
|
||
|
+}
|
||
|
+
|
||
|
+status_t pci_get_slot(char *device, char *slot_path, struct slot_response *slot_res)
|
||
|
+{
|
||
|
+ struct pci_slot *slot = NULL;
|
||
|
+ struct block_device *block_device = NULL;
|
||
|
+
|
||
|
+ if (device && device[0] != '\0') {
|
||
|
+ char *sub_path = basename(device);
|
||
|
+ if (sub_path == NULL) {
|
||
|
+ log_error("Device name %s is invalid.", device);
|
||
|
+ return STATUS_DATA_ERROR;
|
||
|
+ }
|
||
|
+
|
||
|
+ block_device = get_block_device_from_sysfs_path(sub_path + 1);
|
||
|
+ if (block_device == NULL) {
|
||
|
+ log_error("Device %s not found.", device);
|
||
|
+ return STATUS_DATA_ERROR;
|
||
|
+ }
|
||
|
+ slot = vmdssd_find_pci_slot(block_device->sysfs_path);
|
||
|
+ } else if (slot_path && slot_path[0] != '\0') {
|
||
|
+ slot = find_pci_slot_by_number(basename(slot_path));
|
||
|
+ }
|
||
|
+
|
||
|
+ if (slot == NULL) {
|
||
|
+ log_error("Specified slot was not found.");
|
||
|
+ return STATUS_DATA_ERROR;
|
||
|
+ }
|
||
|
+
|
||
|
+ return set_slot_response(slot, slot_res);
|
||
|
+}
|
||
|
+
|
||
|
+status_t pci_set_slot(char *slot_path, enum ibpi_pattern state)
|
||
|
+{
|
||
|
+ struct pci_slot *slot = NULL;
|
||
|
+
|
||
|
+ slot = find_pci_slot_by_number(basename(slot_path));
|
||
|
+ if (slot == NULL) {
|
||
|
+ log_error("Slot %s not found.", slot_path);
|
||
|
+ return STATUS_NULL_POINTER;
|
||
|
+ }
|
||
|
+
|
||
|
+ return vmdssd_write_attention_buf(slot, state);
|
||
|
+}
|
||
|
diff --git a/src/pci_slot.h b/src/pci_slot.h
|
||
|
index b971de3d89b7..9a181ce6057e 100644
|
||
|
--- a/src/pci_slot.h
|
||
|
+++ b/src/pci_slot.h
|
||
|
@@ -20,6 +20,10 @@
|
||
|
#ifndef PCI_SLOT_H_INCLUDED_
|
||
|
#define PCI_SLOT_H_INCLUDED_
|
||
|
|
||
|
+#include "ibpi.h"
|
||
|
+#include "slot.h"
|
||
|
+#include "status.h"
|
||
|
+
|
||
|
/**
|
||
|
* @brief PCI hotplug slot structure.
|
||
|
*
|
||
|
@@ -35,11 +39,6 @@ struct pci_slot {
|
||
|
* PCI hotplug slot address.
|
||
|
*/
|
||
|
char *address;
|
||
|
-
|
||
|
- /**
|
||
|
- * State of the Amber LED of the PCI slot.
|
||
|
- */
|
||
|
- int attention;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
@@ -69,4 +68,30 @@ struct pci_slot *pci_slot_init(const char *path);
|
||
|
*/
|
||
|
void pci_slot_fini(struct pci_slot *slot);
|
||
|
|
||
|
+/**
|
||
|
+ * @brief Gets led state for slot.
|
||
|
+ *
|
||
|
+ * This function finds slot connected to given identifier
|
||
|
+ * and fills slot response related to the slot.
|
||
|
+ *
|
||
|
+ * @param[in] device Requested device name.
|
||
|
+ * @param[in] slot_num Requested identifier of the slot.
|
||
|
+ * @param[in] slot_res Pointer to the slot response.
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+status_t pci_get_slot(char *device, char *slot_num, struct slot_response *slot_res);
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Sets led state for slot.
|
||
|
+ *
|
||
|
+ * This function finds slot connected to given number or device name
|
||
|
+ * and set given led state.
|
||
|
+ *
|
||
|
+ * @param[in] slot_num Requested number of the slot.
|
||
|
+ * @param[in] state IBPI state based on slot request.
|
||
|
+ *
|
||
|
+ * @return STATUS_SUCCESS if successful, otherwise a valid status_t status code.
|
||
|
+ */
|
||
|
+status_t pci_set_slot(char *slot_num, enum ibpi_pattern state);
|
||
|
#endif // PCI_SLOT_H_INCLUDED_
|
||
|
diff --git a/src/slot.h b/src/slot.h
|
||
|
new file mode 100644
|
||
|
index 000000000000..82b2e9405b89
|
||
|
--- /dev/null
|
||
|
+++ b/src/slot.h
|
||
|
@@ -0,0 +1,63 @@
|
||
|
+/*
|
||
|
+ * Intel(R) Enclosure LED Utilities
|
||
|
+ * Copyright (C) 2022-2022 Intel Corporation.
|
||
|
+ *
|
||
|
+ * This program is free software; you can redistribute it and/or modify it
|
||
|
+ * under the terms and conditions of the GNU General Public License,
|
||
|
+ * version 2, as published by the Free Software Foundation.
|
||
|
+ *
|
||
|
+ * This program is distributed in the hope it will be useful, but WITHOUT
|
||
|
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||
|
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||
|
+ * more details.
|
||
|
+ *
|
||
|
+ * You should have received a copy of the GNU General Public License along with
|
||
|
+ * this program; if not, write to the Free Software Foundation, Inc.,
|
||
|
+ * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
|
||
|
+ *
|
||
|
+ */
|
||
|
+
|
||
|
+#ifndef SLOT_H_
|
||
|
+#define SLOT_H_
|
||
|
+
|
||
|
+#include <stdio.h>
|
||
|
+
|
||
|
+#include "ibpi.h"
|
||
|
+#include "utils.h"
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief slot response parameters
|
||
|
+ *
|
||
|
+ * This structure contains slot properties.
|
||
|
+ */
|
||
|
+struct slot_response {
|
||
|
+ /**
|
||
|
+ * Name of the device.
|
||
|
+ */
|
||
|
+ char device[PATH_MAX];
|
||
|
+
|
||
|
+ /**
|
||
|
+ * Unique slot identifier.
|
||
|
+ */
|
||
|
+ char slot[PATH_MAX];
|
||
|
+
|
||
|
+ /**
|
||
|
+ * IBPI state.
|
||
|
+ */
|
||
|
+ enum ibpi_pattern state;
|
||
|
+};
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Print address, slot identifier and led state.
|
||
|
+ *
|
||
|
+ * @param[in] res Structure with slot response.
|
||
|
+ *
|
||
|
+ * @return This function does not return a value.
|
||
|
+ */
|
||
|
+static inline void print_slot_state(struct slot_response *res)
|
||
|
+{
|
||
|
+ printf("slot: %-15s led state: %-15s device: %-15s\n",
|
||
|
+ basename(res->slot), ibpi2str(res->state), res->device);
|
||
|
+}
|
||
|
+
|
||
|
+#endif // SLOT_H_INCLUDED_
|
||
|
diff --git a/src/sysfs.c b/src/sysfs.c
|
||
|
index 87e83391f328..f1f9caad05a3 100644
|
||
|
--- a/src/sysfs.c
|
||
|
+++ b/src/sysfs.c
|
||
|
@@ -48,7 +48,6 @@
|
||
|
*/
|
||
|
#define SYSFS_CLASS_BLOCK "/sys/block"
|
||
|
#define SYSFS_CLASS_ENCLOSURE "/sys/class/enclosure"
|
||
|
-#define SYSFS_PCI_DEVICES "/sys/bus/pci/devices"
|
||
|
#define SYSFS_PCI_SLOTS "/sys/bus/pci/slots"
|
||
|
|
||
|
/**
|
||
|
diff --git a/src/sysfs.h b/src/sysfs.h
|
||
|
index efebea840cee..ea0109f6aa33 100644
|
||
|
--- a/src/sysfs.h
|
||
|
+++ b/src/sysfs.h
|
||
|
@@ -1,6 +1,6 @@
|
||
|
/*
|
||
|
* Intel(R) Enclosure LED Utilities
|
||
|
- * Copyright (C) 2009-2018 Intel Corporation.
|
||
|
+ * Copyright (C) 2009-2022 Intel Corporation.
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify it
|
||
|
* under the terms and conditions of the GNU General Public License,
|
||
|
@@ -22,6 +22,8 @@
|
||
|
|
||
|
#include "status.h"
|
||
|
|
||
|
+#define SYSFS_PCI_DEVICES "/sys/bus/pci/devices"
|
||
|
+
|
||
|
/**
|
||
|
* @brief Initializes sysfs module.
|
||
|
*
|
||
|
diff --git a/src/utils.c b/src/utils.c
|
||
|
index 7f52194f5d1c..085cdb1d5c2f 100644
|
||
|
--- a/src/utils.c
|
||
|
+++ b/src/utils.c
|
||
|
@@ -505,7 +505,7 @@ int get_log_fd(void)
|
||
|
|
||
|
void print_opt(const char *long_opt, const char *short_opt, const char *desc)
|
||
|
{
|
||
|
- printf("%-20s%-10s%s\n", long_opt, short_opt, desc);
|
||
|
+ printf("%-70s%-40s%s\n", long_opt, short_opt, desc);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
@@ -581,6 +581,13 @@ struct option longopt_all[] = {
|
||
|
[OPT_LIST_CTRL] = {"list-controllers", no_argument, NULL, 'L'},
|
||
|
[OPT_LISTED_ONLY] = {"listed-only", no_argument, NULL, 'x'},
|
||
|
[OPT_FOREGROUND] = {"foreground", no_argument, NULL, '\0'},
|
||
|
+ [OPT_LIST_SLOTS] = {"list-slots", no_argument, NULL, 'P'},
|
||
|
+ [OPT_GET_SLOT] = {"get-slot", no_argument, NULL, 'G'},
|
||
|
+ [OPT_SET_SLOT] = {"set-slot", no_argument, NULL, 'S'},
|
||
|
+ [OPT_CONTROLLER] = {"controller", required_argument, NULL, 'c'},
|
||
|
+ [OPT_DEVICE] = {"device", required_argument, NULL, 'd'},
|
||
|
+ [OPT_SLOT] = {"slot", required_argument, NULL, 'p'},
|
||
|
+ [OPT_STATE] = {"state", required_argument, NULL, 's'},
|
||
|
[OPT_NULL_ELEMENT] = {NULL, no_argument, NULL, '\0'}
|
||
|
};
|
||
|
|
||
|
@@ -699,3 +706,43 @@ const char *ibpi2str(enum ibpi_pattern ibpi)
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Returns value based on IBPI state
|
||
|
+ *
|
||
|
+ * @param[in] value Value for led state.
|
||
|
+ * @param[in] ibpi_values Array with defined IBPI states and values.
|
||
|
+ *
|
||
|
+ * @return Integer value which represents given IBPI state.
|
||
|
+ */
|
||
|
+int get_value_for_ibpi(enum ibpi_pattern ibpi, const struct ibpi_value ibpi_values[])
|
||
|
+{
|
||
|
+ const struct ibpi_value *tmp = ibpi_values;
|
||
|
+
|
||
|
+ while (tmp->ibpi != IBPI_PATTERN_UNKNOWN) {
|
||
|
+ if (tmp->ibpi == ibpi)
|
||
|
+ break;
|
||
|
+ tmp++;
|
||
|
+ }
|
||
|
+ return tmp->value;
|
||
|
+}
|
||
|
+
|
||
|
+/**
|
||
|
+ * @brief Returns IBPI pattern based on value
|
||
|
+ *
|
||
|
+ * @param[in] value Value for led state.
|
||
|
+ * @param[in] ibpi_values Array with defined IBPI states and values.
|
||
|
+ *
|
||
|
+ * @return Enum with IBPI value, which represents given value.
|
||
|
+ */
|
||
|
+enum ibpi_pattern get_ibpi_for_value(const int value, const struct ibpi_value ibpi_values[])
|
||
|
+{
|
||
|
+ const struct ibpi_value *tmp = ibpi_values;
|
||
|
+
|
||
|
+ while (tmp->ibpi != IBPI_PATTERN_UNKNOWN) {
|
||
|
+ if (tmp->value == value)
|
||
|
+ break;
|
||
|
+ tmp++;
|
||
|
+ }
|
||
|
+ return tmp->ibpi;
|
||
|
+}
|
||
|
diff --git a/src/utils.h b/src/utils.h
|
||
|
index 3f5a77fc6f74..5ef3020e6dce 100644
|
||
|
--- a/src/utils.h
|
||
|
+++ b/src/utils.h
|
||
|
@@ -1,6 +1,6 @@
|
||
|
/*
|
||
|
* Intel(R) Enclosure LED Utilities
|
||
|
- * Copyright (C) 2009-2019 Intel Corporation.
|
||
|
+ * Copyright (C) 2009-2022 Intel Corporation.
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify it
|
||
|
* under the terms and conditions of the GNU General Public License,
|
||
|
@@ -59,6 +59,10 @@ struct log_level_info {
|
||
|
int priority;
|
||
|
};
|
||
|
|
||
|
+struct ibpi_value {
|
||
|
+ int ibpi, value;
|
||
|
+};
|
||
|
+
|
||
|
/**
|
||
|
*/
|
||
|
#define PREFIX_DEBUG " DEBUG: "
|
||
|
@@ -418,6 +422,13 @@ enum opt {
|
||
|
OPT_LIST_CTRL,
|
||
|
OPT_LISTED_ONLY,
|
||
|
OPT_FOREGROUND,
|
||
|
+ OPT_LIST_SLOTS,
|
||
|
+ OPT_GET_SLOT,
|
||
|
+ OPT_SET_SLOT,
|
||
|
+ OPT_CONTROLLER,
|
||
|
+ OPT_DEVICE,
|
||
|
+ OPT_SLOT,
|
||
|
+ OPT_STATE,
|
||
|
OPT_NULL_ELEMENT
|
||
|
};
|
||
|
|
||
|
@@ -428,5 +439,7 @@ int get_option_id(const char *optarg);
|
||
|
status_t set_verbose_level(int log_level);
|
||
|
|
||
|
const char *ibpi2str(enum ibpi_pattern ibpi);
|
||
|
+int get_value_for_ibpi(enum ibpi_pattern ibpi, const struct ibpi_value ibpi_values[]);
|
||
|
+enum ibpi_pattern get_ibpi_for_value(const int value, const struct ibpi_value ibpi_values[]);
|
||
|
|
||
|
#endif /* _UTILS_H_INCLUDED_ */
|
||
|
diff --git a/src/vmdssd.c b/src/vmdssd.c
|
||
|
index 3c6d24bdc604..015d3489c276 100644
|
||
|
--- a/src/vmdssd.c
|
||
|
+++ b/src/vmdssd.c
|
||
|
@@ -1,6 +1,6 @@
|
||
|
/*
|
||
|
* Intel(R) Enclosure LED Utilities
|
||
|
- * Copyright (c) 2016-2019, Intel Corporation
|
||
|
+ * Copyright (c) 2016-2022, Intel Corporation
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify it
|
||
|
* under the terms and conditions of the GNU General Public License,
|
||
|
@@ -37,6 +37,13 @@
|
||
|
#define ATTENTION_REBUILD 0x5 /* (0101) Attention On, Power On */
|
||
|
#define ATTENTION_FAILURE 0xD /* (1101) Attention On, Power Off */
|
||
|
|
||
|
+struct ibpi_value ibpi_to_attention[] = {
|
||
|
+ {IBPI_PATTERN_LOCATE, ATTENTION_LOCATE},
|
||
|
+ {IBPI_PATTERN_FAILED_DRIVE, ATTENTION_FAILURE},
|
||
|
+ {IBPI_PATTERN_REBUILD, ATTENTION_REBUILD},
|
||
|
+ {IBPI_PATTERN_LOCATE_OFF, ATTENTION_OFF}
|
||
|
+};
|
||
|
+
|
||
|
#define SYSFS_PCIEHP "/sys/module/pciehp"
|
||
|
|
||
|
static char *get_slot_from_syspath(char *path)
|
||
|
@@ -61,24 +68,6 @@ static char *get_slot_from_syspath(char *path)
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
-static void get_ctrl(enum ibpi_pattern ibpi, uint16_t *new)
|
||
|
-{
|
||
|
- switch (ibpi) {
|
||
|
- case IBPI_PATTERN_LOCATE:
|
||
|
- *new = ATTENTION_LOCATE;
|
||
|
- break;
|
||
|
- case IBPI_PATTERN_FAILED_DRIVE:
|
||
|
- *new = ATTENTION_FAILURE;
|
||
|
- break;
|
||
|
- case IBPI_PATTERN_REBUILD:
|
||
|
- *new = ATTENTION_REBUILD;
|
||
|
- break;
|
||
|
- default:
|
||
|
- *new = ATTENTION_OFF;
|
||
|
- break;
|
||
|
- }
|
||
|
-}
|
||
|
-
|
||
|
static int check_slot_module(const char *slot_path)
|
||
|
{
|
||
|
char module_path[PATH_MAX], real_module_path[PATH_MAX];
|
||
|
@@ -120,11 +109,29 @@ struct pci_slot *vmdssd_find_pci_slot(char *device_path)
|
||
|
return slot;
|
||
|
}
|
||
|
|
||
|
-int vmdssd_write(struct block_device *device, enum ibpi_pattern ibpi)
|
||
|
+status_t vmdssd_write_attention_buf(struct pci_slot *slot, enum ibpi_pattern ibpi)
|
||
|
{
|
||
|
char attention_path[PATH_MAX];
|
||
|
char buf[WRITE_BUFFER_SIZE];
|
||
|
uint16_t val;
|
||
|
+
|
||
|
+ log_debug("%s before: 0x%x\n", slot->address,
|
||
|
+ get_int(slot->sysfs_path, 0, "attention"));
|
||
|
+ val = get_value_for_ibpi(ibpi, ibpi_to_attention);
|
||
|
+ snprintf(buf, WRITE_BUFFER_SIZE, "%u", val);
|
||
|
+ snprintf(attention_path, PATH_MAX, "%s/attention", slot->sysfs_path);
|
||
|
+ if (buf_write(attention_path, buf) != (ssize_t) strnlen(buf, WRITE_BUFFER_SIZE)) {
|
||
|
+ log_error("%s write error: %d\n", slot->sysfs_path, errno);
|
||
|
+ return STATUS_FILE_WRITE_ERROR;
|
||
|
+ }
|
||
|
+ log_debug("%s after: 0x%x\n", slot->address,
|
||
|
+ get_int(slot->sysfs_path, 0, "attention"));
|
||
|
+
|
||
|
+ return STATUS_SUCCESS;
|
||
|
+}
|
||
|
+
|
||
|
+int vmdssd_write(struct block_device *device, enum ibpi_pattern ibpi)
|
||
|
+{
|
||
|
struct pci_slot *slot;
|
||
|
char *short_name = strrchr(device->sysfs_path, '/');
|
||
|
|
||
|
@@ -145,21 +152,7 @@ int vmdssd_write(struct block_device *device, enum ibpi_pattern ibpi)
|
||
|
__set_errno_and_return(ENODEV);
|
||
|
}
|
||
|
|
||
|
- log_debug("%s before: 0x%x\n", short_name,
|
||
|
- get_int(slot->sysfs_path, 0, "attention"));
|
||
|
-
|
||
|
- get_ctrl(ibpi, &val);
|
||
|
- snprintf(buf, WRITE_BUFFER_SIZE, "%u", val);
|
||
|
- snprintf(attention_path, PATH_MAX, "%s/attention", slot->sysfs_path);
|
||
|
- if (buf_write(attention_path, buf) != (ssize_t) strnlen(buf, WRITE_BUFFER_SIZE)) {
|
||
|
- log_error("%s write error: %d\n", slot->sysfs_path, errno);
|
||
|
- return -1;
|
||
|
- }
|
||
|
-
|
||
|
- log_debug("%s after: 0x%x\n", short_name,
|
||
|
- get_int(slot->sysfs_path, 0, "attention"));
|
||
|
-
|
||
|
- return 0;
|
||
|
+ return vmdssd_write_attention_buf(slot, ibpi);
|
||
|
}
|
||
|
|
||
|
char *vmdssd_get_path(const char *cntrl_path)
|
||
|
diff --git a/src/vmdssd.h b/src/vmdssd.h
|
||
|
index 4c90fcb35ea2..f07d989280bc 100644
|
||
|
--- a/src/vmdssd.h
|
||
|
+++ b/src/vmdssd.h
|
||
|
@@ -1,6 +1,6 @@
|
||
|
/*
|
||
|
* Intel(R) Enclosure LED Utilities
|
||
|
- * Copyright (c) 2016-2019, Intel Corporation
|
||
|
+ * Copyright (c) 2016-2022, Intel Corporation
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify it
|
||
|
* under the terms and conditions of the GNU General Public License,
|
||
|
@@ -22,9 +22,18 @@
|
||
|
|
||
|
#include "block.h"
|
||
|
#include "ibpi.h"
|
||
|
+#include "utils.h"
|
||
|
+
|
||
|
+#define ATTENTION_OFF 0xF /* (1111) Attention Off, Power Off */
|
||
|
+#define ATTENTION_LOCATE 0x7 /* (0111) Attention Off, Power On */
|
||
|
+#define ATTENTION_REBUILD 0x5 /* (0101) Attention On, Power On */
|
||
|
+#define ATTENTION_FAILURE 0xD /* (1101) Attention On, Power Off */
|
||
|
+
|
||
|
+extern struct ibpi_value ibpi_to_attention[];
|
||
|
|
||
|
int vmdssd_write(struct block_device *device, enum ibpi_pattern ibpi);
|
||
|
char *vmdssd_get_path(const char *cntrl_path);
|
||
|
struct pci_slot *vmdssd_find_pci_slot(char *device_path);
|
||
|
+status_t vmdssd_write_attention_buf(struct pci_slot *slot, enum ibpi_pattern ibpi);
|
||
|
|
||
|
#endif
|
||
|
--
|
||
|
2.34.1
|
||
|
|