forked from pool/qclib
Accepting request 782230 from hardware
OBS-URL: https://build.opensuse.org/request/show/782230 OBS-URL: https://build.opensuse.org/package/show/openSUSE:Factory/qclib?expand=0&rev=8
This commit is contained in:
commit
418b057cf0
873
qclib-2.0.1-add-support-for-secure-boot-related-attributes.patch
Normal file
873
qclib-2.0.1-add-support-for-secure-boot-related-attributes.patch
Normal file
@ -0,0 +1,873 @@
|
|||||||
|
commit 11c01ad586bbf242ab9b929cb3beabb4783da4ee
|
||||||
|
Author: Stefan Raspl <raspl@linux.ibm.com>
|
||||||
|
Date: Sat Feb 22 14:04:12 2020 -0700
|
||||||
|
|
||||||
|
Add new attributes in support of secure boot in all final layers
|
||||||
|
|
||||||
|
Per a request from SUSE, adding the two attributes that will indicate
|
||||||
|
whether secure boot is available and in use. SUSE was actually only asking
|
||||||
|
for one of the attributes, but figured it really makes sense to have both.
|
||||||
|
Note that we will ever get the values for the final layer, as documented in
|
||||||
|
the README entry. Otherwise, we would need to have the values provided via
|
||||||
|
e.g. STHYI for further layers - which is currently not architectured.
|
||||||
|
Might lead to some confusion, when the topmost layer indicates support,
|
||||||
|
but the lower layers have no info available. Was thinking about post-
|
||||||
|
processing the data into lower layers, but that won't work at all: In case
|
||||||
|
secure boot is available but not set, we cannot know at which of the lower
|
||||||
|
layers things broke.
|
||||||
|
|
||||||
|
Signed-off-by: Stefan Raspl <raspl@linux.ibm.com>
|
||||||
|
|
||||||
|
diff --git a/Makefile b/Makefile
|
||||||
|
index cff31899..376a5a51 100644
|
||||||
|
--- a/Makefile
|
||||||
|
+++ b/Makefile
|
||||||
|
@@ -7,8 +7,8 @@ VERSION = 2.0.x
|
||||||
|
VERSION = 2.0.1
|
||||||
|
VERM = $(shell echo $(VERSION) | cut -d '.' -f 1)
|
||||||
|
CFLAGS ?= -g -Wall -O2
|
||||||
|
-CFILES = query_capacity.c query_capacity_data.c query_capacity_sysinfo.c query_capacity_ocf.c \
|
||||||
|
- query_capacity_hypfs.c query_capacity_sthyi.c
|
||||||
|
+CFILES = query_capacity.c query_capacity_data.c query_capacity_sysinfo.c \
|
||||||
|
+ query_capacity_sysfs.c query_capacity_hypfs.c query_capacity_sthyi.c
|
||||||
|
OBJECTS = $(patsubst %.c,%.o,$(CFILES))
|
||||||
|
.SUFFIXES: .o .c
|
||||||
|
|
||||||
|
diff --git a/README b/README
|
||||||
|
index 2521d604..49d6fec2 100644
|
||||||
|
--- a/README
|
||||||
|
+++ b/README
|
||||||
|
@@ -27,9 +27,9 @@ by:
|
||||||
|
Instruction'.
|
||||||
|
* hypfs file system - for more information, refer to 'Device Drivers,
|
||||||
|
Features, and Commands', chapter 'S/390 hypervisor file system'.
|
||||||
|
- * Firmware interface /sys/firmware/ocf - for more information, refer to
|
||||||
|
- 'Device Drivers, Features, and Commands', chapter 'Identifying the z
|
||||||
|
- Systems hardware'.
|
||||||
|
+ * Firmware and other interfaces as made available through sysfs. For more
|
||||||
|
+ information, refer to 'Device Drivers, Features, and Commands', chapter
|
||||||
|
+ 'Identifying the z Systems hardware'.
|
||||||
|
|
||||||
|
Please refer to:
|
||||||
|
http://www.ibm.com/developerworks/linux/linux390/qclib.html
|
||||||
|
diff --git a/qc_test.c b/qc_test.c
|
||||||
|
index 8a1db825..a83ae199 100644
|
||||||
|
--- a/qc_test.c
|
||||||
|
+++ b/qc_test.c
|
||||||
|
@@ -77,6 +77,8 @@ const char *attr2char(enum qc_attr_id id) {
|
||||||
|
case qc_capping: return "qc_capping";
|
||||||
|
case qc_capping_num: return "qc_capping_num";
|
||||||
|
case qc_mobility_enabled: return "qc_mobility_enabled";
|
||||||
|
+ case qc_has_secure: return "qc_has_secure";
|
||||||
|
+ case qc_secure: return "qc_secure";
|
||||||
|
case qc_has_multiple_cpu_types: return "qc_has_multiple_cpu_types";
|
||||||
|
case qc_cp_dispatch_limithard: return "qc_cp_dispatch_limithard";
|
||||||
|
case qc_ifl_dispatch_limithard: return "qc_ifl_dispatch_limithard";
|
||||||
|
@@ -476,7 +478,7 @@ void print_cec_information(void *hdl, int layer, int indent) {
|
||||||
|
print_string_attr(hdl, qc_layer_category, "n/a", layer, indent);
|
||||||
|
print_int_attr(hdl, qc_layer_type_num, "n/a", layer, indent);
|
||||||
|
print_int_attr(hdl, qc_layer_category_num, "n/a", layer, indent);
|
||||||
|
- print_string_attr(hdl, qc_layer_name, "O V", layer, indent);
|
||||||
|
+ print_string_attr(hdl, qc_layer_name, "F V", layer, indent);
|
||||||
|
print_string_attr(hdl, qc_manufacturer, "S V", layer, indent);
|
||||||
|
print_string_attr(hdl, qc_type, "S V", layer, indent);
|
||||||
|
print_string_attr(hdl, qc_type_name, "S ", layer, indent);
|
||||||
|
@@ -551,6 +553,8 @@ void print_lpar_information(void *hdl, int layer, int indent) {
|
||||||
|
print_string_attr(hdl, qc_partition_char, "S ", layer, indent);
|
||||||
|
print_int_attr(hdl, qc_partition_char_num, "S ", layer, indent);
|
||||||
|
print_int_attr(hdl, qc_adjustment, "S ", layer, indent);
|
||||||
|
+ print_int_attr(hdl, qc_has_secure, "F ", layer, indent);
|
||||||
|
+ print_int_attr(hdl, qc_secure, "F ", layer, indent);
|
||||||
|
|
||||||
|
print_break();
|
||||||
|
print_int_attr(hdl, qc_num_core_total, "S ", layer, indent);
|
||||||
|
@@ -717,6 +721,8 @@ void print_zoszcxserver_information(void *hdl, int layer, int indent) {
|
||||||
|
print_string_attr(hdl, qc_layer_name, "S V", layer, indent);
|
||||||
|
print_string_attr(hdl, qc_capping, " H ", layer, indent);
|
||||||
|
print_int_attr(hdl, qc_capping_num, " H ", layer, indent);
|
||||||
|
+ print_int_attr(hdl, qc_has_secure, "F ", layer, indent);
|
||||||
|
+ print_int_attr(hdl, qc_secure, "F ", layer, indent);
|
||||||
|
|
||||||
|
print_break();
|
||||||
|
print_int_attr(hdl, qc_num_cpu_total, "S V", layer, indent);
|
||||||
|
@@ -759,6 +765,9 @@ void print_zvmguest_information(void *hdl, int layer, int indent) {
|
||||||
|
print_string_attr(hdl, qc_layer_name, "S V", layer, indent);
|
||||||
|
print_string_attr(hdl, qc_capping, " H ", layer, indent);
|
||||||
|
print_int_attr(hdl, qc_capping_num, " H ", layer, indent);
|
||||||
|
+ print_int_attr(hdl, qc_mobility_enabled, " V", layer, indent);
|
||||||
|
+ print_int_attr(hdl, qc_has_secure, "F ", layer, indent);
|
||||||
|
+ print_int_attr(hdl, qc_secure, "F ", layer, indent);
|
||||||
|
|
||||||
|
print_break();
|
||||||
|
print_int_attr(hdl, qc_num_cpu_total, "S V", layer, indent);
|
||||||
|
@@ -791,7 +800,6 @@ void print_zvmguest_information(void *hdl, int layer, int indent) {
|
||||||
|
print_int_attr(hdl, qc_ziip_capped_capacity, " V", layer, indent);
|
||||||
|
|
||||||
|
print_break();
|
||||||
|
- print_int_attr(hdl, qc_mobility_enabled, " V", layer, indent);
|
||||||
|
print_int_attr(hdl, qc_has_multiple_cpu_types, " V", layer, indent);
|
||||||
|
|
||||||
|
// check an attribute that only exists at a different layer
|
||||||
|
@@ -835,6 +843,8 @@ void print_kvmguest_information(void *hdl, int layer, int indent) {
|
||||||
|
print_string_attr(hdl, qc_layer_name, "S ", layer, indent);
|
||||||
|
print_string_attr(hdl, qc_layer_extended_name, "S ", layer, indent);
|
||||||
|
print_string_attr(hdl, qc_layer_uuid, "S ", layer, indent);
|
||||||
|
+ print_int_attr(hdl, qc_has_secure, "F ", layer, indent);
|
||||||
|
+ print_int_attr(hdl, qc_secure, "F ", layer, indent);
|
||||||
|
|
||||||
|
print_break();
|
||||||
|
print_int_attr(hdl, qc_num_cpu_total, "S ", layer, indent);
|
||||||
|
diff --git a/query_capacity.c b/query_capacity.c
|
||||||
|
index 3a109520..644508da 100644
|
||||||
|
--- a/query_capacity.c
|
||||||
|
+++ b/query_capacity.c
|
||||||
|
@@ -679,7 +679,8 @@ fail:
|
||||||
|
|
||||||
|
static void *_qc_open(struct qc_handle *hdl, int *rc) {
|
||||||
|
// sysinfo needs to be handled first, or our LGM check later on will have loopholes
|
||||||
|
- struct qc_data_src *src, *sources[] = {&sysinfo, &ocf, &hypfs, &sthyi, NULL};
|
||||||
|
+ // sysfs needs to be handled last, as part of the attributes apply to top-most layer only
|
||||||
|
+ struct qc_data_src *src, *sources[] = {&sysinfo, &hypfs, &sthyi, &sysfs, NULL};
|
||||||
|
struct qc_handle *lparhdl;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
@@ -907,7 +908,7 @@ static struct qc_handle *qc_get_layer_handle(void *config, int layer) {
|
||||||
|
}
|
||||||
|
|
||||||
|
static int qc_is_attr_id_valid(enum qc_attr_id id) {
|
||||||
|
- return id <= qc_num_ziip_threads;
|
||||||
|
+ return id <= qc_secure;
|
||||||
|
}
|
||||||
|
|
||||||
|
int qc_get_attribute_string(void *cfg, enum qc_attr_id id, int layer, const char **value) {
|
||||||
|
diff --git a/query_capacity.h b/query_capacity.h
|
||||||
|
index 5786a9b2..69d71d6e 100644
|
||||||
|
--- a/query_capacity.h
|
||||||
|
+++ b/query_capacity.h
|
||||||
|
@@ -21,7 +21,7 @@
|
||||||
|
* what layers. The letter encoding in the '\c Src' column describes how the
|
||||||
|
* value is gained:
|
||||||
|
* - **S**: Provided by \c /proc/sysinfo, which is present in all Linux on z flavors.
|
||||||
|
- * - **O**: Provided by \c /sys/firmware/ocf. Available in Linux kernel 3.0 or higher.
|
||||||
|
+ * - **F**: Provided by firmware as made available through the \c sysfs filesystem.
|
||||||
|
* - **H**: Provided by hypfs, which is (preferably) available through \c debugfs at
|
||||||
|
* \c /sys/kernel/debug/s390_hypfs, or \c s390_hypfs (typically mounted at
|
||||||
|
* \c /sys/hypervisor/s390).
|
||||||
|
@@ -86,11 +86,11 @@
|
||||||
|
* #qc_layer_category_num | int | | Hardcoded to \c #QC_LAYER_CAT_HOST
|
||||||
|
* #qc_layer_type |string| | Hardcoded to \c "CEC"
|
||||||
|
* #qc_layer_category |string| | Hardcoded to \c "HOST"
|
||||||
|
- * #qc_layer_name |string|<CODE>O V</CODE>| CPC name of machine
|
||||||
|
+ * #qc_layer_name |string|<CODE>F V</CODE>| CPC name of machine. Available in Linux kernel 3.0 or higher.
|
||||||
|
* #qc_manufacturer |string|<CODE>S V</CODE>| \n
|
||||||
|
* #qc_type |string|<CODE>S V</CODE>| \n
|
||||||
|
* #qc_type_name |string|<CODE>S </CODE>| \n
|
||||||
|
- * #qc_type_family | int |<CODE>S </CODE>| \n
|
||||||
|
+ * #qc_type_family | int |<CODE>S </CODE>| \n
|
||||||
|
* #qc_model_capacity |string|<CODE>S </CODE>| \n
|
||||||
|
* #qc_model |string|<CODE>S </CODE>| \n
|
||||||
|
* #qc_sequence_code |string|<CODE>S V</CODE>| \n
|
||||||
|
@@ -143,6 +143,8 @@
|
||||||
|
* #qc_partition_char |string|<CODE>S </CODE>| \n
|
||||||
|
* #qc_partition_char_num | int |<CODE>S </CODE>| \n
|
||||||
|
* #qc_adjustment | int |<CODE>S </CODE>| \n
|
||||||
|
+ * #qc_has_secure | int |<CODE>F </CODE>| \n
|
||||||
|
+ * #qc_secure | int |<CODE>F </CODE>| \n
|
||||||
|
* #qc_num_core_total | int |<CODE>S</CODE>| Total number of CPs and IFLs configured in the LPARs activation profile
|
||||||
|
* #qc_num_core_configured | int |<CODE>S </CODE>| <b>Note</b>: \b [5]
|
||||||
|
* #qc_num_core_standby | int |<CODE>S </CODE>| Operational cores that require add'l configuration within the LPAR image to become usable<br><b>Note</b>: \b [5]
|
||||||
|
@@ -225,6 +227,9 @@
|
||||||
|
* #qc_layer_name |string|<CODE>S V</CODE>| Userid of guest
|
||||||
|
* #qc_capping |string|<CODE> H</CODE>| \n
|
||||||
|
* #qc_capping_num | int |<CODE> H</CODE>| \n
|
||||||
|
+ * #qc_mobility_enabled | int |<CODE> V</CODE>| \n
|
||||||
|
+ * #qc_has_secure | int |<CODE>F </CODE>| \n
|
||||||
|
+ * #qc_secure | int |<CODE>F </CODE>| \n
|
||||||
|
* #qc_num_cpu_total | int |<CODE>S V</CODE>| Sum of #qc_num_cpu_configured, #qc_num_cpu_standby and #qc_num_cpu_reserved, or #qc_num_cpu_dedicated and #qc_num_cpu_shared
|
||||||
|
* #qc_num_cpu_configured | int |<CODE>S </CODE>| \n
|
||||||
|
* #qc_num_cpu_standby | int |<CODE>S </CODE>| \n
|
||||||
|
@@ -240,7 +245,6 @@
|
||||||
|
* #qc_num_ziip_total | int |<CODE> V</CODE>| Sum of #qc_num_ziip_dedicated and #qc_num_ziip_shared<br>Reported in unit of CPUs
|
||||||
|
* #qc_num_ziip_dedicated | int |<CODE> V</CODE>| Reported in unit of CPUs
|
||||||
|
* #qc_num_ziip_shared | int |<CODE> V</CODE>| Reported in unit of CPUs
|
||||||
|
- * #qc_mobility_enabled | int |<CODE> V</CODE>| \n
|
||||||
|
* #qc_has_multiple_cpu_types | int |<CODE> V</CODE>| \n
|
||||||
|
* #qc_cp_dispatch_limithard | int |<CODE> V</CODE>| \n
|
||||||
|
* #qc_cp_dispatch_type | int |<CODE> V</CODE>| Only set in presence of CPs
|
||||||
|
@@ -300,6 +304,8 @@
|
||||||
|
* #qc_layer_name |string|<CODE>S V</CODE>| Userid of guest
|
||||||
|
* #qc_capping |string|<CODE> H</CODE>| \n
|
||||||
|
* #qc_capping_num | int |<CODE> H</CODE>| \n
|
||||||
|
+ * #qc_has_secure | int |<CODE>F </CODE>| \n
|
||||||
|
+ * #qc_secure | int |<CODE>F </CODE>| \n
|
||||||
|
* #qc_num_cpu_total | int |<CODE>S V</CODE>| Sum of #qc_num_cpu_configured, #qc_num_cpu_standby and #qc_num_cpu_reserved, or #qc_num_cpu_dedicated and #qc_num_cpu_shared
|
||||||
|
* #qc_num_cpu_configured | int |<CODE>S </CODE>| \n
|
||||||
|
* #qc_num_cpu_standby | int |<CODE>S </CODE>| \n
|
||||||
|
@@ -349,6 +355,8 @@
|
||||||
|
* #qc_layer_name |string|<CODE>S </CODE>| Guest name truncated to 8 characters<br><b>Note</b>: \b [1]
|
||||||
|
* #qc_layer_extended_name |string|<CODE>S </CODE>| Guest name with up to 256 characters<br><b>Note</b>: Requires Linux kernel 3.19 or higher, [1]
|
||||||
|
* #qc_layer_uuid |string|<CODE>S </CODE>| <b>Note</b>: Requires Linux kernel 3.19 or higher
|
||||||
|
+ * #qc_has_secure | int |<CODE>F </CODE>| \n
|
||||||
|
+ * #qc_secure | int |<CODE>F </CODE>| \n
|
||||||
|
* #qc_num_cpu_total | int |<CODE>S </CODE>| Sum of #qc_num_cpu_configured, #qc_num_cpu_standby and #qc_num_cpu_reserved, or #qc_num_cpu_dedicated and #qc_num_cpu_shared
|
||||||
|
* #qc_num_cpu_configured | int |<CODE>S </CODE>| \n
|
||||||
|
* #qc_num_cpu_standby | int |<CODE>S </CODE>| \n
|
||||||
|
@@ -533,6 +541,14 @@ enum qc_attr_id {
|
||||||
|
/** Deprecated, see #qc_mobility_enabled */
|
||||||
|
qc_mobility_eligible = 32,
|
||||||
|
#endif
|
||||||
|
+ /** Indicates whether secure boot is available to the entity.
|
||||||
|
+ Requires Linux kernel 5.3 or later.
|
||||||
|
+ Note: This attribute is only ever available for the topmost layer. */
|
||||||
|
+ qc_has_secure = 77,
|
||||||
|
+ /** Indicates whether entity was booted using the secure boot feature
|
||||||
|
+ Requires Linux kernel 5.3 or later.
|
||||||
|
+ Note: This attribute is only ever available for the topmost layer. */
|
||||||
|
+ qc_secure = 78,
|
||||||
|
/** Model identifier, see \c STSI instruction in [2] */
|
||||||
|
qc_model = 33,
|
||||||
|
/** Model capacity of machine, see \c STSI instruction in [2] */
|
||||||
|
diff --git a/query_capacity_data.c b/query_capacity_data.c
|
||||||
|
index 9147703f..afc2b5f6 100644
|
||||||
|
--- a/query_capacity_data.c
|
||||||
|
+++ b/query_capacity_data.c
|
||||||
|
@@ -122,6 +122,8 @@ struct qc_lpar {
|
||||||
|
char layer_extended_name[QC_LEN_LAYER_EXTENDED_NAME];
|
||||||
|
char layer_uuid[QC_LEN_LAYER_UUID];
|
||||||
|
int adjustment;
|
||||||
|
+ int has_secure;
|
||||||
|
+ int secure;
|
||||||
|
int num_core_total;
|
||||||
|
int num_core_configured;
|
||||||
|
int num_core_standby;
|
||||||
|
@@ -201,6 +203,9 @@ struct qc_zvm_guest {
|
||||||
|
char layer_name[QC_LEN_LAYER_NAME];
|
||||||
|
char capping[QC_LEN_CAPPING];
|
||||||
|
int capping_num;
|
||||||
|
+ int mobility_enabled;
|
||||||
|
+ int has_secure;
|
||||||
|
+ int secure;
|
||||||
|
int num_cpu_total;
|
||||||
|
int num_cpu_configured;
|
||||||
|
int num_cpu_standby;
|
||||||
|
@@ -216,7 +221,6 @@ struct qc_zvm_guest {
|
||||||
|
int num_ziip_total;
|
||||||
|
int num_ziip_dedicated;
|
||||||
|
int num_ziip_shared;
|
||||||
|
- int mobility_enabled;
|
||||||
|
int has_multiple_cpu_types;
|
||||||
|
int cp_dispatch_limithard;
|
||||||
|
int cp_dispatch_type;
|
||||||
|
@@ -273,6 +277,8 @@ struct qc_zos_zcx_server {
|
||||||
|
char layer_name[QC_LEN_LAYER_NAME];
|
||||||
|
char capping[QC_LEN_CAPPING];
|
||||||
|
int capping_num;
|
||||||
|
+ int has_secure;
|
||||||
|
+ int secure;
|
||||||
|
int num_cpu_total;
|
||||||
|
int num_cpu_configured;
|
||||||
|
int num_cpu_standby;
|
||||||
|
@@ -320,6 +326,8 @@ struct qc_kvm_guest {
|
||||||
|
char layer_name[QC_LEN_LAYER_NAME];
|
||||||
|
char layer_extended_name[QC_LEN_LAYER_EXTENDED_NAME];
|
||||||
|
char layer_uuid[QC_LEN_LAYER_UUID];
|
||||||
|
+ int has_secure;
|
||||||
|
+ int secure;
|
||||||
|
int num_cpu_total;
|
||||||
|
int num_cpu_configured;
|
||||||
|
int num_cpu_standby;
|
||||||
|
@@ -408,6 +416,8 @@ static struct qc_attr lpar_attrs[] = {
|
||||||
|
{qc_layer_extended_name, string, offsetof(struct qc_lpar, layer_extended_name)},
|
||||||
|
{qc_layer_uuid, string, offsetof(struct qc_lpar, layer_uuid)},
|
||||||
|
{qc_adjustment, integer, offsetof(struct qc_lpar, adjustment)},
|
||||||
|
+ {qc_has_secure, integer, offsetof(struct qc_lpar, has_secure)},
|
||||||
|
+ {qc_secure, integer, offsetof(struct qc_lpar, secure)},
|
||||||
|
{qc_num_core_total, integer, offsetof(struct qc_lpar, num_core_total)},
|
||||||
|
{qc_num_core_configured, integer, offsetof(struct qc_lpar, num_core_configured)},
|
||||||
|
{qc_num_core_standby, integer, offsetof(struct qc_lpar, num_core_standby)},
|
||||||
|
@@ -547,6 +557,9 @@ static struct qc_attr zvm_guest_attrs[] = {
|
||||||
|
{qc_layer_name, string, offsetof(struct qc_zvm_guest, layer_name)},
|
||||||
|
{qc_capping, string, offsetof(struct qc_zvm_guest, capping)},
|
||||||
|
{qc_capping_num, integer, offsetof(struct qc_zvm_guest, capping_num)},
|
||||||
|
+ {qc_mobility_enabled, integer, offsetof(struct qc_zvm_guest, mobility_enabled)},
|
||||||
|
+ {qc_has_secure, integer, offsetof(struct qc_zvm_guest, has_secure)},
|
||||||
|
+ {qc_secure, integer, offsetof(struct qc_zvm_guest, secure)},
|
||||||
|
{qc_num_cpu_total, integer, offsetof(struct qc_zvm_guest, num_cpu_total)},
|
||||||
|
{qc_num_cpu_configured, integer, offsetof(struct qc_zvm_guest, num_cpu_configured)},
|
||||||
|
{qc_num_cpu_standby, integer, offsetof(struct qc_zvm_guest, num_cpu_standby)},
|
||||||
|
@@ -562,7 +575,6 @@ static struct qc_attr zvm_guest_attrs[] = {
|
||||||
|
{qc_num_ziip_total, integer, offsetof(struct qc_zvm_guest, num_ziip_total)},
|
||||||
|
{qc_num_ziip_dedicated, integer, offsetof(struct qc_zvm_guest, num_ziip_dedicated)},
|
||||||
|
{qc_num_ziip_shared, integer, offsetof(struct qc_zvm_guest, num_ziip_shared)},
|
||||||
|
- {qc_mobility_enabled, integer, offsetof(struct qc_zvm_guest, mobility_enabled)},
|
||||||
|
{qc_has_multiple_cpu_types, integer, offsetof(struct qc_zvm_guest, has_multiple_cpu_types)},
|
||||||
|
{qc_cp_dispatch_limithard, integer, offsetof(struct qc_zvm_guest, cp_dispatch_limithard)},
|
||||||
|
{qc_cp_capped_capacity, integer, offsetof(struct qc_zvm_guest, cp_capped_capacity)},
|
||||||
|
@@ -584,6 +596,8 @@ static struct qc_attr zos_zcx_server_attrs[] = {
|
||||||
|
{qc_layer_name, string, offsetof(struct qc_zos_zcx_server, layer_name)},
|
||||||
|
{qc_capping, string, offsetof(struct qc_zos_zcx_server, capping)},
|
||||||
|
{qc_capping_num, integer, offsetof(struct qc_zos_zcx_server, capping_num)},
|
||||||
|
+ {qc_has_secure, integer, offsetof(struct qc_zos_zcx_server, has_secure)},
|
||||||
|
+ {qc_secure, integer, offsetof(struct qc_zos_zcx_server, secure)},
|
||||||
|
{qc_num_cpu_total, integer, offsetof(struct qc_zos_zcx_server, num_cpu_total)},
|
||||||
|
{qc_num_cpu_configured, integer, offsetof(struct qc_zos_zcx_server, num_cpu_configured)},
|
||||||
|
{qc_num_cpu_standby, integer, offsetof(struct qc_zos_zcx_server, num_cpu_standby)},
|
||||||
|
@@ -614,6 +628,8 @@ static struct qc_attr kvm_guest_attrs[] = {
|
||||||
|
{qc_layer_name, string, offsetof(struct qc_kvm_guest, layer_name)},
|
||||||
|
{qc_layer_extended_name, string, offsetof(struct qc_kvm_guest, layer_extended_name)},
|
||||||
|
{qc_layer_uuid, string, offsetof(struct qc_kvm_guest, layer_uuid)},
|
||||||
|
+ {qc_has_secure, integer, offsetof(struct qc_kvm_guest, has_secure)},
|
||||||
|
+ {qc_secure, integer, offsetof(struct qc_kvm_guest, secure)},
|
||||||
|
{qc_num_cpu_total, integer, offsetof(struct qc_kvm_guest, num_cpu_total)},
|
||||||
|
{qc_num_cpu_configured, integer, offsetof(struct qc_kvm_guest, num_cpu_configured)},
|
||||||
|
{qc_num_cpu_standby, integer, offsetof(struct qc_kvm_guest, num_cpu_standby)},
|
||||||
|
@@ -681,6 +697,8 @@ const char *qc_attr_id_to_char(struct qc_handle *hdl, enum qc_attr_id id) {
|
||||||
|
case qc_capping: return "capping";
|
||||||
|
case qc_capping_num: return "capping_num";
|
||||||
|
case qc_mobility_enabled: return "mobility_enabled";
|
||||||
|
+ case qc_has_secure: return "has_secure";
|
||||||
|
+ case qc_secure: return "secure";
|
||||||
|
case qc_has_multiple_cpu_types: return "has_multiple_cpu_types";
|
||||||
|
case qc_cp_dispatch_limithard: return "cp_dispatch_limithard";
|
||||||
|
case qc_ifl_dispatch_limithard: return "ifl_dispatch_limithard";
|
||||||
|
@@ -1073,6 +1091,12 @@ struct qc_handle *qc_get_root_handle(struct qc_handle *hdl) {
|
||||||
|
return hdl ? hdl->root : NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
+struct qc_handle *qc_get_top_handle(struct qc_handle *hdl) {
|
||||||
|
+ for (; hdl->next != NULL; hdl = hdl->next);
|
||||||
|
+
|
||||||
|
+ return hdl;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
struct qc_handle *qc_get_prev_handle(struct qc_handle *hdl) {
|
||||||
|
struct qc_handle *prev_hdl = NULL;
|
||||||
|
|
||||||
|
diff --git a/query_capacity_int.h b/query_capacity_int.h
|
||||||
|
index 6e11c103..f2109916 100644
|
||||||
|
--- a/query_capacity_int.h
|
||||||
|
+++ b/query_capacity_int.h
|
||||||
|
@@ -21,7 +21,7 @@
|
||||||
|
#define STR_BUF_SIZE 257
|
||||||
|
|
||||||
|
#define ATTR_SRC_SYSINFO 'S'
|
||||||
|
-#define ATTR_SRC_OCF 'O'
|
||||||
|
+#define ATTR_SRC_SYSFS 'F'
|
||||||
|
#define ATTR_SRC_HYPFS 'H'
|
||||||
|
#define ATTR_SRC_STHYI 'V'
|
||||||
|
#define ATTR_SRC_POSTPROC 'P' // Note: Post-processed attributes can have multiple origins - would be
|
||||||
|
@@ -60,7 +60,7 @@ struct qc_data_src {
|
||||||
|
char *priv;
|
||||||
|
};
|
||||||
|
|
||||||
|
-extern struct qc_data_src sysinfo, ocf, hypfs, sthyi;
|
||||||
|
+extern struct qc_data_src sysinfo, sysfs, hypfs, sthyi;
|
||||||
|
|
||||||
|
/* Utility functions */
|
||||||
|
int qc_ebcdic_to_ascii(struct qc_handle *hdl, char *inbuf, size_t insz);
|
||||||
|
@@ -74,6 +74,7 @@ struct qc_handle *qc_get_cec_handle(struct qc_handle *hdl);
|
||||||
|
struct qc_handle *qc_get_lpar_handle(struct qc_handle *hdl);
|
||||||
|
struct qc_handle *qc_get_root_handle(struct qc_handle *hdl);
|
||||||
|
struct qc_handle *qc_get_prev_handle(struct qc_handle *hdl);
|
||||||
|
+struct qc_handle *qc_get_top_handle(struct qc_handle *hdl);
|
||||||
|
|
||||||
|
/* Debugging-related functions and variables */
|
||||||
|
extern long qc_dbg_level;
|
||||||
|
diff --git a/query_capacity_ocf.c b/query_capacity_ocf.c
|
||||||
|
deleted file mode 100644
|
||||||
|
index 3f5d2a12..00000000
|
||||||
|
--- a/query_capacity_ocf.c
|
||||||
|
+++ /dev/null
|
||||||
|
@@ -1,135 +0,0 @@
|
||||||
|
-/* Copyright IBM Corp. 2016 */
|
||||||
|
-
|
||||||
|
-#define _GNU_SOURCE
|
||||||
|
-#include <fcntl.h>
|
||||||
|
-#include <sys/types.h>
|
||||||
|
-#include <sys/stat.h>
|
||||||
|
-
|
||||||
|
-#include "query_capacity_data.h"
|
||||||
|
-
|
||||||
|
-
|
||||||
|
-#define FILE_CPC_NAME "/sys/firmware/ocf/cpc_name"
|
||||||
|
-
|
||||||
|
-
|
||||||
|
-static void qc_ocf_dump(struct qc_handle *hdl, char *data) {
|
||||||
|
- char *path = NULL;
|
||||||
|
- FILE *fp;
|
||||||
|
- int rc;
|
||||||
|
-
|
||||||
|
- qc_debug(hdl, "Dump ocf\n");
|
||||||
|
- qc_debug_indent_inc();
|
||||||
|
- if (!data)
|
||||||
|
- goto out;
|
||||||
|
- if (asprintf(&path, "%s/ocf", qc_dbg_dump_dir) == -1) {
|
||||||
|
- qc_debug(hdl, "Error: Mem alloc failure, cannot dump sysinfo\n");
|
||||||
|
- goto out_err;
|
||||||
|
- }
|
||||||
|
- if (mkdir(path, 0700) == -1) {
|
||||||
|
- qc_debug(hdl, "Error: Could not create directory for ocf dump: %s\n", strerror(errno));
|
||||||
|
- goto out_err;
|
||||||
|
- }
|
||||||
|
- free(path);
|
||||||
|
- if (asprintf(&path, "%s/ocf/cpc_name", qc_dbg_dump_dir) == -1) {
|
||||||
|
- qc_debug(hdl, "Error: Mem alloc failure, cannot dump sysinfo\n");
|
||||||
|
- goto out_err;
|
||||||
|
- }
|
||||||
|
- if ((fp = fopen(path, "w")) == NULL) {
|
||||||
|
- qc_debug(hdl, "Error: Failed to open %s to write ocf dump\n", path);
|
||||||
|
- goto out_err;
|
||||||
|
- }
|
||||||
|
- rc = fprintf(fp, "%s", data);
|
||||||
|
- fclose(fp);
|
||||||
|
- if (rc < 0) {
|
||||||
|
- qc_debug(hdl, "Error: Failed to write dump to '%s'\n", path);
|
||||||
|
- goto out_err;
|
||||||
|
- }
|
||||||
|
- goto out;
|
||||||
|
-
|
||||||
|
-out_err:
|
||||||
|
- qc_mark_dump_incomplete(hdl, "ocf");
|
||||||
|
-out:
|
||||||
|
- free(path);
|
||||||
|
- qc_debug_indent_dec();
|
||||||
|
-
|
||||||
|
- return;
|
||||||
|
-}
|
||||||
|
-
|
||||||
|
-static int qc_ocf_open(struct qc_handle *hdl, char **data) {
|
||||||
|
- char *fname = NULL;
|
||||||
|
- int rc = 0;
|
||||||
|
- size_t n;
|
||||||
|
- FILE *fp;
|
||||||
|
-
|
||||||
|
- qc_debug(hdl, "Retrieve ocf data\n");
|
||||||
|
- qc_debug_indent_inc();
|
||||||
|
- *data = NULL;
|
||||||
|
- if (qc_dbg_use_dump) {
|
||||||
|
- qc_debug(hdl, "Read ocf from dump\n");
|
||||||
|
- if (asprintf(&fname, "%s/ocf/cpc_name", qc_dbg_use_dump) == -1) {
|
||||||
|
- qc_debug(hdl, "Error: Mem alloc failed, cannot open dump\n");
|
||||||
|
- rc = -1;
|
||||||
|
- goto out;
|
||||||
|
- }
|
||||||
|
- } else
|
||||||
|
- qc_debug(hdl, "Read ocf from " FILE_CPC_NAME "\n");
|
||||||
|
-
|
||||||
|
- if (access(fname ? fname : FILE_CPC_NAME, F_OK)) {
|
||||||
|
- qc_debug(hdl, "No ocf data available\n");
|
||||||
|
- goto out;
|
||||||
|
- }
|
||||||
|
- fp = fopen(fname ? fname : FILE_CPC_NAME, "r");
|
||||||
|
- if (!fp) {
|
||||||
|
- qc_debug(hdl, "Error: Failed to open file '%s': %s\n",
|
||||||
|
- fname ? fname : FILE_CPC_NAME, strerror(errno));
|
||||||
|
- rc = -2;
|
||||||
|
- goto out;
|
||||||
|
- }
|
||||||
|
- rc = getline(data, &n, fp);
|
||||||
|
- fclose(fp);
|
||||||
|
- if (rc == -1) {
|
||||||
|
- qc_debug(hdl, "Error: Failed to read content: %s\n", strerror(errno));
|
||||||
|
- *data = NULL;
|
||||||
|
- rc = -3;
|
||||||
|
- goto out;
|
||||||
|
- }
|
||||||
|
- rc = 0;
|
||||||
|
- if (strcmp(*data, "\n") == 0 || **data == '\0') {
|
||||||
|
- qc_debug(hdl, FILE_CPC_NAME " contains no data, discarding\n");
|
||||||
|
- free(*data);
|
||||||
|
- *data = NULL;
|
||||||
|
- goto out;
|
||||||
|
- }
|
||||||
|
-
|
||||||
|
-out:
|
||||||
|
- free(fname);
|
||||||
|
- qc_debug(hdl, "Done reading ocf data\n");
|
||||||
|
- qc_debug_indent_dec();
|
||||||
|
-
|
||||||
|
- return rc;
|
||||||
|
-}
|
||||||
|
-
|
||||||
|
-static void qc_ocf_close(struct qc_handle *hdl, char *data) {
|
||||||
|
- free(data);
|
||||||
|
-}
|
||||||
|
-
|
||||||
|
-static int qc_ocf_process(struct qc_handle *hdl, char *data) {
|
||||||
|
- qc_debug(hdl, "Process ocf\n");
|
||||||
|
- qc_debug_indent_inc();
|
||||||
|
- if (!data) {
|
||||||
|
- qc_debug(hdl, "No ocf data, skipping\n");
|
||||||
|
- goto out;
|
||||||
|
- }
|
||||||
|
- if (qc_set_attr_string(hdl->root, qc_layer_name, data, ATTR_SRC_OCF))
|
||||||
|
- qc_debug(hdl, "Error: Failed to set CEC name to %s\n", data);
|
||||||
|
-out:
|
||||||
|
- qc_debug_indent_dec();
|
||||||
|
-
|
||||||
|
- return 0;
|
||||||
|
-}
|
||||||
|
-
|
||||||
|
-struct qc_data_src ocf = {qc_ocf_open,
|
||||||
|
- qc_ocf_process,
|
||||||
|
- qc_ocf_dump,
|
||||||
|
- qc_ocf_close,
|
||||||
|
- NULL,
|
||||||
|
- NULL};
|
||||||
|
diff --git a/query_capacity_sysfs.c b/query_capacity_sysfs.c
|
||||||
|
new file mode 100644
|
||||||
|
index 00000000..8594d950
|
||||||
|
--- /dev/null
|
||||||
|
+++ b/query_capacity_sysfs.c
|
||||||
|
@@ -0,0 +1,351 @@
|
||||||
|
+/* Copyright IBM Corp. 2020 */
|
||||||
|
+
|
||||||
|
+#define _GNU_SOURCE
|
||||||
|
+#include <fcntl.h>
|
||||||
|
+#include <sys/types.h>
|
||||||
|
+#include <sys/stat.h>
|
||||||
|
+
|
||||||
|
+#include "query_capacity_data.h"
|
||||||
|
+
|
||||||
|
+
|
||||||
|
+#define SYSFS_NA 0
|
||||||
|
+#define SYSFS_AVAILABLE 1
|
||||||
|
+
|
||||||
|
+#define FILE_CPC_NAME "/sys/firmware/ocf/cpc_name"
|
||||||
|
+#define FILE_SEC_IPL_HAS_SEC "/sys/firmware/ipl/has_secure"
|
||||||
|
+#define FILE_SEC_IPL_SEC "/sys/firmware/ipl/secure"
|
||||||
|
+
|
||||||
|
+static const char *sysfs_dirs[] = {"/sys",
|
||||||
|
+ "/sys/firmware",
|
||||||
|
+ "/sys/firmware/ocf",
|
||||||
|
+ "/sys/firmware/ipl",
|
||||||
|
+ NULL
|
||||||
|
+ };
|
||||||
|
+
|
||||||
|
+struct sysfs_priv {
|
||||||
|
+ int avail;
|
||||||
|
+ char *cpc_name; // NULL if n/a
|
||||||
|
+ int has_secure; // <0 if n/a
|
||||||
|
+ int secure; // <0 if n/a
|
||||||
|
+};
|
||||||
|
+
|
||||||
|
+static int qc_sysfs_mkpath(struct qc_handle *hdl, const char *a, const char *b, char **path) {
|
||||||
|
+ free(*path);
|
||||||
|
+ *path = NULL;
|
||||||
|
+
|
||||||
|
+ if (asprintf(path, "%s/%s", a, b) == -1) {
|
||||||
|
+ qc_debug(hdl, "Error: Mem alloc failed\n");
|
||||||
|
+ return -1;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ return 0;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+/** Create directory structure that we need for our dumps - if we don't have any content later on,
|
||||||
|
+ then there simply won't be any files in there */
|
||||||
|
+static int qc_sysfs_create_dump_dirs(struct qc_handle *hdl) {
|
||||||
|
+ char *path = NULL;
|
||||||
|
+ int rc = -1, i;
|
||||||
|
+
|
||||||
|
+ for (i = 0; sysfs_dirs[i]; ++i) {
|
||||||
|
+ if (qc_sysfs_mkpath(hdl, qc_dbg_dump_dir, sysfs_dirs[i], &path))
|
||||||
|
+ goto out;
|
||||||
|
+ if (mkdir(path, 0700) == -1) {
|
||||||
|
+ qc_debug(hdl, "Error: Could not create directory %s for sysfs dump: %s\n", path, strerror(errno));
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+ rc = 0;
|
||||||
|
+
|
||||||
|
+out:
|
||||||
|
+ free(path);
|
||||||
|
+
|
||||||
|
+ return rc;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+static FILE *qc_sysfs_open_dump_file(struct qc_handle *hdl, const char* file) {
|
||||||
|
+ char *path = NULL;
|
||||||
|
+ FILE *fp = NULL;
|
||||||
|
+
|
||||||
|
+ if (qc_sysfs_mkpath(hdl, qc_dbg_dump_dir, file, &path))
|
||||||
|
+ goto out;
|
||||||
|
+ if ((fp = fopen(path, "w")) == NULL) {
|
||||||
|
+ qc_debug(hdl, "Error: Failed to open '%s' to write sysfs dump\n", path);
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+out:
|
||||||
|
+ free(path);
|
||||||
|
+
|
||||||
|
+ return fp;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+static int qc_sysfs_dump_file_char(struct qc_handle *hdl, const char* file, const char *val) {
|
||||||
|
+ FILE *fp;
|
||||||
|
+ int rc;
|
||||||
|
+
|
||||||
|
+ if (!val) {
|
||||||
|
+ qc_debug(hdl, "No data for '%s', skipping\n", file);
|
||||||
|
+ return 0;
|
||||||
|
+ }
|
||||||
|
+ fp = qc_sysfs_open_dump_file(hdl, file);
|
||||||
|
+ if (!fp)
|
||||||
|
+ return -1;
|
||||||
|
+ rc = fprintf(fp, "%s", val);
|
||||||
|
+ fclose(fp);
|
||||||
|
+ if (rc < 0) {
|
||||||
|
+ qc_debug(hdl, "Error: Failed to write dump to '%s'\n", file);
|
||||||
|
+ return -1;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ return 0;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+static int qc_sysfs_dump_file_int(struct qc_handle *hdl, const char* file, int val) {
|
||||||
|
+ FILE *fp;
|
||||||
|
+ int rc;
|
||||||
|
+
|
||||||
|
+ if (val < 0) {
|
||||||
|
+ qc_debug(hdl, "No data for '%s', skipping\n", file);
|
||||||
|
+ return 0;
|
||||||
|
+ }
|
||||||
|
+ fp = qc_sysfs_open_dump_file(hdl, file);
|
||||||
|
+ if (!fp)
|
||||||
|
+ return -1;
|
||||||
|
+ rc = fprintf(fp, "%d", val);
|
||||||
|
+ fclose(fp);
|
||||||
|
+ if (rc < 0) {
|
||||||
|
+ qc_debug(hdl, "Error: Failed to write dump to '%s'\n", file);
|
||||||
|
+ return -1;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ return 0;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+static void qc_sysfs_dump(struct qc_handle *hdl, char *data) {
|
||||||
|
+ struct sysfs_priv *p;
|
||||||
|
+
|
||||||
|
+ qc_debug(hdl, "Dump sysfs\n");
|
||||||
|
+ qc_debug_indent_inc();
|
||||||
|
+ if (!data)
|
||||||
|
+ goto out;
|
||||||
|
+ p = (struct sysfs_priv *)data;
|
||||||
|
+ if (qc_sysfs_create_dump_dirs(hdl)) {
|
||||||
|
+ qc_debug(hdl, "Error: Failed to create directory structure\n");
|
||||||
|
+ goto out_err;
|
||||||
|
+ }
|
||||||
|
+ if (qc_sysfs_dump_file_char(hdl, FILE_CPC_NAME, p->cpc_name) ||
|
||||||
|
+ qc_sysfs_dump_file_int(hdl, FILE_SEC_IPL_HAS_SEC, p->has_secure) ||
|
||||||
|
+ qc_sysfs_dump_file_int(hdl, FILE_SEC_IPL_SEC, p->secure))
|
||||||
|
+ goto out_err;
|
||||||
|
+ qc_debug(hdl, "sysfs data dumped to '%s%s'\n", qc_dbg_dump_dir, *sysfs_dirs);
|
||||||
|
+ goto out;
|
||||||
|
+
|
||||||
|
+out_err:
|
||||||
|
+ qc_mark_dump_incomplete(hdl, "sysfs");
|
||||||
|
+out:
|
||||||
|
+ qc_debug_indent_dec();
|
||||||
|
+
|
||||||
|
+ return;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+static int qc_sysfs_is_old_dump_format(struct qc_handle *hdl) {
|
||||||
|
+ char *fname = NULL;
|
||||||
|
+ int rc = -1;
|
||||||
|
+
|
||||||
|
+ if (qc_sysfs_mkpath(hdl, qc_dbg_use_dump, "ocf", &fname))
|
||||||
|
+ goto out;
|
||||||
|
+ if (access(fname, F_OK) == 0) {
|
||||||
|
+ qc_debug(hdl, "Old ocf-based dump format\n");
|
||||||
|
+ rc = 1;
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+ rc = 0;
|
||||||
|
+out:
|
||||||
|
+ free(fname);
|
||||||
|
+
|
||||||
|
+ return rc;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+/** On success, returns 0 on success and filles data with respective file content.
|
||||||
|
+ Returns >0 if file is not available, and <0 on error. */
|
||||||
|
+static int qc_sysfs_get_file_content(struct qc_handle *hdl, char *file, char **content) {
|
||||||
|
+ FILE *fp = NULL;
|
||||||
|
+ size_t n;
|
||||||
|
+ int rc;
|
||||||
|
+
|
||||||
|
+ if (access(file, F_OK)) {
|
||||||
|
+ qc_debug(hdl, "File '%s' not available\n", file);
|
||||||
|
+ *content = NULL;
|
||||||
|
+ return 1;
|
||||||
|
+ }
|
||||||
|
+ fp = fopen(file, "r");
|
||||||
|
+ if (!fp) {
|
||||||
|
+ qc_debug(hdl, "Error: Failed to open file '%s': %s\n", file, strerror(errno));
|
||||||
|
+ return -1;
|
||||||
|
+ }
|
||||||
|
+ rc = getline(content, &n, fp);
|
||||||
|
+ fclose(fp);
|
||||||
|
+ if (rc == -1) {
|
||||||
|
+ qc_debug(hdl, "Error: Failed to read content of '%s': %s\n", file, strerror(errno));
|
||||||
|
+ *content = NULL;
|
||||||
|
+ return -2;
|
||||||
|
+ }
|
||||||
|
+ rc = 0;
|
||||||
|
+ if (strcmp(*content, "\n") == 0 || **content == '\0') {
|
||||||
|
+ qc_debug(hdl, "'%s' contains no data, discarding\n", file);
|
||||||
|
+ free(*content);
|
||||||
|
+ *content = NULL;
|
||||||
|
+ return 2;
|
||||||
|
+ }
|
||||||
|
+ qc_debug(hdl, "Read file %s\n", file);
|
||||||
|
+
|
||||||
|
+ return 0;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+/** Handle numeric attributes */
|
||||||
|
+static int qc_sysfs_num_attr(struct qc_handle *hdl, char *file, int *attr) {
|
||||||
|
+ char *content = NULL;
|
||||||
|
+ int rc;
|
||||||
|
+
|
||||||
|
+ rc = qc_sysfs_get_file_content(hdl, file, &content);
|
||||||
|
+ if (rc) {
|
||||||
|
+ *attr = -1;
|
||||||
|
+ if (rc > 0)
|
||||||
|
+ rc = 0;
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+ *attr = atoi(content);
|
||||||
|
+ if (*attr < 0) {
|
||||||
|
+ // we're not prepared to handle negative values (yet)
|
||||||
|
+ qc_debug(hdl, "Negative content for '%s': %s\n", file, content);
|
||||||
|
+ rc = -1;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+out:
|
||||||
|
+ free(content);
|
||||||
|
+
|
||||||
|
+ return rc;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+static struct sysfs_priv *qc_sysfs_init_data(struct qc_handle *hdl, char **data) {
|
||||||
|
+ struct sysfs_priv *p;
|
||||||
|
+
|
||||||
|
+ if ((*data = malloc(sizeof(struct sysfs_priv))) == NULL) {
|
||||||
|
+ qc_debug(hdl, "Error: Failed to allocate private data for sysfs\n");
|
||||||
|
+ return NULL;
|
||||||
|
+ }
|
||||||
|
+ p = (struct sysfs_priv *)*data;
|
||||||
|
+ memset(p, 0, sizeof(struct sysfs_priv));
|
||||||
|
+ p->avail = SYSFS_NA;
|
||||||
|
+ p->has_secure = -1;
|
||||||
|
+ p->secure = -1;
|
||||||
|
+
|
||||||
|
+ return p;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+static int qc_sysfs_open(struct qc_handle *hdl, char **data) {
|
||||||
|
+ struct sysfs_priv *p;
|
||||||
|
+ char *path = NULL;
|
||||||
|
+ int rc = 0, lrc;
|
||||||
|
+
|
||||||
|
+ qc_debug(hdl, "Retrieve sysfs data\n");
|
||||||
|
+ qc_debug_indent_inc();
|
||||||
|
+ p = qc_sysfs_init_data(hdl, data);
|
||||||
|
+ if (!p) {
|
||||||
|
+ rc = -1;
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+ if (qc_dbg_use_dump) {
|
||||||
|
+ qc_debug(hdl, "Read sysfs from dump\n");
|
||||||
|
+ if (qc_sysfs_is_old_dump_format(hdl)) {
|
||||||
|
+ // Note: previously, we had a directory called 'ocf' where only one piece of data was
|
||||||
|
+ // residing. But we have switched over to a more general sys directory instead.
|
||||||
|
+ qc_debug(hdl, "Old, ocf-based format\n");
|
||||||
|
+ if (qc_sysfs_mkpath(hdl, qc_dbg_use_dump, "ocf/cpc_name", &path)) {
|
||||||
|
+ rc = -1;
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+ lrc = qc_sysfs_get_file_content(hdl, path, &p->cpc_name);
|
||||||
|
+ if (lrc != 0) {
|
||||||
|
+ rc = (lrc < 0 ? -1 : 0);
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+ p->avail = SYSFS_AVAILABLE;
|
||||||
|
+ } else {
|
||||||
|
+ qc_debug(hdl, "New, sysfs-based format\n");
|
||||||
|
+ if (qc_sysfs_mkpath(hdl, qc_dbg_use_dump, FILE_CPC_NAME, &path) ||
|
||||||
|
+ qc_sysfs_get_file_content(hdl, path, &p->cpc_name) < 0 ||
|
||||||
|
+ qc_sysfs_mkpath(hdl, qc_dbg_use_dump, FILE_SEC_IPL_HAS_SEC, &path) ||
|
||||||
|
+ qc_sysfs_num_attr(hdl, path, &p->has_secure) ||
|
||||||
|
+ qc_sysfs_mkpath(hdl, qc_dbg_use_dump, FILE_SEC_IPL_SEC, &path) ||
|
||||||
|
+ qc_sysfs_num_attr(hdl, path, &p->secure))
|
||||||
|
+ rc = -1;
|
||||||
|
+ else
|
||||||
|
+ p->avail = SYSFS_AVAILABLE;
|
||||||
|
+ }
|
||||||
|
+ } else {
|
||||||
|
+ qc_debug(hdl, "Read sysfs from system\n");
|
||||||
|
+ if (qc_sysfs_get_file_content(hdl, FILE_CPC_NAME, &p->cpc_name) < 0 ||
|
||||||
|
+ qc_sysfs_num_attr(hdl, FILE_SEC_IPL_HAS_SEC, &p->has_secure) ||
|
||||||
|
+ qc_sysfs_num_attr(hdl, FILE_SEC_IPL_SEC, &p->secure))
|
||||||
|
+ rc = -1;
|
||||||
|
+ else
|
||||||
|
+ p->avail = SYSFS_AVAILABLE;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+out:
|
||||||
|
+ qc_debug(hdl, "Done reading sysfs data\n");
|
||||||
|
+ qc_debug_indent_dec();
|
||||||
|
+ free(path);
|
||||||
|
+
|
||||||
|
+ return rc;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+static void qc_sysfs_close(struct qc_handle *hdl, char *data) {
|
||||||
|
+ struct sysfs_priv *p = (struct sysfs_priv *)data;
|
||||||
|
+
|
||||||
|
+ if (p) {
|
||||||
|
+ free(p->cpc_name);
|
||||||
|
+ free(data);
|
||||||
|
+ }
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+static int qc_sysfs_process(struct qc_handle *hdl, char *data) {
|
||||||
|
+ struct sysfs_priv *p = (struct sysfs_priv *)data;
|
||||||
|
+ int rc = 0;
|
||||||
|
+
|
||||||
|
+ qc_debug(hdl, "Process sysfs\n");
|
||||||
|
+ qc_debug_indent_inc();
|
||||||
|
+ if (!p) {
|
||||||
|
+ qc_debug(hdl, "No sysfs data, skipping\n");
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ // Set CEC layer attribute
|
||||||
|
+ if ((p->cpc_name && qc_set_attr_string(hdl->root, qc_layer_name, p->cpc_name, ATTR_SRC_SYSFS))) {
|
||||||
|
+ rc = -1;
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ // Set top layer attributes.
|
||||||
|
+ // Note: This implies that all top layers must feature these attributes!
|
||||||
|
+ hdl = qc_get_top_handle(hdl);
|
||||||
|
+ if ((p->has_secure >= 0 && qc_set_attr_int(hdl, qc_has_secure, p->has_secure, ATTR_SRC_SYSFS)) ||
|
||||||
|
+ (p->secure >= 0 && qc_set_attr_int(hdl, qc_secure, p->secure, ATTR_SRC_SYSFS))) {
|
||||||
|
+ rc = -1;
|
||||||
|
+ goto out;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+out:
|
||||||
|
+ qc_debug_indent_dec();
|
||||||
|
+
|
||||||
|
+ return rc;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+struct qc_data_src sysfs = {qc_sysfs_open,
|
||||||
|
+ qc_sysfs_process,
|
||||||
|
+ qc_sysfs_dump,
|
||||||
|
+ qc_sysfs_close,
|
||||||
|
+ NULL,
|
||||||
|
+ NULL};
|
@ -1,3 +1,10 @@
|
|||||||
|
-------------------------------------------------------------------
|
||||||
|
Tue Mar 3 19:38:58 UTC 2020 - Mark Post <mpost@suse.com>
|
||||||
|
|
||||||
|
- Added qclib-2.0.1-add-support-for-secure-boot-related-attributes.patch
|
||||||
|
Add an attribute in qclib so that we can programmatically detect
|
||||||
|
if secure boot is available and active in the current LPAR. (bsc#1165018).
|
||||||
|
|
||||||
-------------------------------------------------------------------
|
-------------------------------------------------------------------
|
||||||
Wed Jan 22 16:01:05 UTC 2020 - Martin Pluskal <mpluskal@suse.com>
|
Wed Jan 22 16:01:05 UTC 2020 - Martin Pluskal <mpluskal@suse.com>
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#
|
#
|
||||||
# spec file for package qclib
|
# spec file for package qclib
|
||||||
#
|
#
|
||||||
# Copyright (c) 2020 SUSE LLC
|
# Copyright (c) 2017-2020 SUSE LLC
|
||||||
#
|
#
|
||||||
# All modifications and additions to the file contributed by third parties
|
# All modifications and additions to the file contributed by third parties
|
||||||
# remain the property of their copyright owners, unless otherwise agreed
|
# remain the property of their copyright owners, unless otherwise agreed
|
||||||
@ -26,6 +26,7 @@ URL: http://www.ibm.com/developerworks/linux/linux390/qclib.html
|
|||||||
Source: %{name}-%{version}.tgz
|
Source: %{name}-%{version}.tgz
|
||||||
Source1: %{name}-rpmlintrc
|
Source1: %{name}-rpmlintrc
|
||||||
Patch1: qclib.makefile.libdir.patch
|
Patch1: qclib.makefile.libdir.patch
|
||||||
|
Patch2: qclib-2.0.1-add-support-for-secure-boot-related-attributes.patch
|
||||||
BuildRequires: doxygen
|
BuildRequires: doxygen
|
||||||
BuildRequires: gcc-c++
|
BuildRequires: gcc-c++
|
||||||
ExclusiveArch: s390 s390x
|
ExclusiveArch: s390 s390x
|
||||||
|
Loading…
Reference in New Issue
Block a user