Blame SOURCES/0067-vdo-use-defines-also-for-configuration-defines.patch

f358f0
From b16082b05639d4321cbf699d3309fe24a8bc71fa Mon Sep 17 00:00:00 2001
f358f0
From: Zdenek Kabelac <zkabelac@redhat.com>
f358f0
Date: Fri, 24 Jun 2022 15:54:08 +0200
f358f0
Subject: [PATCH 2/3] vdo: use defines also for configuration defines
f358f0
f358f0
Keep single source for most of values printed in lvm.conf
f358f0
(still needs some conversion)
f358f0
f358f0
Correct max for logical threads to 60
f358f0
(we may refuse some older configuration which might eventually
f358f0
user higher numbers - but so far let's assume no user have ever set this
f358f0
as it's been non-trivial and if would complicate code unnecessarily.)
f358f0
f358f0
Accept maximum of 4PiB for virtual size of VDO LV
f358f0
(lvm2 will drop 'header borders to 0 for this case').
f358f0
f358f0
(cherry picked from commit b5c8e591ed9ee30b67e79d60705d3c0bb8509a2a)
f358f0
---
f358f0
 conf/example.conf.in           |  9 +++---
f358f0
 device_mapper/vdo/vdo_limits.h | 55 ++++++++++++++++++----------------
f358f0
 device_mapper/vdo/vdo_target.c | 11 +++----
f358f0
 lib/config/config_settings.h   | 32 +++++++++++++-------
f358f0
 4 files changed, 60 insertions(+), 47 deletions(-)
f358f0
f358f0
diff --git a/conf/example.conf.in b/conf/example.conf.in
f358f0
index a78ed7333..897622b9d 100644
f358f0
--- a/conf/example.conf.in
f358f0
+++ b/conf/example.conf.in
f358f0
@@ -625,13 +625,12 @@ allocation {
f358f0
 	# Enables or disables whether VDO volume should tag its latency-critical
f358f0
 	# writes with the REQ_SYNC flag. Some device mapper targets such as dm-raid5
f358f0
 	# process writes with this flag at a higher priority.
f358f0
-	# Default is enabled.
f358f0
 	# This configuration option has an automatic default value.
f358f0
 	# vdo_use_metadata_hints = 1
f358f0
 
f358f0
 	# Configuration option allocation/vdo_minimum_io_size.
f358f0
 	# The minimum IO size for VDO volume to accept, in bytes.
f358f0
-	# Valid values are 512 or 4096. The recommended and default value is 4096.
f358f0
+	# Valid values are 512 or 4096. The recommended value is 4096.
f358f0
 	# This configuration option has an automatic default value.
f358f0
 	# vdo_minimum_io_size = 4096
f358f0
 
f358f0
@@ -684,7 +683,7 @@ allocation {
f358f0
 	# Configuration option allocation/vdo_bio_threads.
f358f0
 	# Specifies the number of threads to use for submitting I/O
f358f0
 	# operations to the storage device of VDO volume.
f358f0
-	# The value must be in range [1..100]
f358f0
+	# The value must be in range [1..100].
f358f0
 	# Each additional thread after the first will use an additional 18MiB of RAM,
f358f0
 	# plus 1.12 MiB of RAM per megabyte of configured read cache size.
f358f0
 	# This configuration option has an automatic default value.
f358f0
@@ -698,7 +697,7 @@ allocation {
f358f0
 
f358f0
 	# Configuration option allocation/vdo_cpu_threads.
f358f0
 	# Specifies the number of threads to use for CPU-intensive work such as
f358f0
-	# hashing or compression for VDO volume. The value must be in range [1..100]
f358f0
+	# hashing or compression for VDO volume. The value must be in range [1..100].
f358f0
 	# This configuration option has an automatic default value.
f358f0
 	# vdo_cpu_threads = 2
f358f0
 
f358f0
@@ -716,7 +715,7 @@ allocation {
f358f0
 	# processing based on the hash value computed from the block data.
f358f0
 	# A logical thread count of 9 or more will require explicitly specifying
f358f0
 	# a sufficiently large block map cache size, as well.
f358f0
-	# The value must be in range [0..100].
f358f0
+	# The value must be in range [0..60].
f358f0
 	# vdo_hash_zone_threads, vdo_logical_threads and vdo_physical_threads must be
f358f0
 	# either all zero or all non-zero.
f358f0
 	# This configuration option has an automatic default value.
f358f0
diff --git a/device_mapper/vdo/vdo_limits.h b/device_mapper/vdo/vdo_limits.h
f358f0
index e145100b1..db365ace2 100644
f358f0
--- a/device_mapper/vdo/vdo_limits.h
f358f0
+++ b/device_mapper/vdo/vdo_limits.h
f358f0
@@ -1,5 +1,5 @@
f358f0
 /*
f358f0
- * Copyright (C) 2018 Red Hat, Inc. All rights reserved.
f358f0
+ * Copyright (C) 2018-2022 Red Hat, Inc. All rights reserved.
f358f0
  *
f358f0
  * This file is part of the device-mapper userspace tools.
f358f0
  *
f358f0
@@ -15,49 +15,52 @@
f358f0
 #ifndef DEVICE_MAPPER_VDO_LIMITS_H
f358f0
 #define DEVICE_MAPPER_VDO_LIMITS_H
f358f0
 
f358f0
+#ifndef SECTOR_SHIFT
f358f0
+#define SECTOR_SHIFT 9L
f358f0
+#endif
f358f0
+
f358f0
 #define DM_VDO_BLOCK_SIZE			UINT64_C(8)		// 4KiB in sectors
f358f0
+#define DM_VDO_BLOCK_SIZE_KB			(DM_VDO_BLOCK_SIZE << SECTOR_SHIFT)
f358f0
 
f358f0
 #define DM_VDO_BLOCK_MAP_CACHE_SIZE_MINIMUM_MB	(128)			// 128MiB
f358f0
 #define DM_VDO_BLOCK_MAP_CACHE_SIZE_MAXIMUM_MB	(16 * 1024 * 1024 - 1)	// 16TiB - 1
f358f0
 #define DM_VDO_BLOCK_MAP_CACHE_SIZE_MINIMUM_PER_LOGICAL_THREAD  (4096 * DM_VDO_BLOCK_SIZE_KB)
f358f0
 
f358f0
-#define DM_VDO_BLOCK_MAP_ERA_LENGTH_MINIMUM	(1)
f358f0
-#define DM_VDO_BLOCK_MAP_ERA_LENGTH_MAXIMUM	(16380)
f358f0
+#define DM_VDO_BLOCK_MAP_ERA_LENGTH_MINIMUM	1
f358f0
+#define DM_VDO_BLOCK_MAP_ERA_LENGTH_MAXIMUM	16380
f358f0
 
f358f0
-#define DM_VDO_INDEX_MEMORY_SIZE_MINIMUM_MB	(256)			// 0.25 GiB
f358f0
+#define DM_VDO_INDEX_MEMORY_SIZE_MINIMUM_MB	256			// 0.25 GiB
f358f0
 #define DM_VDO_INDEX_MEMORY_SIZE_MAXIMUM_MB	(1024 * 1024 * 1024)	// 1TiB
f358f0
 
f358f0
-//#define DM_VDO_READ_CACHE_SIZE_MINIMUM_MB	(0)
f358f0
-#define DM_VDO_READ_CACHE_SIZE_MAXIMUM_MB	(16 * 1024 * 1024 - 1)	// 16TiB - 1
f358f0
-
f358f0
-#define DM_VDO_SLAB_SIZE_MINIMUM_MB		(128)			// 128MiB
f358f0
+#define DM_VDO_SLAB_SIZE_MINIMUM_MB		128			// 128MiB
f358f0
 #define DM_VDO_SLAB_SIZE_MAXIMUM_MB		(32 * 1024)		// 32GiB
f358f0
+#define DM_VDO_SLABS_MAXIMUM			8192
f358f0
 
f358f0
-//#define DM_VDO_LOGICAL_SIZE_MINIMUM_MB	(0)
f358f0
-#define DM_VDO_LOGICAL_SIZE_MAXIMUM_MB	(UINT64_C(4) * 1024 * 1024 * 1024) // 4PiB
f358f0
+#define DM_VDO_LOGICAL_SIZE_MAXIMUM	(UINT64_C(4) * 1024 * 1024 * 1024 * 1024 * 1024 >> SECTOR_SHIFT) // 4PiB
f358f0
+#define DM_VDO_PHYSICAL_SIZE_MAXIMUM	(UINT64_C(64) * DM_VDO_BLOCK_SIZE_KB * 1024 * 1024 * 1024 >> SECTOR_SHIFT) // 256TiB
f358f0
 
f358f0
-//#define DM_VDO_ACK_THREADS_MINIMUM		(0)
f358f0
-#define DM_VDO_ACK_THREADS_MAXIMUM		(100)
f358f0
+#define DM_VDO_ACK_THREADS_MINIMUM		0
f358f0
+#define DM_VDO_ACK_THREADS_MAXIMUM		100
f358f0
 
f358f0
-#define DM_VDO_BIO_THREADS_MINIMUM		(1)
f358f0
-#define DM_VDO_BIO_THREADS_MAXIMUM		(100)
f358f0
+#define DM_VDO_BIO_THREADS_MINIMUM		1
f358f0
+#define DM_VDO_BIO_THREADS_MAXIMUM		100
f358f0
 
f358f0
-#define DM_VDO_BIO_ROTATION_MINIMUM		(1)
f358f0
-#define DM_VDO_BIO_ROTATION_MAXIMUM		(1024)
f358f0
+#define DM_VDO_BIO_ROTATION_MINIMUM		1
f358f0
+#define DM_VDO_BIO_ROTATION_MAXIMUM		1024
f358f0
 
f358f0
-#define DM_VDO_CPU_THREADS_MINIMUM		(1)
f358f0
-#define DM_VDO_CPU_THREADS_MAXIMUM		(100)
f358f0
+#define DM_VDO_CPU_THREADS_MINIMUM		1
f358f0
+#define DM_VDO_CPU_THREADS_MAXIMUM		100
f358f0
 
f358f0
-//#define DM_VDO_HASH_ZONE_THREADS_MINIMUM	(0)
f358f0
-#define DM_VDO_HASH_ZONE_THREADS_MAXIMUM	(100)
f358f0
+#define DM_VDO_HASH_ZONE_THREADS_MINIMUM	0
f358f0
+#define DM_VDO_HASH_ZONE_THREADS_MAXIMUM	100
f358f0
 
f358f0
-//#define DM_VDO_LOGICAL_THREADS_MINIMUM	(0)
f358f0
-#define DM_VDO_LOGICAL_THREADS_MAXIMUM		(100)
f358f0
+#define DM_VDO_LOGICAL_THREADS_MINIMUM		0
f358f0
+#define DM_VDO_LOGICAL_THREADS_MAXIMUM		60
f358f0
 
f358f0
-//#define DM_VDO_PHYSICAL_THREADS_MINIMUM	(0)
f358f0
-#define DM_VDO_PHYSICAL_THREADS_MAXIMUM		(16)
f358f0
+#define DM_VDO_PHYSICAL_THREADS_MINIMUM		0
f358f0
+#define DM_VDO_PHYSICAL_THREADS_MAXIMUM		16
f358f0
 
f358f0
-#define DM_VDO_MAX_DISCARD_MINIMUM		(1)
f358f0
-#define DM_VDO_MAX_DISCARD_MAXIMUM		(UINT32_MAX / 4096)
f358f0
+#define DM_VDO_MAX_DISCARD_MINIMUM		1
f358f0
+#define DM_VDO_MAX_DISCARD_MAXIMUM		(UINT32_MAX / (uint32_t)(DM_VDO_BLOCK_SIZE_KB))
f358f0
 
f358f0
 #endif // DEVICE_MAPPER_VDO_LIMITS_H
f358f0
diff --git a/device_mapper/vdo/vdo_target.c b/device_mapper/vdo/vdo_target.c
f358f0
index 0e5abd162..3ebe0592e 100644
f358f0
--- a/device_mapper/vdo/vdo_target.c
f358f0
+++ b/device_mapper/vdo/vdo_target.c
f358f0
@@ -18,14 +18,15 @@
f358f0
 #include "vdo_limits.h"
f358f0
 #include "target.h"
f358f0
 
f358f0
+/* validate vdo target parameters and  'vdo_size' in sectors */
f358f0
 bool dm_vdo_validate_target_params(const struct dm_vdo_target_params *vtp,
f358f0
 				   uint64_t vdo_size)
f358f0
 {
f358f0
 	bool valid = true;
f358f0
 
f358f0
 	/* 512 or 4096 bytes only ATM */
f358f0
-	if ((vtp->minimum_io_size != 1) &&
f358f0
-	    (vtp->minimum_io_size != 8)) {
f358f0
+	if ((vtp->minimum_io_size != (512 >> SECTOR_SHIFT)) &&
f358f0
+	    (vtp->minimum_io_size != (4096 >> SECTOR_SHIFT))) {
f358f0
 		log_error("VDO minimum io size %u is unsupported.",
f358f0
 			  vtp->minimum_io_size);
f358f0
 		valid = false;
f358f0
@@ -127,10 +128,10 @@ bool dm_vdo_validate_target_params(const struct dm_vdo_target_params *vtp,
f358f0
 		valid = false;
f358f0
 	}
f358f0
 
f358f0
-	if (vdo_size >= (DM_VDO_LOGICAL_SIZE_MAXIMUM_MB * UINT64_C(1024 * 2))) {
f358f0
+	if (vdo_size > DM_VDO_LOGICAL_SIZE_MAXIMUM) {
f358f0
 		log_error("VDO logical size is by " FMTu64 "KiB bigger then limit " FMTu64 "TiB.",
f358f0
-			  (vdo_size - (DM_VDO_LOGICAL_SIZE_MAXIMUM_MB * UINT64_C(1024 * 2))) / 2,
f358f0
-			  DM_VDO_LOGICAL_SIZE_MAXIMUM_MB / UINT64_C(1024) / UINT64_C(1024));
f358f0
+			  (vdo_size - DM_VDO_LOGICAL_SIZE_MAXIMUM) / 2,
f358f0
+			  DM_VDO_LOGICAL_SIZE_MAXIMUM / (UINT64_C(1024) * 1024 * 1024 * 1024 >> SECTOR_SHIFT));
f358f0
 		valid = false;
f358f0
 	}
f358f0
 
f358f0
diff --git a/lib/config/config_settings.h b/lib/config/config_settings.h
f358f0
index d280e7adb..2c91e8bb6 100644
f358f0
--- a/lib/config/config_settings.h
f358f0
+++ b/lib/config/config_settings.h
f358f0
@@ -118,6 +118,7 @@
f358f0
  * the previous default value was set (uncommented) in lvm.conf.
f358f0
  */
f358f0
 #include "lib/config/defaults.h"
f358f0
+#include "device_mapper/vdo/vdo_limits.h"
f358f0
 
f358f0
 cfg_section(root_CFG_SECTION, "(root)", root_CFG_SECTION, 0, vsn(0, 0, 0), 0, NULL, NULL)
f358f0
 
f358f0
@@ -708,12 +709,11 @@ cfg(allocation_vdo_use_deduplication_CFG, "vdo_use_deduplication", allocation_CF
f358f0
 cfg(allocation_vdo_use_metadata_hints_CFG, "vdo_use_metadata_hints", allocation_CFG_SECTION, CFG_PROFILABLE | CFG_PROFILABLE_METADATA | CFG_DEFAULT_COMMENTED, CFG_TYPE_INT, DEFAULT_VDO_USE_METADATA_HINTS, VDO_1ST_VSN, NULL, 0, NULL,
f358f0
 	"Enables or disables whether VDO volume should tag its latency-critical\n"
f358f0
 	"writes with the REQ_SYNC flag. Some device mapper targets such as dm-raid5\n"
f358f0
-	"process writes with this flag at a higher priority.\n"
f358f0
-	"Default is enabled.\n")
f358f0
+	"process writes with this flag at a higher priority.\n")
f358f0
 
f358f0
 cfg(allocation_vdo_minimum_io_size_CFG, "vdo_minimum_io_size", allocation_CFG_SECTION, CFG_PROFILABLE | CFG_PROFILABLE_METADATA | CFG_DEFAULT_COMMENTED, CFG_TYPE_INT, DEFAULT_VDO_MINIMUM_IO_SIZE, VDO_1ST_VSN, NULL, 0, NULL,
f358f0
 	"The minimum IO size for VDO volume to accept, in bytes.\n"
f358f0
-	"Valid values are 512 or 4096. The recommended and default value is 4096.\n")
f358f0
+	"Valid values are 512 or 4096. The recommended value is 4096.\n")
f358f0
 
f358f0
 cfg(allocation_vdo_block_map_cache_size_mb_CFG, "vdo_block_map_cache_size_mb", allocation_CFG_SECTION, CFG_PROFILABLE | CFG_PROFILABLE_METADATA | CFG_DEFAULT_COMMENTED, CFG_TYPE_INT, DEFAULT_VDO_BLOCK_MAP_CACHE_SIZE_MB, VDO_1ST_VSN, NULL, 0, NULL,
f358f0
 	"Specifies the amount of memory in MiB allocated for caching block map\n"
f358f0
@@ -726,7 +726,8 @@ cfg(allocation_vdo_block_map_era_length_CFG, "vdo_block_map_period", allocation_
f358f0
 	"The speed with which the block map cache writes out modified block map pages.\n"
f358f0
 	"A smaller era length is likely to reduce the amount time spent rebuilding,\n"
f358f0
 	"at the cost of increased block map writes during normal operation.\n"
f358f0
-	"The maximum and recommended value is 16380; the minimum value is 1.\n")
f358f0
+	"The maximum and recommended value is " DM_TO_STRING(DM_VDO_BLOCK_MAP_ERA_LENGTH_MAXIMUM)
f358f0
+	"; the minimum value is " DM_TO_STRING(DM_VDO_BLOCK_MAP_ERA_LENGTH_MINIMUM) ".\n")
f358f0
 
f358f0
 cfg(allocation_vdo_check_point_frequency_CFG, "vdo_check_point_frequency", allocation_CFG_SECTION, CFG_PROFILABLE | CFG_PROFILABLE_METADATA | CFG_DEFAULT_COMMENTED, CFG_TYPE_INT, DEFAULT_VDO_CHECK_POINT_FREQUENCY, VDO_1ST_VSN, NULL, 0, NULL,
f358f0
 	"The default check point frequency for VDO volume.\n")
f358f0
@@ -748,27 +749,34 @@ cfg(allocation_vdo_slab_size_mb_CFG, "vdo_slab_size_mb", allocation_CFG_SECTION,
f358f0
 cfg(allocation_vdo_ack_threads_CFG, "vdo_ack_threads", allocation_CFG_SECTION, CFG_PROFILABLE | CFG_PROFILABLE_METADATA | CFG_DEFAULT_COMMENTED, CFG_TYPE_INT, DEFAULT_VDO_ACK_THREADS, VDO_1ST_VSN, NULL, 0, NULL,
f358f0
 	"Specifies the number of threads to use for acknowledging\n"
f358f0
 	"completion of requested VDO I/O operations.\n"
f358f0
-	"The value must be at in range [0..100].\n")
f358f0
+	"The value must be at in range [" DM_TO_STRING(DM_VDO_ACK_THREADS_MINIMUM) ".."
f358f0
+	DM_TO_STRING(DM_VDO_ACK_THREADS_MAXIMUM) "].\n")
f358f0
 
f358f0
 cfg(allocation_vdo_bio_threads_CFG, "vdo_bio_threads", allocation_CFG_SECTION, CFG_PROFILABLE | CFG_PROFILABLE_METADATA | CFG_DEFAULT_COMMENTED, CFG_TYPE_INT, DEFAULT_VDO_BIO_THREADS, VDO_1ST_VSN, NULL, 0, NULL,
f358f0
 	"Specifies the number of threads to use for submitting I/O\n"
f358f0
 	"operations to the storage device of VDO volume.\n"
f358f0
-	"The value must be in range [1..100]\n"
f358f0
+	"The value must be in range [" DM_TO_STRING(DM_VDO_BIO_THREADS_MINIMUM) ".."
f358f0
+	DM_TO_STRING(DM_VDO_BIO_THREADS_MAXIMUM) "].\n"
f358f0
 	"Each additional thread after the first will use an additional 18MiB of RAM,\n"
f358f0
 	"plus 1.12 MiB of RAM per megabyte of configured read cache size.\n")
f358f0
 
f358f0
 cfg(allocation_vdo_bio_rotation_CFG, "vdo_bio_rotation", allocation_CFG_SECTION, CFG_PROFILABLE | CFG_PROFILABLE_METADATA | CFG_DEFAULT_COMMENTED, CFG_TYPE_INT, DEFAULT_VDO_BIO_ROTATION, VDO_1ST_VSN, NULL, 0, NULL,
f358f0
 	"Specifies the number of I/O operations to enqueue for each bio-submission\n"
f358f0
-	"thread before directing work to the next. The value must be in range [1..1024].\n")
f358f0
+	"thread before directing work to the next. The value must be in range ["
f358f0
+	DM_TO_STRING(DM_VDO_BIO_ROTATION_MINIMUM) ".."
f358f0
+	DM_TO_STRING(DM_VDO_BIO_ROTATION_MAXIMUM) "].\n")
f358f0
 
f358f0
 cfg(allocation_vdo_cpu_threads_CFG, "vdo_cpu_threads", allocation_CFG_SECTION, CFG_PROFILABLE | CFG_PROFILABLE_METADATA | CFG_DEFAULT_COMMENTED, CFG_TYPE_INT, DEFAULT_VDO_CPU_THREADS, VDO_1ST_VSN, NULL, 0, NULL,
f358f0
 	"Specifies the number of threads to use for CPU-intensive work such as\n"
f358f0
-	"hashing or compression for VDO volume. The value must be in range [1..100]\n")
f358f0
+	"hashing or compression for VDO volume. The value must be in range ["
f358f0
+	DM_TO_STRING(DM_VDO_CPU_THREADS_MINIMUM) ".."
f358f0
+	DM_TO_STRING(DM_VDO_CPU_THREADS_MAXIMUM) "].\n")
f358f0
 
f358f0
 cfg(allocation_vdo_hash_zone_threads_CFG, "vdo_hash_zone_threads", allocation_CFG_SECTION, CFG_PROFILABLE | CFG_PROFILABLE_METADATA | CFG_DEFAULT_COMMENTED, CFG_TYPE_INT, DEFAULT_VDO_HASH_ZONE_THREADS, VDO_1ST_VSN, NULL, 0, NULL,
f358f0
 	"Specifies the number of threads across which to subdivide parts of the VDO\n"
f358f0
 	"processing based on the hash value computed from the block data.\n"
f358f0
-	"The value must be at in range [0..100].\n"
f358f0
+	"The value must be at in range [" DM_TO_STRING(DM_VDO_HASH_ZONE_THREADS_MINIMUM) ".."
f358f0
+	DM_TO_STRING(DM_VDO_HASH_ZONE_THREADS_MAXIMUM) "].\n"
f358f0
 	"vdo_hash_zone_threads, vdo_logical_threads and vdo_physical_threads must be\n"
f358f0
 	"either all zero or all non-zero.\n")
f358f0
 
f358f0
@@ -777,7 +785,8 @@ cfg(allocation_vdo_logical_threads_CFG, "vdo_logical_threads", allocation_CFG_SE
f358f0
 	"processing based on the hash value computed from the block data.\n"
f358f0
 	"A logical thread count of 9 or more will require explicitly specifying\n"
f358f0
 	"a sufficiently large block map cache size, as well.\n"
f358f0
-	"The value must be in range [0..100].\n"
f358f0
+	"The value must be in range [" DM_TO_STRING(DM_VDO_LOGICAL_THREADS_MINIMUM) ".."
f358f0
+	DM_TO_STRING(DM_VDO_LOGICAL_THREADS_MAXIMUM) "].\n"
f358f0
 	"vdo_hash_zone_threads, vdo_logical_threads and vdo_physical_threads must be\n"
f358f0
 	"either all zero or all non-zero.\n")
f358f0
 
f358f0
@@ -785,7 +794,8 @@ cfg(allocation_vdo_physical_threads_CFG, "vdo_physical_threads", allocation_CFG_
f358f0
 	"Specifies the number of threads across which to subdivide parts of the VDO\n"
f358f0
 	"processing based on physical block addresses.\n"
f358f0
 	"Each additional thread after the first will use an additional 10MiB of RAM.\n"
f358f0
-	"The value must be in range [0..16].\n"
f358f0
+	"The value must be in range [" DM_TO_STRING(DM_VDO_PHYSICAL_THREADS_MINIMUM) ".."
f358f0
+	DM_TO_STRING(DM_VDO_PHYSICAL_THREADS_MAXIMUM) "].\n"
f358f0
 	"vdo_hash_zone_threads, vdo_logical_threads and vdo_physical_threads must be\n"
f358f0
 	"either all zero or all non-zero.\n")
f358f0
 
f358f0
-- 
f358f0
2.38.1
f358f0