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

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