diff --git a/59-dasd.rules-wait_for.patch b/59-dasd.rules-wait_for.patch
index 03d86e9..98a032e 100644
--- a/59-dasd.rules-wait_for.patch
+++ b/59-dasd.rules-wait_for.patch
@@ -8,7 +8,7 @@
-# on device add set request queue scheduler to deadline
-SUBSYSTEM!="block", GOTO="sched_end"
-
--ACTION!="add", GOTO="sched_end"
--KERNEL=="dasd*[!0-9]", WAIT_FOR="queue/scheduler", ATTR{queue/scheduler}="deadline"
+-ACTION!="change", GOTO="sched_end"
+-KERNEL=="dasd*[!0-9]", TEST=="queue/scheduler", ATTR{queue/scheduler}="deadline"
-
-LABEL="sched_end"
diff --git a/boot.cpi b/boot.cpi
deleted file mode 100644
index 8a18fcc..0000000
--- a/boot.cpi
+++ /dev/null
@@ -1,56 +0,0 @@
-#!/bin/sh
-#
-# Copyright (c) 2001-2002 SuSE Linux AG, Nuernberg, Germany.
-# Copyright (c) 2008 SuSE LINUX Products GmbH, Nuernberg, Germany.
-# All rights reserved.
-#
-# /etc/init.d/boot.cpi
-#
-### BEGIN INIT INFO
-# Provides: boot.cpi
-# Required-Start: boot.rootfsck
-# Required-Stop: $null
-# Should-Start:
-# X-Start-Before:
-# Default-Start: B S
-# Default-Stop:
-# Description: Set Control Program Identification data
-# Short-Description: Set Control Program Identification data
-### END INIT INFO
-
-. /etc/rc.status
-. /etc/sysconfig/cpi
-
-rc_reset
-
-case "$1" in
- start|restart|reload)
- if test "$CPI_SET" == "yes"; then
- if test ! -e /sys/firmware/cpi; then
- echo -n "sclp_cpi interface not found"
- rc_status -s
- rc_exit
- fi
- echo "$CPI_SYSTEM_NAME" >/sys/firmware/cpi/system_name
- echo "$CPI_SYSPLEX_NAME" >/sys/firmware/cpi/sysplex_name
- echo "LINUX " >/sys/firmware/cpi/system_type
- awk "BEGIN { split(\"`uname -r`\",x,\".\"); printf(\"0x0000000000%02x%02x%02x\", x[1],x[2],x[3]) }" >/sys/firmware/cpi/system_level
- echo 1 >/sys/firmware/cpi/set 2>/dev/null
- rc_status -v -r
- else
- rc_status -u
- fi
- ;;
- stop)
- rc_status -s
- ;;
- status)
- rc_status -s
- ;;
- *)
- echo "Usage: $0 {start|stop|status|restart|reload}"
- exit 1
- ;;
-esac
-
-rc_exit
diff --git a/dasdfmt-retry-BIODASDINFO-if-device-is-busy.patch b/dasdfmt-retry-BIODASDINFO-if-device-is-busy.patch
index 11cd2b0..2fdc4e2 100644
--- a/dasdfmt-retry-BIODASDINFO-if-device-is-busy.patch
+++ b/dasdfmt-retry-BIODASDINFO-if-device-is-busy.patch
@@ -23,49 +23,44 @@ diff --git a/dasdfmt/dasdfmt.c b/dasdfmt/dasdfmt.c
index e1877ac..f03cbad 100644
--- a/dasdfmt/dasdfmt.c
+++ b/dasdfmt/dasdfmt.c
-@@ -270,7 +270,7 @@ static void init_info(dasdfmt_info_t *info)
+@@ -588,7 +616,7 @@
+ */
static void check_disk(dasdfmt_info_t *info)
{
- dasd_information_t dasd_info;
- int ro, errno_save;
+ int ro, errno_save, i = 0;
if (ioctl(filedes, BLKROGET, &ro) != 0) {
errno_save = errno;
-@@ -283,7 +283,7 @@ static void check_disk(dasdfmt_info_t *info)
- if (ro) {
+@@ -602,9 +630,27 @@
+ if (ro)
ERRMSG_EXIT(EXIT_FAILURE, "Disk is read only!\n");
- }
--
-+retry:
- if (ioctl(filedes, BIODASDINFO, &dasd_info) != 0) {
- errno_save = errno;
- close(filedes);
-@@ -294,8 +294,23 @@ static void check_disk(dasdfmt_info_t *info)
- }
- if (!info->force)
-- if (dasd_info.open_count > 1)
+- if (!info->force)
++ if (!info->force) {
+ /*
+ * udev strikes again.
+ * Modern udev will issue a 'change' event whenever
+ * a device opened with O_RDWR is closed again.
-+ * On the ground that program _might_ have changed
++ * On the grounds that program _might_ have changed
+ * the partition table.
+ * And confusing the hell out ouf anyone else.
+ * Bah.
-+ */
-+ if (dasd_info.open_count > 1) {
-+ if (i < 5) {
-+ ++i;
++ */
++ for ( i = 0 ; i < 6 ; i++ ) {
++ if (info->dasd_info.open_count > 1) {
++ get_device_info(info);
+ sleep(1);
-+ goto retry;
+ }
- ERRMSG_EXIT(EXIT_BUSY, "Disk in use!\n");
++ else break;
++
+ }
+ if (info->dasd_info.open_count > 1)
+ ERRMSG_EXIT(EXIT_BUSY, "Disk in use!\n");
++ }
- info->usage_count = dasd_info.open_count;
- info->devno = dasd_info.devno;
+ if (strncmp(info->dasd_info.type, "ECKD", 4) != 0) {
+ ERRMSG_EXIT(EXIT_FAILURE,
--
1.8.4.5
diff --git a/mkdump.8 b/mkdump.8
index a1335af..b80e543 100644
--- a/mkdump.8
+++ b/mkdump.8
@@ -6,7 +6,7 @@ mkdump \- Preparing disks for use as S/390 dump device.
.B mkdump
[\fIOPTIONS\fR] [\fIDEVICE\fR]...
.SH DESCRIPTION
-mkdump 2.0.2
+mkdump 2.0.3
.PP
Prepare one or more volumes for use as S/390 dump device. Supported devices
are ECKD DASD and SCSI over zFCP disks, while multi\-volumes are limited to DASD.
diff --git a/mkdump.pl b/mkdump.pl
index b69a4ac..f354f57 100644
--- a/mkdump.pl
+++ b/mkdump.pl
@@ -4,7 +4,8 @@
# mkdump.pl - Preparing disks for use as S/390 dump device
#
# Copyright (c) 2011 Tim Hardeck, SUSE LINUX Products GmbH
-# bases on mkdump.sh (c) 2004 Hannes Reinecke, SuSE AG
+# Copyright (c) 2017 SUSE LINUX GmbH, Nuernberg, Germany.
+# Based on mkdump.sh (c) 2004 Hannes Reinecke, SuSE AG
#
# License:
#
@@ -30,7 +31,7 @@ use warnings;
use Fcntl;
use Getopt::Long;
-my $VERSION = "2.0.2";
+my $VERSION = "2.0.3";
my $BLKID = "/sbin/blkid";
my $PARTED = "/usr/sbin/parted";
@@ -41,7 +42,6 @@ my $ZIPL = "/sbin/zipl";
my $UDEVADM = "/sbin/udevadm";
my $ZGETDUMP = "/sbin/zgetdump";
-my $MNTPOINT = "/var/run/mkdump." . getppid();
# temporary DASD device configuration file for Zipl
my $MDPATH = "/tmp/mvdump.conf";
# zFCP dump dir, without a leading '/'
@@ -57,11 +57,6 @@ sub cleanup
if (-e $MDPATH) {
system("rm -f $MDPATH");
}
- # zFCP
- if (-d $MNTPOINT) {
- system("umount $MNTPOINT");
- system("rmdir $MNTPOINT");
- }
}
sub exit_with
@@ -230,15 +225,10 @@ sub print_device
$output .= "\t$adapter\t$wwpn\t$lun";
# check for dump record
- system("mkdir -p $MNTPOINT");
- if (get_partition_num($device) == 1 and system("mount -o ro ${device}1 $MNTPOINT 2>/dev/null") == 0) {
- if ( -r "$MNTPOINT/bootmap" and -d "$MNTPOINT/$ZFCP_DUMP_DIR") {
- $dump_device = 1;
- $output .= "\tdumpdevice";
- }
- system("umount $MNTPOINT");
+ if (system("$ZGETDUMP -d ${device} >/dev/null 2>&1") == 0) {
+ $dump_device = 1;
+ $output .= "\tdumpdevice";
}
- system("rmdir $MNTPOINT");
}
if ($only_dump_disks) {
if ($dump_device) {
@@ -324,7 +314,7 @@ sub prepare_dasd
# check formatting
for my $device (@devices) {
# determine disk layout
- my ($fmtstr) = `$DASDVIEW -x -f $device` =~ /(\w\w\w) formatted/;
+ my ($fmtstr) = `$DASDVIEW -x $device` =~ /(\w\w\w) formatted/;
SWITCH:
for($fmtstr) {
diff --git a/s390-tools-1.34.0.tar.bz2 b/s390-tools-1.34.0.tar.bz2
deleted file mode 100644
index 9092dfa..0000000
--- a/s390-tools-1.34.0.tar.bz2
+++ /dev/null
@@ -1,3 +0,0 @@
-version https://git-lfs.github.com/spec/v1
-oid sha256:6f4e9d6f27cfad77ff84ac7a332a7fb2e17b620b3c2da4fc2ba47d2c77c94287
-size 814967
diff --git a/s390-tools-2.1.0.tar.gz b/s390-tools-2.1.0.tar.gz
new file mode 100644
index 0000000..aada5a0
--- /dev/null
+++ b/s390-tools-2.1.0.tar.gz
@@ -0,0 +1,3 @@
+version https://git-lfs.github.com/spec/v1
+oid sha256:b5ce3fa0de074dfe7d16c74771fcb5092c4a6f575168bf4e13f88275a6370122
+size 1046910
diff --git a/s390-tools-sles12-fdasd-skip-partition-check-and-BLKRRPART-ioctl.patch b/s390-tools-sles12-fdasd-skip-partition-check-and-BLKRRPART-ioctl.patch
index 72c85e2..60237ca 100644
--- a/s390-tools-sles12-fdasd-skip-partition-check-and-BLKRRPART-ioctl.patch
+++ b/s390-tools-sles12-fdasd-skip-partition-check-and-BLKRRPART-ioctl.patch
@@ -19,27 +19,9 @@ diff --git a/fdasd/fdasd.c b/fdasd/fdasd.c
index 4503d3e..f04dc3d 100644
--- a/fdasd/fdasd.c
+++ b/fdasd/fdasd.c
-@@ -826,7 +826,7 @@ fdasd_check_conffile_input (fdasd_anchor_t *anc,
- * Verifies the specified block device.
- */
- static void
--fdasd_verify_device (fdasd_anchor_t *anc, char *name)
-+fdasd_verify_device (fdasd_anchor_t *anc, char *name)
- {
- struct stat dst;
- char err_str[ERROR_STRING_SIZE];
-@@ -847,7 +847,7 @@ fdasd_verify_device (fdasd_anchor_t *anc, char *name)
- fdasd_error(anc, device_verification_failed, err_str);
+@@ -1247,9 +1247,12 @@
}
-- if (minor (dst.st_rdev) & PARTN_MASK) {
-+ if (!anc->force_virtual && minor (dst.st_rdev) & PARTN_MASK) {
- snprintf(err_str, ERROR_STRING_SIZE,
- "Partition '%s' (%d/%d) detected where device is "
- "required\n", name,
-@@ -1156,9 +1156,12 @@ fdasd_reread_partition_table (fdasd_anchor_t *anc)
- }
-
if (ioctl(fd, BLKRRPART, NULL) != 0) {
- close(fd);
- fdasd_error(anc, unable_to_ioctl, "Error while rereading "
@@ -52,6 +34,4 @@ index 4503d3e..f04dc3d 100644
+ }
}
close(fd);
- }
---
-1.8.5.2
+ }
diff --git a/s390-tools-sles12-pardasdfmt.patch b/s390-tools-sles12-pardasdfmt.patch
index c0a4b14..e8d077d 100644
--- a/s390-tools-sles12-pardasdfmt.patch
+++ b/s390-tools-sles12-pardasdfmt.patch
@@ -10,12 +10,12 @@
- [-r \fIcylinder\fR] [-b \fIblksize\fR] [-l \fIvolser\fR] [-d \fIlayout\fR]
+ [-r \fIcylinder\fR] [-b \fIblksize\fR] [-l \fIvolser\fR] [-d \fIlayout\fR] [-P maxpar]
.br
-- [-L] [-V] [-F] [-k] [-C] \fIdevice\fR
-+ [-L] [-V] [-F] [-k] [-C] \fIdevice\fR ...
+- [-L] [-V] [-F] [-k] [-C] [-M \fImode\fR] \fIdevice\fR
++ [-L] [-V] [-F] [-k] [-C] [-M \fImode\fR] \fIdevice\fR ...
.SH DESCRIPTION
\fBdasdfmt\fR formats a DASD (ECKD) disk drive to prepare it
-@@ -91,7 +91,7 @@
+@@ -95,7 +95,7 @@
running in background or redirecting the output to a file.
.TP
@@ -24,7 +24,7 @@
Print one line for each formatted cylinder showing the number of the
cylinder and percentage of formatting process.
Intended to be used by higher level interfaces.
-@@ -123,6 +123,20 @@
+@@ -152,6 +152,20 @@
and always be a power of two. The recommended blocksize is 4096 bytes.
.TP
@@ -47,7 +47,7 @@
to disk after formatting. If no label is specified, a sensible default
--- s390-tools-1.34.0/dasdfmt/dasdfmt.h 2016-04-14 16:43:51.000000000 -0400
+++ s390-tools-1.34.0/dasdfmt/dasdfmt.h 2016-04-14 16:35:01.000000000 -0400
-@@ -195,6 +195,7 @@
+@@ -260,6 +260,7 @@
#define LABEL_LENGTH 14
#define VLABEL_CHARS 84
#define LINE_LENGTH 80
diff --git a/s390-tools-sles12-zipl_boot_msg.patch b/s390-tools-sles12-zipl_boot_msg.patch
index b933842..1a84cb2 100644
--- a/s390-tools-sles12-zipl_boot_msg.patch
+++ b/s390-tools-sles12-zipl_boot_msg.patch
@@ -1,13 +1,16 @@
--- s390-tools-1.24.1/zipl/boot/menu.c 2013-12-18 11:11:45.000000000 -0500
+++ s390-tools-1.24.1/zipl/boot/menu.c 2013-12-18 11:59:18.000000000 -0500
-@@ -166,7 +166,9 @@
+@@ -167,8 +167,11 @@
+ /* print config list */
menu_list();
- if (is_zvm())
+- if (is_zvm())
- printf("Note: VM users please use '#cp vi vmsg '\n");
++ if (is_zvm()) {
+ printf(" \n");
+ printf("Note: VM users please use '#cp vi vmsg '\n");
+ printf(" \n");
++ }
value = menu_read();
diff --git a/s390-tools-sles12sp2-chiucvallow-verify.patch b/s390-tools-sles12sp2-chiucvallow-verify.patch
deleted file mode 100644
index 4cb9479..0000000
--- a/s390-tools-sles12sp2-chiucvallow-verify.patch
+++ /dev/null
@@ -1,35 +0,0 @@
-Subject: [PATCH] [BZ 141695] chiucvallow: correct verification return code
-From: Hendrik Brueckner
-
-Description: chiucvallow: correct verification return code
-Symptom: When running test_chiucvallow.sh, the script is
- failing to validate filter files with special
- characters.
- because the chiucvallow.in -V is
-Problem: The chiucvallow -V is returning code 2 and for
- this reason, the script gives an error message,
- but the filter file is validated successfully
- manually.
-Solution: Correct the return code of chiucvallow to handle
- verification failures correctly.
-Reproduction: See Symptom.
-Upstream-ID: -
-Problem-ID: 141695
-
-Signed-off-by: Hendrik Brueckner
----
- iucvterm/bin/chiucvallow.in | 3 ++-
- 1 file changed, 2 insertions(+), 1 deletion(-)
-
---- a/iucvterm/bin/chiucvallow.in
-+++ b/iucvterm/bin/chiucvallow.in
-@@ -115,7 +115,8 @@ verify_filter(){
-
- printf "\n$PRG: Verification summary: verified=%d failed=%d size=%d bytes\n" \
- $count $failed $fsize
-- return 2
-+
-+ test $failed -eq 0 || return 2
- }
-
- #
diff --git a/s390-tools-sles12sp2-chreipl-virtio.patch b/s390-tools-sles12sp2-chreipl-virtio.patch
deleted file mode 100644
index 6684311..0000000
--- a/s390-tools-sles12sp2-chreipl-virtio.patch
+++ /dev/null
@@ -1,49 +0,0 @@
-Subject: [PATCH] [BZ 143839] chreipl/virtio: fix chreipl node for virtio-blk disks
-From: Christian Borntraeger
-
-Description: chreipl/virtio: fix chreipl node for virtio-blk disks
-Symptom: chreipl node for a virtio-blk disk fails:
- Could not find DASD CCW device "virtio1"
-Problem: The sysfs walking code to resolve the device is not
- handling virtio devices correctly.
-Solution: Use the realpath of the device to get the busid for all
- supported devices.
-Reproduction: Run chreipl node with virtio-blk devices as
- backing for mount point.
-Upstream-ID: -
-Problem-ID: 143839
-
-Signed-off-by: Christian Borntraeger
----
- ipl_tools/ccw.c | 12 +++++-------
- 1 file changed, 5 insertions(+), 7 deletions(-)
-
---- a/ipl_tools/ccw.c
-+++ b/ipl_tools/ccw.c
-@@ -76,21 +76,19 @@ out_fclose:
- static int ccw_busid_get_sysfs_new(const char *device, char *busid)
- {
- char path[PATH_MAX], buf[4096];
-- char *ptr;
-
- memset(buf, 0, sizeof(buf));
- snprintf(path, sizeof(path), "/sys/block/%s/device", device);
-- if (readlink(path, buf, sizeof(buf) - 1) == -1)
-+ if (realpath(path, buf) == NULL)
- return -1;
-
- /*
- * The output has the following format:
-- * ../../../0.0.4e13
-+ * /sys/devices/css0/0.0.0119/0.0.3f19/block/dasda
-+ * /sys/devices/css0/0.0.0000/0.0.0000/virtio0/block/vda
- */
-- ptr = strrchr(buf, '/');
-- if (!ptr)
-- ERR_EXIT("Could not read \"%s\"", path);
-- strncpy(busid, ptr + 1, 9);
-+ if (sscanf(buf, "/sys/devices/css0/%*[0-9a-f.]/%[0-9a-f.]", busid) != 1)
-+ return -1;
- return 0;
- }
-
diff --git a/s390-tools-sles12sp2-chzdev-disable-root-update.patch b/s390-tools-sles12sp2-chzdev-disable-root-update.patch
deleted file mode 100644
index 405e9be..0000000
--- a/s390-tools-sles12sp2-chzdev-disable-root-update.patch
+++ /dev/null
@@ -1,32 +0,0 @@
-Subject: [PATCH] [BZ 147113] chzdev: Disable root device update mechanism
-From: Peter Oberparleiter
-
-Description: chzdev: Disable root device update mechanism
-Symptom: chzdev reports problems updating /etc/zipl.conf and running
- zipl.
-Problem: When chzdev changes the root device persistent configuration,
- it performs additional steps to ensure that these changes are
- correctly applied. Example for such steps are: updating the
- initial RAM disk, modifying the zipl.conf configuration file
- and running zipl. As these steps are incompatible with the target
- distribution, they fail.
-Solution: Suppress chzdev's root device update mechanism.
-Reproduction: chzdev --enable --persistent --by-path /
-Upstream-ID: -
-Problem-ID: 147113
-
-Signed-off-by: Peter Oberparleiter
----
- zdev/src/chzdev.c | 1 +
- 1 file changed, 1 insertion(+)
-
---- a/zdev/src/chzdev.c
-+++ b/zdev/src/chzdev.c
-@@ -233,6 +233,7 @@ static void init_options(struct options
- opts->settings = strlist_new();
- opts->remove = strlist_new();
- opts->base = strlist_new();
-+ opts->no_root_check = 1;
- }
-
- /* Release memory used in options data structure. */
diff --git a/s390-tools-sles12sp2-feat-01-dasd-query-host.patch b/s390-tools-sles12sp2-feat-01-dasd-query-host.patch
deleted file mode 100644
index b9bc686..0000000
--- a/s390-tools-sles12sp2-feat-01-dasd-query-host.patch
+++ /dev/null
@@ -1,744 +0,0 @@
-Subject: [PATCH] [FEAT LS1213] dasd: add query host access to volume support
-From: Stefan Haberland
-
-Summary: dasd: add query host access to volume support
-Description: With this feature, applications can query if a DASD volume is
- online to another operating system instances by checking the
- online status of all attached hosts from the storage server.
-Upstream-ID: -
-Problem-ID: LS1213
-
-Signed-off-by: Stefan Haberland
----
- dasdfmt/dasdfmt.8 | 7 ++
- dasdfmt/dasdfmt.c | 38 ++++++++++++--
- dasdfmt/dasdfmt.h | 4 +
- fdasd/fdasd.8 | 9 ++-
- fdasd/fdasd.c | 32 ++++++++++++
- fdasd/fdasd.h | 28 +++++-----
- include/libzds.h | 1
- include/u2s.h | 5 +
- libu2s/u2s.c | 43 ++++++++++++++++
- libzds/Makefile | 2
- libzds/libzds.c | 46 +++++++++++++++--
- zconf/lsdasd | 139 +++++++++++++++++++++++++++++++++++++++++++++++++++++-
- zconf/lsdasd.8 | 5 +
- zdsfs/zdsfs.1 | 4 +
- zdsfs/zdsfs.c | 18 ++++++
- 15 files changed, 345 insertions(+), 36 deletions(-)
-
---- a/dasdfmt/dasdfmt.8
-+++ b/dasdfmt/dasdfmt.8
-@@ -7,7 +7,7 @@ dasdfmt \- formatting of DASD (ECKD) dis
- .br
- [-r \fIcylinder\fR] [-b \fIblksize\fR] [-l \fIvolser\fR] [-d \fIlayout\fR]
- .br
-- [-L] [-V] [-F] [-k] \fIdevice\fR
-+ [-L] [-V] [-F] [-k] [-C] \fIdevice\fR
-
- .SH DESCRIPTION
- \fBdasdfmt\fR formats a DASD (ECKD) disk drive to prepare it
-@@ -70,6 +70,11 @@ Print version number and exit.
- Formats the device without checking, if the device is in use.
-
- .TP
-+\fB-C\fR or \fB--check_host_count\fR
-+Force dasdfmt to check the host access open count to ensure the device
-+is not online on another operating system instance
-+
-+.TP
- \fB-d\fR \fIlayout\fR or \fB--disk_layout\fR=\fIlayout\fR
- Formats the device with compatible disk layout or linux disk layout.
- \fIlayout\fR is either \fIcdl\fR for the compatible disk layout
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -18,6 +18,8 @@
- #include "util_proc.h"
- #include "dasd_sys.h"
-
-+#define BUSIDSIZE 8
-+
- /* Full tool name */
- static const char tool_name[] = "dasdfmt: zSeries DASD format program";
-
-@@ -46,7 +48,7 @@ print_version (void)
- */
- static void exit_usage(int exitcode)
- {
-- printf("Usage: %s [-htvypPLVFk]\n"
-+ printf("Usage: %s [-htvypPLVFkC]\n"
- " [-l | --label=]\n"
- " [-b | --blocksize=]\n"
- " [-d | --disk_layout=]\n"
-@@ -65,6 +67,9 @@ static void exit_usage(int exitcode)
- " -v means verbose mode\n"
- " -F means don't check if the device is in use\n"
- " -k means keep volume serial\n"
-+ " -C or --check_host_count means force dasdfmt to check\n"
-+ " the host access open count to ensure the device\n"
-+ " is not online on another operating system instance\n"
- " --norecordzero prevent storage server from modifying"
- " record 0\n\n"
- " is the volume identifier, which is converted\n"
-@@ -190,7 +195,6 @@ static void get_device_name(dasdfmt_info
- }
- }
-
--
- /*
- * initialize the dasdfmt info structure
- */
-@@ -214,6 +218,7 @@ static void init_info(dasdfmt_info_t *in
- info->node_specified = 0;
- info->device_id = 0;
- info->keep_volser = 0;
-+ info->force_host = 0;
- }
-
-
-@@ -776,6 +781,7 @@ static void do_format_dasd(dasdfmt_info_
- dasd_information_t dasd_info;
- struct dasd_eckd_characteristics *characteristics;
- unsigned int cylinders, heads;
-+ int count;
-
- if (info->verbosity > 0) printf("Retrieving disk geometry...\n");
-
-@@ -835,6 +841,27 @@ static void do_format_dasd(dasdfmt_info_
- if ((info->verbosity > 0) || (!info->withoutprompt))
- dasdfmt_print_info(info, vlabel, cylinders, heads, p);
-
-+ count = u2s_get_host_access_count(info->devname);
-+ if (info->force_host) {
-+ if (count > 1) {
-+ ERRMSG_EXIT(EXIT_FAILURE,
-+ "\n%s: Disk %s is online on OS instances in %d different LPARs.\n"
-+ "Note: Your installation might include z/VM systems that are configured to\n"
-+ "automatically vary on disks, regardless of whether they are subsequently used.\n\n",
-+ prog_name, info->devname, count);
-+ } else if (count < 0) {
-+ ERRMSG("\nHosts access information not available for disk %s.\n\n",
-+ info->devname);
-+ return;
-+ }
-+ } else if (count > 1)
-+ ERRMSG("\nWARNING:\n"
-+ "Disk %s is online on operating system instances in %d different LPARs.\n"
-+ "Ensure that the disk is not being used by a system outside your LPAR.\n"
-+ "Note: Your installation might include z/VM systems that are configured to\n"
-+ "automatically vary on disks, regardless of whether they are subsequently used.\n",
-+ info->devname, count);
-+
- if (!info->testmode) {
- if (!info->withoutprompt) {
- printf("\n--->> ATTENTION! <<---\n");
-@@ -916,10 +943,6 @@ int main(int argc,char *argv[])
- info.force=1;
- break;
-
-- case 'C':
-- format_params.intensity |= DASD_FMT_INT_COMPAT;
-- break;
--
- case 'd' :
- if (strncmp(optarg,"cdl",3)==0)
- {
-@@ -1017,6 +1040,9 @@ int main(int argc,char *argv[])
- case 'k' :
- info.keep_volser=1;
- break;
-+ case 'C':
-+ info.force_host = 1;
-+ break;
- case -1:
- /* End of options string - start of devices list */
- info.device_id = optind;
---- a/dasdfmt/dasdfmt.h
-+++ b/dasdfmt/dasdfmt.h
-@@ -214,7 +214,7 @@ typedef struct format_data_t {
- if (*endptr) ERRMSG_EXIT(EXIT_MISUSE,"%s: " str " " \
- "is in invalid format\n",prog_name);}
-
--#define dasdfmt_getopt_string "b:n:l:f:d:m:r:hpPLtyvVFk"
-+#define dasdfmt_getopt_string "b:n:l:f:d:m:r:hpPLtyvVFkC"
-
- static struct option dasdfmt_getopt_long_options[]=
- {
-@@ -233,6 +233,7 @@ static struct option dasdfmt_getopt_long
- { "help", 0, 0, 'h'},
- { "keep_volser", 0, 0, 'k'},
- { "norecordzero", 0, 0, 'z'},
-+ { "check_host_count", 0, 0, 'C'},
- {0, 0, 0, 0}
- };
-
-@@ -265,6 +266,7 @@ typedef struct dasdfmt_info {
- int node_specified;
- int device_id;
- int keep_volser;
-+ int force_host;
- } dasdfmt_info_t;
-
-
---- a/fdasd/fdasd.8
-+++ b/fdasd/fdasd.8
-@@ -4,11 +4,11 @@ fdasd \- partitioning tool.
- .SH SYNOPSIS
- interactive mode:
- .br
-- \fBfdasd\fR [-s] [-r] \fIdevice\fR
-+ \fBfdasd\fR [-s] [-r] [-C] \fIdevice\fR
- .br
- command line mode:
- .br
-- \fBfdasd\fR [-s] [-r] {-a[-k|-l \fIvolser\fR]|-i|-p|-c \fIconf_file\fR}
-+ \fBfdasd\fR [-s] [-r] [-C] {-a[-k|-l \fIvolser\fR]|-i|-p|-c \fIconf_file\fR}
- [-f \fI[type,blocksize]\fR] \fIdevice\fR
- .br
- help:
-@@ -131,6 +131,11 @@ In combination with the -s option fdasd
- partition table.
-
- .TP
-+\fB-C\fR or \fB--check_host_count\fR
-+Force fdasd to check the host access open count to ensure the device
-+is not online on another operating system instance
-+
-+.TP
- \fB-f\fR \fI[type,blocksize]\fR or \fB--force\fR \fI[type,blocksize]\fR
- Force fdasd to work on non DASD devices.
- .br
---- a/fdasd/fdasd.c
-+++ b/fdasd/fdasd.c
-@@ -369,7 +369,10 @@ fdasd_usage (void)
- " found in CONFIGFILE\n"
- "-i, --volser Print volume serial\n"
- "-p, --table Print partition table\n"
-- "-f, --force Force fdasd to work on non DASD devices\n");
-+ "-f, --force Force fdasd to work on non DASD devices\n"
-+ "-C, --check_host_count Force fdasd to check the host access\n"
-+ " open count to ensure the device is not\n"
-+ " online on another operating system instance\n");
- }
-
-
-@@ -572,6 +575,9 @@ fdasd_parse_options (fdasd_anchor_t *anc
- anc->force_virtual++;
- fdasd_parse_force_options(anc, optarg);
- break;
-+ case 'C':
-+ anc->force_host++;
-+ break;
- case -1:
- /* End of options string - start of devices list */
- break;
-@@ -824,6 +830,7 @@ fdasd_verify_device (fdasd_anchor_t *anc
- {
- struct stat dst;
- char err_str[ERROR_STRING_SIZE];
-+ int count;
-
- if ((stat(name, &dst)) < 0 ) {
- snprintf(err_str, ERROR_STRING_SIZE,
-@@ -856,6 +863,29 @@ fdasd_verify_device (fdasd_anchor_t *anc
- fdasd_error(anc, device_verification_failed, err_str);
- }
-
-+ count = u2s_get_host_access_count(name);
-+ if (anc->force_host) {
-+ if (count > 1) {
-+ snprintf(err_str, ERROR_STRING_SIZE,
-+ "Disk %s is online on operating system instances in %d different LPARs.\n"
-+ "Note: Your installation might include z/VM systems that are configured to\n"
-+ "automatically vary on disks, regardless of whether they are subsequently used.\n",
-+ name, count);
-+ fdasd_error(anc, device_verification_failed, err_str);
-+ } else if (count < 0) {
-+ snprintf(err_str, ERROR_STRING_SIZE,
-+ "Hosts access information not available for disk %s.\n",
-+ name);
-+ fdasd_error(anc, device_verification_failed, err_str);
-+ }
-+ } else if (count > 1)
-+ printf("\nWARNING:\n"
-+ "Disk %s is online on operating system instances in %d different LPARs.\n"
-+ "Ensure that the disk is not being used by a system outside your LPAR.\n"
-+ "Note: Your installation might include z/VM systems that are configured to\n"
-+ "automatically vary on disks, regardless of whether they are subsequently used.\n\n",
-+ name, count);
-+
- if (anc->verbose)
- printf("Verification successful for '%s' (%d/%d)\n", name,
- (unsigned short) major(dst.st_rdev),
---- a/fdasd/fdasd.h
-+++ b/fdasd/fdasd.h
-@@ -157,22 +157,23 @@ struct dasd_eckd_characteristics {
- #define PARTITION_GPFS 5
-
- static struct option fdasd_long_options[] = {
-- { "version", no_argument, NULL, 'v'},
-- { "auto", no_argument, NULL, 'a'},
-- { "silent", no_argument, NULL, 's'},
-- { "verbose", no_argument, NULL, 'r'},
-- { "label", required_argument, NULL, 'l'},
-- { "config", required_argument, NULL, 'c'},
-- { "help", no_argument, NULL, 'h'},
-- { "table", no_argument, NULL, 'p'},
-- { "volser", no_argument, NULL, 'i'},
-- { "keep_volser", no_argument, NULL, 'k'},
-- { "force", optional_argument, NULL, 'f'},
-- { 0, 0, 0, 0 }
-+ { "version", no_argument, NULL, 'v'},
-+ { "auto", no_argument, NULL, 'a'},
-+ { "silent", no_argument, NULL, 's'},
-+ { "verbose", no_argument, NULL, 'r'},
-+ { "label", required_argument, NULL, 'l'},
-+ { "config", required_argument, NULL, 'c'},
-+ { "help", no_argument, NULL, 'h'},
-+ { "table", no_argument, NULL, 'p'},
-+ { "volser", no_argument, NULL, 'i'},
-+ { "keep_volser", no_argument, NULL, 'k'},
-+ { "force", optional_argument, NULL, 'f'},
-+ { "check_host_count", no_argument, NULL, 'C'},
-+ { 0, 0, 0, 0 }
- };
-
- /* Command line option abbreviations */
--static const char option_string[] = "vasrl:c:hpikf::";
-+static const char option_string[] = "vasrl:c:hpikf::C";
-
- struct fdasd_options {
- char *device;
-@@ -214,6 +215,7 @@ typedef struct fdasd_anchor {
- int print_volser;
- int keep_volser;
- int force_virtual;
-+ int force_host;
- int big_disk;
- int silent;
- int verbose;
---- a/include/libzds.h
-+++ b/include/libzds.h
-@@ -803,6 +803,7 @@ int lzds_zdsroot_extract_datasets_from_d
- void lzds_DS1RECFM_to_recfm(char DS1RECFM, char *buffer);
-
-
-+int lzds_analyse_open_count(struct zdsroot *root, int warn);
-
- /** @} */ /* end of group libzds_functions_helper */
-
---- a/include/u2s.h
-+++ b/include/u2s.h
-@@ -13,7 +13,8 @@
-
- #define U2S_BUS_ID_SIZE 32
-
--int
--u2s_getbusid(char * devicenode, char * busid);
-+int u2s_getbusid(char *, char *);
-+int u2s_read_attribute(char *, char *, char *, size_t);
-+int u2s_get_host_access_count(char *);
-
- #endif /* U2S_H */
---- a/libu2s/u2s.c
-+++ b/libu2s/u2s.c
-@@ -17,6 +17,7 @@
- #include
- #include
- #include
-+#include
-
- #include "u2s.h"
-
-@@ -282,3 +283,45 @@ int u2s_getbusid(char *devicenode, char
-
- return rc;
- }
-+
-+/*
-+ * Attempts to find the sysfs entry for the given busid and reads
-+ * the contents of a specified attribute to the buffer
-+ */
-+int u2s_read_attribute(char *busid, char *attribute, char *buffer,
-+ size_t count)
-+{
-+ char path[100];
-+ int rc, fd;
-+ ssize_t rcount;
-+
-+ rc = 0;
-+ snprintf(path, sizeof(path), "/sys/bus/ccw/devices/%s/%s",
-+ busid, attribute);
-+ fd = open(path, O_RDONLY);
-+ if (fd < 0)
-+ return errno;
-+ rcount = read(fd, buffer, count);
-+ if (rcount < 0)
-+ rc = errno;
-+ close(fd);
-+ return rc;
-+}
-+
-+int u2s_get_host_access_count(char *devicenode)
-+{
-+ char busid[BUSIDSIZE];
-+ unsigned long value;
-+ char buffer[10];
-+ char *endp;
-+
-+ u2s_getbusid(devicenode, busid);
-+ u2s_read_attribute(busid, "host_access_count", buffer, sizeof(buffer));
-+
-+ value = strtoul(buffer, &endp, 0);
-+
-+ if (endp == buffer)
-+ return -EINVAL;
-+
-+ return value;
-+}
---- a/libzds/Makefile
-+++ b/libzds/Makefile
-@@ -8,7 +8,7 @@ CFLAGS += -D_FILE_OFFSET_BITS=64
-
- all: libzds.a
-
--libzds.a: libzds.o $(rootdir)/libutil/util_list.o $(rootdir)/libvtoc/vtoc.o
-+libzds.a: libzds.o $(rootdir)/libutil/util_list.o $(rootdir)/libvtoc/vtoc.o $(rootdir)/libu2s/u2s.o
-
- libzds.o: ../include/libzds.h
-
---- a/libzds/libzds.c
-+++ b/libzds/libzds.c
-@@ -29,6 +29,7 @@
- #include "libzds.h"
-
- #include "util.h"
-+#include "u2s.h"
-
- #include
- #include
-@@ -72,6 +73,8 @@ struct errorlog {
- */
- #define ERRORMSG 240
-
-+#define BUSIDSIZE 8
-+
- /**
- * @brief An internal structure that represents an entry in the error log.
- */
-@@ -3760,15 +3763,46 @@ void lzds_DS1RECFM_to_recfm(char DS1RECF
- */
- }
-
-+int lzds_analyse_open_count(struct zdsroot *root, int warn)
-+{
-+ struct dasd *dasd;
-+ int value;
-+ int rc = 0;
-
-+ util_list_iterate(root->dasdlist, dasd) {
-+ value = u2s_get_host_access_count(dasd->device);
-
-+ if (value < 0) {
-+ fprintf(stderr,
-+ "Hosts access information not available for disk %s.\n",
-+ dasd->device);
-+ rc = value;
-+ continue;
-+ }
-
-+ if (value == 1)
-+ continue;
-
-+ if (warn)
-+ fprintf(stderr,
-+ "\nWARNING:\n"
-+ "Disk %s is online on operating system instances in %d different LPARs.\n"
-+ "Ensure that the disk is not being used by a system outside your LPAR.\n"
-+ "Note: Your installation might include z/VM systems that are configured to\n"
-+ "automatically vary on disks, regardless of whether they are subsequently used.\n",
-+ dasd->device, value);
-+ else {
-+ fprintf(stderr,
-+ "\nERROR:\n"
-+ "Disk %s is online on operating system instances in %d different LPARs.\n"
-+ "Ensure that the disk is not being used by a system outside your LPAR.\n"
-+ "Note: Your installation might include z/VM systems that are configured to\n"
-+ "automatically vary on disks, regardless of whether they are subsequently used.\n",
-+ dasd->device, value);
-+ rc = -EACCES;
-+ }
-+ }
-
--
--
--
--
--
--
-+ return rc;
-+}
-
---- a/zconf/lsdasd
-+++ b/zconf/lsdasd
-@@ -30,6 +30,8 @@ function PrintUsage() {
- Print old version of lsdasd output.
- -l|--long
- Print extended information about DASDs.
-+ -H|--host-access-list
-+ Print information about hosts accessing DASDs.
- -v|--verbose
- For compatibility/future use. Currently ignored.
- --version
-@@ -115,6 +117,33 @@ function listDASDDeviceDirectories() {
- }
-
- #------------------------------------------------------------------------------
-+# find dasd directory in debugfs
-+#------------------------------------------------------------------------------
-+function findDASDDebugfsDirectorie() {
-+ local mntentries
-+
-+ while read -a mntentries
-+ do
-+ if [[ "${mntentries[2]}" == "debugfs" ]]
-+ then
-+ DASD_DBF_DIR="${mntentries[1]}"
-+ break;
-+ fi
-+ done < /etc/mtab
-+ if [[ "$DASD_DBF_DIR" == "" ]]
-+ then
-+ echo "$CMD: No debugfs mount point found" >&2
-+ exit 1
-+ fi
-+ DASD_DBF_DIR="$DASD_DBF_DIR/dasd"
-+ if [[ ! -d "$DASD_DBF_DIR" ]]
-+ then
-+ echo "$CMD: Default DASD debugfs directory $DASD_DBF_DIR does not exist" >&2
-+ exit 1
-+ fi
-+}
-+
-+#------------------------------------------------------------------------------
- # gather device data and call appropriate output function
- #------------------------------------------------------------------------------
- function gatherDeviceData() {
-@@ -197,6 +226,8 @@ function gatherDeviceData() {
- extended
- elif [[ "$PRINTUID" == "true" ]]; then
- uid
-+ elif [[ "$OUTPUT" == "host" ]]; then
-+ host
- else
- newoutput
- fi
-@@ -527,6 +558,104 @@ function extended()
- "${HPF_PATHS[@]}" ;
- }
-
-+function host()
-+{
-+findDASDDebugfsDirectorie
-+
-+if [[ ! -f "$DASD_DBF_DIR/$BUSID/host_access_list" ]]
-+then
-+ printf "\n%s: hosts access information not available\n" "$BUSID"
-+ return
-+fi
-+
-+local temp=`mktemp /tmp/lsdasd.XXXXXX`
-+if test -w $temp ; then :; else
-+ printf "\nCreating temporary file failed\n"
-+ return
-+fi
-+
-+cat $DASD_DBF_DIR/$BUSID/host_access_list > $temp 2> /dev/null
-+ret=$?
-+if [[ $ret -ne 0 ]]
-+then
-+ printf "%s: hosts access information not available\n" "$BUSID"
-+ rm -f $temp
-+ return $ret
-+fi
-+
-+unset index
-+unset array
-+declare -a array
-+
-+index=(pgid status_flags sysplex_name supported_cylinder timestamp)
-+
-+for element in ${index[@]}
-+do
-+ count=0
-+
-+ declare -a $element
-+ OLDIFS=$IFS
-+ IFS=$'\n'
-+ for value in `grep $element $temp`
-+ do
-+ (( ++count ))
-+ value=$(echo -e $value | cut -d ' ' -f2)
-+ eval $element[$count]=$value
-+ done
-+ IFS=$OLDIFS
-+done
-+
-+printf "Host information for %s\n" "$BUSID";
-+printf "Path-Group-ID LPAR CPU FL Status Sysplex Max_Cyls Time\n";
-+printf "================================================================================\n";
-+
-+# mask bits for online and reserved state
-+online_reserved_mask=0xE0
-+
-+# print name value lists
-+for i in `seq 1 $count`;
-+do
-+ # get flags field
-+ value=${status_flags[$i]}
-+ # mark as hex value
-+ value=0x$value
-+ # mask online and reserved bits
-+ value=$(($value & $online_reserved_mask))
-+
-+ case $value in
-+ 0 ) # 0x00
-+ STATE="OFF"
-+ ;;
-+ 32 ) # 0x20
-+ STATE="OFF-RSV"
-+ ;;
-+ 64 ) # 0x40
-+ STATE="ON"
-+ ;;
-+ 96 ) # 0x60
-+ STATE="ON-RSV"
-+ ;;
-+ * )
-+ STATE="-"
-+ ;;
-+ esac
-+
-+ printf "%22s %02s %07s %02s %-6s %-8s %11u %10lu\n" \
-+ "${pgid[$i]}" \
-+ "${pgid[$i]:4:2}" \
-+ "${pgid[$i]:6:4}" \
-+ "${status_flags[$i]}" \
-+ "$STATE" \
-+ "${sysplex_name[$i]}" \
-+ "${supported_cylinder[$i]}" \
-+ "${timestamp[$i]}" \
-+ ;
-+done
-+printf "\n";
-+
-+rm -f $temp
-+}
-+
- function uid()
- {
- #-------------------------------------------#
-@@ -586,6 +715,9 @@ while [ $# -gt 0 ]; do
- --long|-l)
- OUTPUT="extended"
- ;;
-+ --host-access-list|-H)
-+ OUTPUT="host"
-+ ;;
- --version)
- PrintVersion
- exit 0
-@@ -626,8 +758,13 @@ fi
-
- # gather information on devices in list
- PROCESSING=" $PROCESSING | gatherDeviceData "
-+
- # sort resulting list
--PROCESSING=" $PROCESSING | sort -t: -k1n -k2 | cut -d: -f3- "
-+if [[ "$OUTPUT" == "host" ]]; then
-+ PROCESSING=" $PROCESSING"
-+else
-+ PROCESSING=" $PROCESSING | sort -t: -k1n -k2 | cut -d: -f3- "
-+fi
-
- if [[ "$PRINTUID" == "true" ]] && [[ "$OUTPUT" != "old" ]]; then
- printf "Bus-ID Name UID\n"
---- a/zconf/lsdasd.8
-+++ b/zconf/lsdasd.8
-@@ -40,11 +40,14 @@ Include only base devices.
- Old output of lsdasd for compatibility.
- .TP
- .BR -l | --long
--Extended output of lsdasd including UID, attributes and path information.
-+Extended output of lsdasd including UID and attributes.
- .TP
- .BR -u | --uid
- Output includes and is sorted by UID.
- .TP
-+.BR -H | --host-acces
-+Show information about all hosts using this device.
-+.TP
- .BR -v | --verbose
- Only for compatibility (and maybe future) use. This option currently does
- nothing.
---- a/zdsfs/zdsfs.1
-+++ b/zdsfs/zdsfs.1
-@@ -150,6 +150,10 @@ If \fI\fR is set to 0, no seek histor
- case `seek' is still supported, but a `seek' operation might result in a
- read from the beginning of the data set.
-
-+.TP
-+\fB\-o\fR check_host_count
-+Stop processing if the device is used by another operating system instance.
-+
- .SS "Applicable FUSE options (version 2.8):"
- This is a selected subset of all FUSE options. Use the zdsfs
- \fB\--help\fR option to print a full list.
---- a/zdsfs/zdsfs.c
-+++ b/zdsfs/zdsfs.c
-@@ -39,6 +39,7 @@ struct zdsfs_info {
- int devcount;
- int allow_inclomplete_multi_volume;
- int keepRDW;
-+ int host_count;
- unsigned int tracks_per_frame;
- unsigned long long seek_buffer_size;
- struct zdsroot *zdsroot;
-@@ -765,6 +766,7 @@ static const struct fuse_opt zdsfs_opts[
- FUSE_OPT_KEY("seekbuffer=", KEY_SEEKBUFFER),
- ZDSFS_OPT("rdw", keepRDW, 1),
- ZDSFS_OPT("ignore_incomplete", allow_inclomplete_multi_volume, 1),
-+ ZDSFS_OPT("check_host_count", host_count, 1),
- FUSE_OPT_END
- };
-
-@@ -790,7 +792,10 @@ static void usage(const char *progname)
- " data set are missing\n"
- " -o tracks=N Size of the track buffer in tracks (default 128)\n"
- " -o seekbuffer=S Upper limit in bytes for the seek history buffer\n"
--" size (default 1048576)\n", progname);
-+" size (default 1048576)\n"
-+" -o check_host_count Stop processing if the device is used by another\n"
-+" operating system instance\n"
-+ , progname);
- }
-
- static void zdsfs_process_device(const char *device)
-@@ -1014,6 +1019,17 @@ int main(int argc, char *argv[])
- argv[0]);
- exit(1);
- }
-+
-+ if (zdsfsinfo.host_count) {
-+ /* check, print error and exit if multiple online */
-+ rc = lzds_analyse_open_count(zdsfsinfo.zdsroot, 0);
-+ if (rc == -EACCES)
-+ goto cleanup;
-+ } else {
-+ /* check, print warning if multiple online */
-+ lzds_analyse_open_count(zdsfsinfo.zdsroot, 1);
-+ }
-+
- rc = zdsfs_verify_datasets();
- if (rc)
- goto cleanup;
diff --git a/s390-tools-sles12sp2-libu2s-Fix-busid-parsing.patch b/s390-tools-sles12sp2-libu2s-Fix-busid-parsing.patch
deleted file mode 100644
index 5283fe9..0000000
--- a/s390-tools-sles12sp2-libu2s-Fix-busid-parsing.patch
+++ /dev/null
@@ -1,37 +0,0 @@
-Subject: [PATCH] [BZ 140500] libu2s: Fix busid parsing
-From: Jan Höppner
-
-Description: libu2s: Fix busid parsing
-Symptom: dasdview displays a wrong busid and shows the error message
- "Error: dasdview: Could not retrieve raw_track_access mode
- information.", which is wrong as well.
-Problem: dasdview retrieves the busid information from libu2s. Whenever
- an FBA DASD is listed prior to the target DASD in
- /proc/dasd/devices, the parsing will fail due to an additional
- whitespace character. For example, an entry for ECKD devices
- starts with "0.0.5e30(ECKD)...", while an entry for FBA starts
- with "0.0.8000(FBA )...".
-Solution: Change the busid parsing in libu2s by explicitly looking for the
- closing bracket.
-Reproduction: Add an FBA device and then an ECKD device to the system and use
- dasdview to display information about the ECKD device.
- # dasdview -i /dev/dasdX
-Upstream-ID: -
-Problem-ID: 140500
-
-Signed-off-by: Jan Höppner
----
- libu2s/u2s.c | 2 +-
- 1 file changed, 1 insertion(+), 1 deletion(-)
-
---- a/libu2s/u2s.c
-+++ b/libu2s/u2s.c
-@@ -237,7 +237,7 @@ static int find_busid_in_proc(int maja,
- filp = fopen("/proc/dasd/devices", "r");
- if (!filp)
- return rc;
-- while (fscanf(filp, "%[^(] %*s at ( %d : %d %*[^\n]\n",
-+ while (fscanf(filp, "%[^(] %*[^)] ) at ( %d : %d %*[^\n]\n",
- bus, &majb, &minb) != EOF) {
- if ((maja == majb) && (mina == minb)) {
- strncpy(busid, bus, BUSIDSIZE + 1);
diff --git a/s390-tools-sles12sp2-lscss-allow-to-specify-devices-from-ssid-3.patch b/s390-tools-sles12sp2-lscss-allow-to-specify-devices-from-ssid-3.patch
deleted file mode 100644
index 9677209..0000000
--- a/s390-tools-sles12sp2-lscss-allow-to-specify-devices-from-ssid-3.patch
+++ /dev/null
@@ -1,37 +0,0 @@
-Subject: lscss: allow to specify devices from ssid > 2
-From: Sebastian Ott
-
-Description: lscss: allow to specify devices from ssid > 2
-Symptom: Specifying devices with ssid > 2 results in:
- "Syntax error: 0.3.0000"
-Problem: SSIDs > 2 are disallowed
-Solution: allow SSIDs > 2
-Reproduction: lscss 0.3.0000
-Upstream-ID: -
-Problem-ID: 150872
-
-Signed-off-by: Sebastian Ott
----
- zconf/lscss | 4 ++--
- 1 file changed, 2 insertions(+), 2 deletions(-)
-
---- a/zconf/lscss
-+++ b/zconf/lscss
-@@ -61,7 +61,7 @@ SUBCHANNEL_TYPE_IO=0
- SUBCHANNEL_TYPE_CHSC=1
- SUBCHANNEL_TYPE_EADM=3
-
--IDFORMAT=[[:xdigit:]]*.[0-3].[[:xdigit:]][[:xdigit:]][[:xdigit:]][[:xdigit:]]
-+IDFORMAT=[[:xdigit:]]*.[[:xdigit:]].[[:xdigit:]][[:xdigit:]][[:xdigit:]][[:xdigit:]]
-
- unset SCH_IO SCH_CHSC SCH_EADM SHOW_AVAIL UPPERCASE SHORTID DEVTYPES DEVRANGE
- unset RANGE SHOW_VPM
-@@ -81,7 +81,7 @@ function check_id() #return 0 if $1 has
- fi
- ;;
- 3)
-- if [ ${#__ID[2]} != 4 -o ${#__ID[1]} != 1 -o "${__ID[1]//[012]/}" \
-+ if [ ${#__ID[2]} != 4 -o ${#__ID[1]} != 1 \
- -o \( ${#__ID[0]} != 2 -a ${#__ID[0]} != 1 \) ] ;then
- return 1
- fi
diff --git a/s390-tools-sles12sp2-zipl-fix-failed-start-subchannel.patch b/s390-tools-sles12sp2-zipl-fix-failed-start-subchannel.patch
deleted file mode 100644
index 68f199c..0000000
--- a/s390-tools-sles12sp2-zipl-fix-failed-start-subchannel.patch
+++ /dev/null
@@ -1,35 +0,0 @@
-Subject: [PATCH] [BZ 143050] zipl/boot: fix failed start subchannel in FBA loader
-From: Stefan Haberland
-
-Description: zipl/boot: fix failed start subchannel in FBA loader
-Symptom: IPL of a FBA device fails with:
- Start subchannel failed
- disabled wait PSW 00020000 80000000 00000000 00004502
-Problem: The FBA loader has only a limited amount of memory to build CCW
- requests. Therefore larger I/O requests need to be split.
- This splitting was off by one leading to the fact that one CCW
- request uses memory of another data structure which in turn leads
- to corrupted data.
-Solution: Fix by correcting the split rule.
-Reproduction: IPL a FBA device.
- The error might occur randomly depending on the size of the
- kernel image and offsets within it.
-Upstream-ID: -
-Problem-ID: 143050
-
-Signed-off-by: Stefan Haberland
----
- zipl/boot/fba2.c | 2 +-
- 1 file changed, 1 insertion(+), 1 deletion(-)
-
---- a/zipl/boot/fba2.c
-+++ b/zipl/boot/fba2.c
-@@ -59,7 +59,7 @@ restart:
- memset(irb, 0, sizeof(struct irb));
- memset(&orb, 0, sizeof(struct orb));
-
-- if (blockptr->blockct > MAX_BLOCKCT) {
-+ if (blockptr->blockct >= MAX_BLOCKCT) {
- record_number = MAX_BLOCKCT - 1;
- blockptr->blockct -= MAX_BLOCKCT;
- } else {
diff --git a/s390-tools-sles12sp3-chmem-try-to-online-zone-movable.patch b/s390-tools-sles12sp3-chmem-try-to-online-zone-movable.patch
deleted file mode 100644
index 1c58336..0000000
--- a/s390-tools-sles12sp3-chmem-try-to-online-zone-movable.patch
+++ /dev/null
@@ -1,35 +0,0 @@
-Subject: [PATCH] [BZ 156627] chmem: try to online memory to zone movable
-From: Gerald Schaefer
-
-Description: chmem: try to online memory to zone movable
-Symptom: Standby/reserved memory that is onlined with chmem will be set
- online to zone normal, instead of zone movable, and so it is much
- less likely to be able to be offlined again.
-Problem: With recent kernels, the default zone for hotplug memory was
- changed from zone movable to zone normal.
-Solution: Let chmem first try to online memory to the zone movable, before
- setting it online to the default zone.
-Reproduction: Set standby/reserved memory online with chmem, and check
- /proc/zoneinfo for zone allocation.
-Upstream-ID: -
-Problem-ID: 156627
-
-Signed-off-by: Gerald Schaefer
----
- zconf/chmem | 5 ++++-
- 1 file changed, 4 insertions(+), 1 deletion(-)
-
---- a/zconf/chmem
-+++ b/zconf/chmem
-@@ -86,7 +86,10 @@ sub chmem_online($)
- {
- my $block = shift;
-
-- qx(echo online > $memdir/memory$block/state 2>/dev/null);
-+ qx(echo online_movable > $memdir/memory$block/state 2>/dev/null);
-+ if ($? >> 8 != 0) {
-+ qx(echo online > $memdir/memory$block/state 2>/dev/null);
-+ }
- return $? >> 8;
- }
-
diff --git a/s390-tools-sles12sp3-dasdfmt-01-Fix-behaviour-of-t-combined-with-y.patch b/s390-tools-sles12sp3-dasdfmt-01-Fix-behaviour-of-t-combined-with-y.patch
deleted file mode 100644
index f05316c..0000000
--- a/s390-tools-sles12sp3-dasdfmt-01-Fix-behaviour-of-t-combined-with-y.patch
+++ /dev/null
@@ -1,42 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: 9e49066f5c779e6591b06d2276add744ea8ca4d0
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Fix behaviour of -t combined with -y
-
- When -t (testmode) is being combined with -y (noprompt), the
- informations about what dasdfmt *would* do, is not being displayed.
-
- Simply fix this by checking whether testmode is set.
-
- Signed-off-by: Jan Höppner
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.c | 2 +-
- 1 file changed, 1 insertion(+), 1 deletion(-)
-
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -894,7 +894,7 @@ static void do_format_dasd(dasdfmt_info_
- vtoc_volume_label_set_label(vlabel, "LNX1");
- }
-
-- if ((info->verbosity > 0) || (!info->withoutprompt))
-+ if ((info->verbosity > 0) || !info->withoutprompt || info->testmode)
- dasdfmt_print_info(info, vlabel, cylinders, heads, p);
-
- count = u2s_get_host_access_count(info->devname);
diff --git a/s390-tools-sles12sp3-dasdfmt-02-Fix-trailing-whitespace.patch b/s390-tools-sles12sp3-dasdfmt-02-Fix-trailing-whitespace.patch
deleted file mode 100644
index 73f49ff..0000000
--- a/s390-tools-sles12sp3-dasdfmt-02-Fix-trailing-whitespace.patch
+++ /dev/null
@@ -1,250 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: -
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Fix trailing whitespace
-
- Signed-off-by: Jan Höppner
- Signed-off-by: Stefan Haberland
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.c | 58 +++++++++++++++++++++++++++---------------------------
- 1 file changed, 29 insertions(+), 29 deletions(-)
-
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -133,7 +133,7 @@ static void program_interrupt_signal (in
-
- rc = close(filedes);
- if (rc)
-- ERRMSG("%s: (signal handler) Unable to close device (%s)\n",
-+ ERRMSG("%s: (signal handler) Unable to close device (%s)\n",
- prog_name, strerror(errno));
-
- signal (sig, SIG_DFL);
-@@ -329,7 +329,7 @@ retry:
-
-
- /*
-- * check the volume serial for special
-+ * check the volume serial for special
- * characters and move blanks to the end
- */
- static int check_volser(char *s, int devno)
-@@ -345,7 +345,7 @@ static int check_volser(char *s, int dev
- s[i] = ' ';
- s[i] = toupper(s[i]);
- }
-- s[6] = 0x00;
-+ s[6] = 0x00;
-
- for (i=0; i<6; i++) {
- if (s[i] == ' ')
-@@ -416,16 +416,16 @@ static format_data_t ask_user_for_blksiz
- break;
-
- rc = sscanf(buffer,"%d%c", ¶ms.blksize, &c);
-- if ((rc == 2) && (c == '\n'))
-+ if ((rc == 2) && (c == '\n'))
- rc = 1;
-- if (rc == -1)
-+ if (rc == -1)
- rc = 1; /* this happens, if enter is pressed */
-- if (rc != 1)
-+ if (rc != 1)
- printf(" -- wrong input, try again.\n");
-
- if (check_param(str, ERR_LENGTH, ¶ms) < 0) {
-- printf(" -- %s\n",str);
-- rc = 0;
-+ printf(" -- %s\n",str);
-+ rc = 0;
- }
- } while (rc != 1);
-
-@@ -479,7 +479,7 @@ static int dasdfmt_get_volser(char * dev
- volume_label_t vlabel;
-
- if ((f = open(devname, O_RDONLY)) == -1)
-- ERRMSG_EXIT(EXIT_FAILURE,"%s: Unable to open device %s: %s\n",
-+ ERRMSG_EXIT(EXIT_FAILURE,"%s: Unable to open device %s: %s\n",
- prog_name, devname, strerror(errno));
-
- if (ioctl(f, BIODASDINFO, &dasd_info) != 0)
-@@ -491,7 +491,7 @@ static int dasdfmt_get_volser(char * dev
- "failed (%s).\n", prog_name, strerror(errno));
-
- if (close(f) != 0)
-- ERRMSG("%s: error during close: %s\ncontinuing...\n",
-+ ERRMSG("%s: error during close: %s\ncontinuing...\n",
- prog_name, strerror(errno));
-
- if ((strncmp(dasd_info.type, "ECKD", 4) == 0) &&
-@@ -505,7 +505,7 @@ static int dasdfmt_get_volser(char * dev
- return -1;
- }
- }
--
-+
- /*
- * do all the labeling (volume label and initial VTOC)
- */
-@@ -577,7 +577,7 @@ static void dasdfmt_write_labels(dasdfmt
- rc = lseek(filedes, label_position, SEEK_SET);
- if (rc != label_position)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: lseek command to %i failed "
-- "(%s).\n", prog_name, label_position,
-+ "(%s).\n", prog_name, label_position,
- strerror(errno));
-
- rc = write(filedes, ipl2_record, ipl2_record_len);
-@@ -633,9 +633,9 @@ static void dasdfmt_write_labels(dasdfmt
- rc = lseek(filedes, label_position, SEEK_SET);
- if (rc != label_position)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: lseek command to %i failed "
-- "(%s).\n", prog_name, label_position,
-+ "(%s).\n", prog_name, label_position,
- strerror(errno));
--
-+
- /* write VTOC FMT4 DSCB */
- rc = write(filedes, &f4, sizeof(format4_label_t));
- if (rc != sizeof(format4_label_t))
-@@ -695,21 +695,21 @@ static void dasdfmt_format(dasdfmt_info_
- "using the default.\n");
- info->hashstep = 10;
- }
--
-+
- if(!info->yast_mode) printf("Printing hashmark every %d cylinders.\n",
- info->hashstep);
- }
-
- format_step.blksize = format_params->blksize;
- format_step.intensity = format_params->intensity;
--
-+
- k = 0;
- cyl = 1;
- if ((info->print_progressbar || info->print_hashmarks) && !info->yast_mode)
- printf("\n");
-
- while (1) {
-- p1 = -1;
-+ p1 = -1;
- p2 = 0;
- if (k + heads * reqsize >= format_params->stop_unit)
- reqsize = 1;
-@@ -734,7 +734,7 @@ static void dasdfmt_format(dasdfmt_info_
- tmp = cyl*50/cylinders;
- for (j=1; j<=tmp; j++)
- printf("#");
-- for (j=tmp+1; j<=50; j++)
-+ for (j=tmp+1; j<=50; j++)
- printf("-");
- printf("|%3d%%", p1);
- }
-@@ -758,15 +758,15 @@ static void dasdfmt_format(dasdfmt_info_
- k += reqsize * heads;
- cyl += reqsize;
- }
-- else
-+ else
- k += format_params->stop_unit % heads;
-
-- if (k > format_params->stop_unit)
-+ if (k > format_params->stop_unit)
- break;
- }
-
- if ((info->print_progressbar || info->print_hashmarks) && !info->yast_mode)
-- printf("\n\n");
-+ printf("\n\n");
- }
-
-
-@@ -790,7 +790,7 @@ static void dasdfmt_prepare_and_format (
-
- if (ioctl(filedes, BIODASDDISABLE, p) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (prepare device) IOCTL "
-- "BIODASDDISABLE failed. (%s)\n", prog_name,
-+ "BIODASDDISABLE failed. (%s)\n", prog_name,
- strerror(errno));
- disk_disabled = 1;
-
-@@ -798,7 +798,7 @@ static void dasdfmt_prepare_and_format (
-
- if (ioctl(filedes, BIODASDFMT, &temp) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (invalidate first track) IOCTL "
-- "BIODASDFMT failed. (%s)\n", prog_name,
-+ "BIODASDFMT failed. (%s)\n", prog_name,
- strerror(errno));
-
- /* except track 0 from standard formatting procss */
-@@ -814,14 +814,14 @@ static void dasdfmt_prepare_and_format (
-
- if (ioctl(filedes, BIODASDFMT, &temp) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (re-validate first track) IOCTL"
-- " BIODASDFMT failed (%s)\n", prog_name,
-+ " BIODASDFMT failed (%s)\n", prog_name,
- strerror(errno));
-
- if (info->verbosity > 0) printf("Re-accessing the device...\n");
-
- if (ioctl(filedes, BIODASDENABLE, p) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (prepare device) IOCTL "
-- "BIODASDENABLE failed. (%s)\n", prog_name,
-+ "BIODASDENABLE failed. (%s)\n", prog_name,
- strerror(errno));
- disk_disabled = 0;
- }
-@@ -830,7 +830,7 @@ static void dasdfmt_prepare_and_format (
- /*
- *
- */
--static void do_format_dasd(dasdfmt_info_t *info, format_data_t *p,
-+static void do_format_dasd(dasdfmt_info_t *info, format_data_t *p,
- volume_label_t *vlabel)
- {
- char inp_buffer[5];
-@@ -943,7 +943,7 @@ static void do_format_dasd(dasdfmt_info_
- if (!info->yast_mode)
- printf("Finished formatting the device.\n");
-
-- if (!info->writenolabel)
-+ if (!info->writenolabel)
- dasdfmt_write_labels(info, vlabel, cylinders, heads);
-
- if (!info->yast_mode)
-@@ -959,7 +959,7 @@ static void do_format_dasd(dasdfmt_info_
- }
-
-
--int main(int argc,char *argv[])
-+int main(int argc,char *argv[])
- {
- dasdfmt_info_t info;
- volume_label_t vlabel;
-@@ -1001,7 +1001,7 @@ int main(int argc,char *argv[])
- rc=getopt_long(argc, argv, dasdfmt_getopt_string,
- dasdfmt_getopt_long_options, &index);
-
-- switch (rc)
-+ switch (rc)
- {
- case 'F':
- info.force=1;
diff --git a/s390-tools-sles12sp3-dasdfmt-03-Apply-coding-convention.patch b/s390-tools-sles12sp3-dasdfmt-03-Apply-coding-convention.patch
deleted file mode 100644
index bda753a..0000000
--- a/s390-tools-sles12sp3-dasdfmt-03-Apply-coding-convention.patch
+++ /dev/null
@@ -1,892 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: -
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Apply coding convention
-
- Improve the overall readability of the code by applying the Kernel
- coding convention.
-
- Signed-off-by: Jan Höppner
- Signed-off-by: Stefan Haberland
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.c | 362 +++++++++++++++++++++++++-----------------------------
- 1 file changed, 171 insertions(+), 191 deletions(-)
-
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -37,11 +37,10 @@ int reqsize;
- /*
- * Print version information.
- */
--static void
--print_version (void)
-+static void print_version(void)
- {
-- printf ("%s version %s\n", tool_name, RELEASE_STRING);
-- printf ("%s\n", copyright_notice);
-+ printf("%s version %s\n", tool_name, RELEASE_STRING);
-+ printf("%s\n", copyright_notice);
- }
-
- /*
-@@ -54,7 +53,7 @@ static void exit_usage(int exitcode)
- " [-b | --blocksize=]\n"
- " [-d | --disk_layout=]\n"
- " [-r | --requestsize=]\n"
-- " \n\n",prog_name);
-+ " \n\n", prog_name);
-
- printf(" -t or --test means testmode\n"
- " -V or --version means print version\n"
-@@ -71,13 +70,13 @@ static void exit_usage(int exitcode)
- " -C or --check_host_count means force dasdfmt to check\n"
- " the host access open count to ensure the device\n"
- " is not online on another operating system instance\n"
-- " --norecordzero prevent storage server from modifying"
-- " record 0\n\n"
-+ " --norecordzero prevent storage server from modifying"
-+ " record 0\n\n"
- " is the volume identifier, which is converted\n"
-- " to EBCDIC and written to disk. \n"
-+ " to EBCDIC and written to disk.\n"
- " (6 characters, e.g. LNX001\n"
- " has to be power of 2 and at least 512\n"
-- " is either \n"
-+ " is either\n"
- " 'cdl' for compatible disk layout (default) or\n"
- " 'ldl' for linux disk layout\n"
- " device node of the device to format\n");
-@@ -106,38 +105,39 @@ static int reread_partition_table(void)
- * signal handler:
- * enables the disk again in case of SIGTERM, SIGINT and SIGQUIT
- */
--static void program_interrupt_signal (int sig)
-+static void program_interrupt_signal(int sig)
- {
- int rc;
-
- if (program_interrupt_in_progress)
-- raise (sig);
-+ raise(sig);
- program_interrupt_in_progress = 1;
-
- if (disk_disabled) {
-- printf("Re-accessing the device... \n");
-+ printf("Re-accessing the device...\n");
- rc = ioctl(filedes, BIODASDENABLE, &format_params);
- if (rc)
-- ERRMSG_EXIT(EXIT_FAILURE,
-- "%s: (signal handler) IOCTL BIODASDENABLE "
-- "failed (%s)\n",prog_name,strerror(errno));
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: (signal handler) IOCTL "
-+ "BIODASDENABLE failed (%s)\n", prog_name,
-+ strerror(errno));
- }
-
-- printf("Rereading the partition table... \n");
-+ printf("Rereading the partition table...\n");
- rc = reread_partition_table();
- if (rc) {
- ERRMSG("%s: (signal handler) Re-reading partition table "
- "failed. (%s)\n", prog_name, strerror(errno));
-- } else
-+ } else {
- printf("Exiting...\n");
-+ }
-
- rc = close(filedes);
- if (rc)
- ERRMSG("%s: (signal handler) Unable to close device (%s)\n",
- prog_name, strerror(errno));
-
-- signal (sig, SIG_DFL);
-- raise (sig);
-+ signal(sig, SIG_DFL);
-+ raise(sig);
- }
-
-
-@@ -171,13 +171,13 @@ static void get_device_name(dasdfmt_info
- char* device;
- int i;
-
-- if ((strchr(name, ' ') != NULL)||(strchr(name, '#') != NULL)||
-- (strchr(name, '[') != NULL)||(strchr(name, ']') != NULL)||
-- (strchr(name, '!') != NULL)||(strchr(name, '>') != NULL)||
-- (strchr(name, '(') != NULL)||(strchr(name, '<') != NULL)||
-- (strchr(name, ')') != NULL)||(strchr(name, ':') != NULL)||
-- (strchr(name, '&') != NULL)||(strchr(name, ';') != NULL))
-- ERRMSG_EXIT(EXIT_MISUSE,"%s: Your filename contains "
-+ if ((strchr(name, ' ') != NULL) || (strchr(name, '#') != NULL) ||
-+ (strchr(name, '[') != NULL) || (strchr(name, ']') != NULL) ||
-+ (strchr(name, '!') != NULL) || (strchr(name, '>') != NULL) ||
-+ (strchr(name, '(') != NULL) || (strchr(name, '<') != NULL) ||
-+ (strchr(name, ')') != NULL) || (strchr(name, ':') != NULL) ||
-+ (strchr(name, '&') != NULL) || (strchr(name, ';') != NULL))
-+ ERRMSG_EXIT(EXIT_MISUSE, "%s: Your filename contains "
- "blanks or special characters!\n",
- prog_name);
-
-@@ -215,14 +215,14 @@ static void get_device_name(dasdfmt_info
- info->devname[PATH_MAX - 1] = '\0';
-
- if (stat(info->devname, &dev_stat) != 0)
-- ERRMSG_EXIT(EXIT_MISUSE,
-- "%s: Could not get information for device node %s: %s\n",
-- prog_name, info->devname, strerror(errno));
-+ ERRMSG_EXIT(EXIT_MISUSE, "%s: Could not get information for "
-+ "device node %s: %s\n", prog_name, info->devname,
-+ strerror(errno));
-
- if (minor(dev_stat.st_rdev) & PARTN_MASK) {
-- ERRMSG_EXIT(EXIT_MISUSE,
-- "%s: Unable to format partition %s. Please specify a device.\n",
-- prog_name, info->devname);
-+ ERRMSG_EXIT(EXIT_MISUSE, "%s: Unable to format partition %s. "
-+ "Please specify a device.\n", prog_name,
-+ info->devname);
- }
-
- if (util_proc_dev_get_entry(dev_stat.st_rdev, 1, &dev_entry) == 0) {
-@@ -253,7 +253,7 @@ static void init_info(dasdfmt_info_t *in
- info->force = 0;
- info->writenolabel = 0;
- info->labelspec = 0;
-- info->cdl_format = 0;
-+ info->cdl_format = 0;
- info->blksize_specified = 0;
- info->reqsize_specified = 0;
- info->node_specified = 0;
-@@ -263,7 +263,6 @@ static void init_info(dasdfmt_info_t *in
- info->yast_mode = 0;
- }
-
--
- /*
- * check for disk type and set some variables (e.g. usage count)
- */
-@@ -280,9 +279,9 @@ static void check_disk(dasdfmt_info_t *i
- "status information failed (%s)\n",
- prog_name, strerror(errno_save));
- }
-- if (ro) {
-+ if (ro)
- ERRMSG_EXIT(EXIT_FAILURE, "Disk is read only!\n");
-- }
-+
- retry:
- if (ioctl(filedes, BIODASDINFO, &dasd_info) != 0) {
- errno_save = errno;
-@@ -314,7 +313,7 @@ retry:
-
- info->usage_count = dasd_info.open_count;
- info->devno = dasd_info.devno;
-- if (strncmp(dasd_info.type, "ECKD",4) != 0) {
-+ if (strncmp(dasd_info.type, "ECKD", 4) != 0) {
- ERRMSG_EXIT(EXIT_FAILURE,
- "%s: Unsupported disk type\n%s is not an "
- "ECKD disk!\n", prog_name, info->devname);
-@@ -327,29 +326,28 @@ retry:
- }
- }
-
--
- /*
- * check the volume serial for special
- * characters and move blanks to the end
- */
- static int check_volser(char *s, int devno)
- {
-- int i,j;
-+ int i, j;
-
-- for (i=0; i<6; i++) {
-+ for (i = 0; i < 6; i++) {
- if ((s[i] < 0x20) || (s[i] > 0x7a) ||
-- ((s[i] >= 0x21)&&(s[i] <= 0x22)) || /* !" */
-- ((s[i] >= 0x26)&&(s[i] <= 0x2f)) || /* &'()*+,-./ */
-- ((s[i] >= 0x3a)&&(s[i] <= 0x3f)) || /* :;<=>? */
-- ((s[i] >= 0x5b)&&(s[i] <= 0x60))) /* \]^_` */
-+ ((s[i] >= 0x21) && (s[i] <= 0x22)) || /* !" */
-+ ((s[i] >= 0x26) && (s[i] <= 0x2f)) || /* &'()*+,-./ */
-+ ((s[i] >= 0x3a) && (s[i] <= 0x3f)) || /* :;<=>? */
-+ ((s[i] >= 0x5b) && (s[i] <= 0x60))) /* \]^_` */
- s[i] = ' ';
- s[i] = toupper(s[i]);
- }
- s[6] = 0x00;
-
-- for (i=0; i<6; i++) {
-+ for (i = 0; i < 6; i++) {
- if (s[i] == ' ')
-- for (j=i; j<6; j++)
-+ for (j = i; j < 6; j++)
- if (s[j] != ' ') {
- s[i] = s[j];
- s[j] = ' ';
-@@ -360,7 +358,7 @@ static int check_volser(char *s, int dev
- if (s[0] == ' ') {
- printf("Usage error, switching to default.\n");
- sprintf(s, "0X%04x", devno);
-- for (i=0; i<6; i++)
-+ for (i = 0; i < 6; i++)
- s[i] = toupper(s[i]);
- return -1;
- }
-@@ -368,7 +366,6 @@ static int check_volser(char *s, int dev
- return 0;
- }
-
--
- /*
- * do some blocksize checks
- */
-@@ -377,7 +374,7 @@ static int check_param(char *s, size_t b
- int tmp = data->blksize;
-
- if ((tmp < 512) || (tmp > 4096)) {
-- strncpy(s,"Blocksize must be one of the following positive "
-+ strncpy(s, "Blocksize must be one of the following positive "
- "integers:\n512, 1024, 2048, 4096.", buffsize);
- if (buffsize > 0)
- s[buffsize - 1] = '\0';
-@@ -386,7 +383,7 @@ static int check_param(char *s, size_t b
-
- while (tmp > 0) {
- if ((tmp % 2) && (tmp != 1)) {
-- strncpy(s,"Blocksize must be a power of 2.", buffsize);
-+ strncpy(s, "Blocksize must be a power of 2.", buffsize);
- if (buffsize > 0)
- s[buffsize - 1] = '\0';
- return -1;
-@@ -397,7 +394,6 @@ static int check_param(char *s, size_t b
- return 0;
- }
-
--
- /*
- * ask the user to specify a blocksize
- */
-@@ -415,7 +411,7 @@ static format_data_t ask_user_for_blksiz
- if (fgets(buffer, sizeof(buffer), stdin) == NULL)
- break;
-
-- rc = sscanf(buffer,"%d%c", ¶ms.blksize, &c);
-+ rc = sscanf(buffer, "%d%c", ¶ms.blksize, &c);
- if ((rc == 2) && (c == '\n'))
- rc = 1;
- if (rc == -1)
-@@ -424,7 +420,7 @@ static format_data_t ask_user_for_blksiz
- printf(" -- wrong input, try again.\n");
-
- if (check_param(str, ERR_LENGTH, ¶ms) < 0) {
-- printf(" -- %s\n",str);
-+ printf(" -- %s\n", str);
- rc = 0;
- }
- } while (rc != 1);
-@@ -432,7 +428,6 @@ static format_data_t ask_user_for_blksiz
- return params;
- }
-
--
- /*
- * print all information needed to format the device
- */
-@@ -449,7 +444,7 @@ static void dasdfmt_print_info(dasdfmt_i
- printf("%s in the following way:\n", info->devname);
- printf(" Device number of device : 0x%x\n", info->devno);
- printf(" Labelling device : %s\n",
-- (info->writenolabel)?"no":"yes");
-+ (info->writenolabel) ? "no" : "yes");
-
- if (!info->writenolabel) {
- vtoc_volume_label_get_label(vlabel, vollbl);
-@@ -460,31 +455,31 @@ static void dasdfmt_print_info(dasdfmt_i
- printf(" Extent start (trk no) : %u\n", p->start_unit);
- printf(" Extent end (trk no) : %u\n", p->stop_unit);
- printf(" Compatible Disk Layout : %s\n",
-- (p->intensity & DASD_FMT_INT_COMPAT)?"yes":"no");
-+ (p->intensity & DASD_FMT_INT_COMPAT) ? "yes" : "no");
- printf(" Blocksize : %d\n", p->blksize);
-
- if (info->testmode)
- printf("Test mode active, omitting ioctl.\n");
- }
-
--
- /*
- * get volser
- */
--static int dasdfmt_get_volser(char * devname, char * volser)
-+static int dasdfmt_get_volser(char *devname, char *volser)
- {
-- dasd_information_t dasd_info;
-+ dasd_information_t dasd_info;
- int blksize;
- int f;
- volume_label_t vlabel;
-
-- if ((f = open(devname, O_RDONLY)) == -1)
-- ERRMSG_EXIT(EXIT_FAILURE,"%s: Unable to open device %s: %s\n",
-+ f = open(devname, O_RDONLY);
-+ if (f == -1)
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: Unable to open device %s: %s\n",
- prog_name, devname, strerror(errno));
-
- if (ioctl(f, BIODASDINFO, &dasd_info) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (label pos) IOCTL BIODASD"
-- "INFO failed (%s).\n",prog_name, strerror(errno));
-+ "INFO failed (%s).\n", prog_name, strerror(errno));
-
- if (ioctl(f, BLKSSZGET, &blksize) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (label pos) IOCTL BLKSSZGET "
-@@ -500,8 +495,7 @@ static int dasdfmt_get_volser(char * dev
- vtoc_read_volume_label(devname, dasd_info.label_block * blksize, &vlabel);
- vtoc_volume_label_get_volser(&vlabel, volser);
- return 0;
-- }
-- else {
-+ } else {
- return -1;
- }
- }
-@@ -512,22 +506,24 @@ static int dasdfmt_get_volser(char * dev
- static void dasdfmt_write_labels(dasdfmt_info_t *info, volume_label_t *vlabel,
- unsigned int cylinders, unsigned int heads)
- {
-- int label_position;
-- dasd_information_t dasd_info;
-- struct hd_geometry geo;
-- format4_label_t f4;
-- format5_label_t f5;
-- format7_label_t f7;
-+ int label_position;
-+ dasd_information_t dasd_info;
-+ struct hd_geometry geo;
-+ format4_label_t f4;
-+ format5_label_t f5;
-+ format7_label_t f7;
- int rc, blksize;
- void *ipl1_record, *ipl2_record;
- int ipl1_record_len, ipl2_record_len;
-
-
-- if (info->verbosity > 0) printf("Retrieving dasd information... ");
-+ if (info->verbosity > 0)
-+ printf("Retrieving dasd information... ");
-
- if (ioctl(filedes, BIODASDINFO, &dasd_info) != 0)
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: (write labels) IOCTL BIODASD"
-- "INFO failed (%s).\n",prog_name, strerror(errno));
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: (write labels) IOCTL "
-+ "BIODASDINFO failed (%s).\n",
-+ prog_name, strerror(errno));
-
- if (ioctl(filedes, BLKSSZGET, &blksize) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (write labels) IOCTL BLKSSZGET "
-@@ -539,13 +535,16 @@ static void dasdfmt_write_labels(dasdfmt
- * vary depending on the format.
- */
- if (ioctl(filedes, HDIO_GETGEO, &geo) != 0)
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: (write labels) IOCTL HDIO_GET"
-- "GEO failed (%s).\n", prog_name, strerror(errno));
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: (write labels) IOCTL "
-+ "HDIO_GETGEO failed (%s).\n",
-+ prog_name, strerror(errno));
-
-- if (info->verbosity > 0) printf("ok\n");
-+ if (info->verbosity > 0)
-+ printf("ok\n");
-
- /* write empty bootstrap (initial IPL records) */
-- if (info->verbosity > 0) printf("Writing empty bootstrap...\n");
-+ if (info->verbosity > 0)
-+ printf("Writing empty bootstrap...\n");
-
- /*
- * Note: ldl labels do not contain the key field
-@@ -596,7 +595,8 @@ static void dasdfmt_write_labels(dasdfmt
-
- label_position = dasd_info.label_block * blksize;
-
-- if (info->verbosity > 0) printf("Writing label...\n");
-+ if (info->verbosity > 0)
-+ printf("Writing label...\n");
-
- rc = lseek(filedes, label_position, SEEK_SET);
- if (rc != label_position)
-@@ -608,10 +608,10 @@ static void dasdfmt_write_labels(dasdfmt
- * Note: cdl volume labels do not contain the 'formatted_blocks' part
- * and ldl labels do not contain the key field
- */
-- if (info->cdl_format)
-+ if (info->cdl_format) {
- rc = write(filedes, vlabel, (sizeof(*vlabel) -
- sizeof(vlabel->formatted_blocks)));
-- else {
-+ } else {
- vlabel->ldl_version = 0xf2; /* EBCDIC '2' */
- vlabel->formatted_blocks = cylinders * heads * geo.sectors;
- rc = write(filedes, &vlabel->vollbl, (sizeof(*vlabel)
-@@ -625,7 +625,8 @@ static void dasdfmt_write_labels(dasdfmt
- ERRMSG_EXIT(EXIT_FAILURE, "%s: Error writing volume label "
- "(%d).\n", prog_name, rc);
-
-- if (info->verbosity > 0) printf("Writing VTOC... ");
-+ if (info->verbosity > 0)
-+ printf("Writing VTOC... ");
-
- label_position = (VTOC_START_CC * heads + VTOC_START_HH) *
- geo.sectors * blksize;
-@@ -673,10 +674,10 @@ static void dasdfmt_write_labels(dasdfmt
-
- fsync(filedes);
-
-- if (info->verbosity > 0) printf("ok\n");
-+ if (info->verbosity > 0)
-+ printf("ok\n");
- }
-
--
- /*
- * formats the disk cylinderwise
- */
-@@ -720,21 +721,20 @@ static void dasdfmt_format(dasdfmt_info_
- format_step.start_unit += 1;
-
- if (ioctl(filedes, BIODASDFMT, &format_step) != 0)
-- ERRMSG_EXIT(EXIT_FAILURE,"%s: (format cylinder) IOCTL "
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: (format cylinder) IOCTL "
- "BIODASDFMT failed. (%s)\n",
- prog_name, strerror(errno));
-
- if (info->print_progressbar) {
- printf("cyl %7d of %7d |", cyl, cylinders);
- p2 = p1;
-- p1 = cyl*100/cylinders;
-- if (p1 != p2)
-- {
-+ p1 = cyl * 100 / cylinders;
-+ if (p1 != p2) {
- /* percent value has changed */
-- tmp = cyl*50/cylinders;
-- for (j=1; j<=tmp; j++)
-+ tmp = cyl * 50 / cylinders;
-+ for (j = 1; j <= tmp; j++)
- printf("#");
-- for (j=tmp+1; j<=50; j++)
-+ for (j = tmp + 1; j <= 50; j++)
- printf("-");
- printf("|%3d%%", p1);
- }
-@@ -748,6 +748,7 @@ static void dasdfmt_format(dasdfmt_info_
- fflush(stdout);
- hashcount++;
- }
-+
- if (info->print_percentage) {
- printf("cyl %7d of %7d |%3d%%\n", cyl, cylinders,
- cyl*100/cylinders);
-@@ -757,9 +758,9 @@ static void dasdfmt_format(dasdfmt_info_
- if (k % heads == 0) {
- k += reqsize * heads;
- cyl += reqsize;
-- }
-- else
-+ } else {
- k += format_params->stop_unit % heads;
-+ }
-
- if (k > format_params->stop_unit)
- break;
-@@ -769,14 +770,9 @@ static void dasdfmt_format(dasdfmt_info_
- printf("\n\n");
- }
-
--
--/*
-- *
-- */
--static void dasdfmt_prepare_and_format (dasdfmt_info_t *info,
-- unsigned int cylinders,
-- unsigned int heads,
-- format_data_t *p)
-+static void dasdfmt_prepare_and_format(dasdfmt_info_t *info,
-+ unsigned int cylinders,
-+ unsigned int heads, format_data_t *p)
- {
- format_data_t temp = {
- start_unit: 0,
-@@ -786,7 +782,8 @@ static void dasdfmt_prepare_and_format (
- | DASD_FMT_INT_INVAL)
- };
-
-- if (info->verbosity > 0) printf("Detaching the device...\n");
-+ if (info->verbosity > 0)
-+ printf("Detaching the device...\n");
-
- if (ioctl(filedes, BIODASDDISABLE, p) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (prepare device) IOCTL "
-@@ -794,7 +791,8 @@ static void dasdfmt_prepare_and_format (
- strerror(errno));
- disk_disabled = 1;
-
-- if (info->verbosity > 0) printf("Invalidate first track...\n");
-+ if (info->verbosity > 0)
-+ printf("Invalidate first track...\n");
-
- if (ioctl(filedes, BIODASDFMT, &temp) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (invalidate first track) IOCTL "
-@@ -806,18 +804,21 @@ static void dasdfmt_prepare_and_format (
-
- dasdfmt_format(info, cylinders, heads, p);
-
-- if (info->verbosity > 0) printf("formatting tracks complete...\n");
-+ if (info->verbosity > 0)
-+ printf("formatting tracks complete...\n");
-
- temp.intensity = p->intensity;
-
-- if (info->verbosity > 0) printf("Revalidate first track...\n");
-+ if (info->verbosity > 0)
-+ printf("Revalidate first track...\n");
-
- if (ioctl(filedes, BIODASDFMT, &temp) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (re-validate first track) IOCTL"
- " BIODASDFMT failed (%s)\n", prog_name,
- strerror(errno));
-
-- if (info->verbosity > 0) printf("Re-accessing the device...\n");
-+ if (info->verbosity > 0)
-+ printf("Re-accessing the device...\n");
-
- if (ioctl(filedes, BIODASDENABLE, p) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (prepare device) IOCTL "
-@@ -826,20 +827,17 @@ static void dasdfmt_prepare_and_format (
- disk_disabled = 0;
- }
-
--
--/*
-- *
-- */
- static void do_format_dasd(dasdfmt_info_t *info, format_data_t *p,
- volume_label_t *vlabel)
- {
-- char inp_buffer[5];
-+ char inp_buffer[5];
- dasd_information_t dasd_info;
- struct dasd_eckd_characteristics *characteristics;
- unsigned int cylinders, heads;
- int count;
-
-- if (info->verbosity > 0) printf("Retrieving disk geometry...\n");
-+ if (info->verbosity > 0)
-+ printf("Retrieving disk geometry...\n");
-
- if (ioctl(filedes, BIODASDINFO, &dasd_info) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (retrieving disk information) "
-@@ -877,22 +875,23 @@ static void do_format_dasd(dasdfmt_info_
- }
- }
- } else {
-- if (!info->labelspec && !info->keep_volser) {
-+ if (!info->labelspec && !info->keep_volser) {
- char buf[7];
-
-- sprintf(buf, "0X%04x", info->devno);
-+ sprintf(buf, "0X%04x", info->devno);
- check_volser(buf, info->devno);
- vtoc_volume_label_set_volser(vlabel, buf);
- }
-
-- if (p->intensity & DASD_FMT_INT_COMPAT) {
-+ if (p->intensity & DASD_FMT_INT_COMPAT) {
- info->cdl_format = 1;
- vtoc_volume_label_set_label(vlabel, "VOL1");
- vtoc_volume_label_set_key(vlabel, "VOL1");
- vtoc_set_cchhb(&vlabel->vtoc, 0x0000, 0x0001, 0x01);
-- } else
-+ } else {
- vtoc_volume_label_set_label(vlabel, "LNX1");
-- }
-+ }
-+ }
-
- if ((info->verbosity > 0) || !info->withoutprompt || info->testmode)
- dasdfmt_print_info(info, vlabel, cylinders, heads, p);
-@@ -926,15 +925,15 @@ static void do_format_dasd(dasdfmt_info_
- "untouched: ");
- if (fgets(inp_buffer, sizeof(inp_buffer), stdin) == NULL)
- return;
-- if (strcasecmp(inp_buffer,"yes") &&
-- strcasecmp(inp_buffer,"yes\n")) {
-+ if (strcasecmp(inp_buffer, "yes") &&
-+ strcasecmp(inp_buffer, "yes\n")) {
- printf("Omitting ioctl call (disk will "
- "NOT be formatted).\n");
- return;
- }
- }
-
-- if (!((info->withoutprompt)&&(info->verbosity<1)))
-+ if (!((info->withoutprompt) && (info->verbosity < 1)))
- printf("Formatting the device. This may take a "
- "while (get yourself a coffee).\n");
-
-@@ -954,12 +953,11 @@ static void do_format_dasd(dasdfmt_info_
- } else {
- if (!info->yast_mode)
- printf("ok\n");
-- }
-+ }
- }
- }
-
--
--int main(int argc,char *argv[])
-+int main(int argc, char *argv[])
- {
- dasdfmt_info_t info;
- volume_label_t vlabel;
-@@ -980,9 +978,9 @@ int main(int argc,char *argv[])
- int running=0;
-
- /* Establish a handler for interrupt signals. */
-- signal (SIGTERM, program_interrupt_signal);
-- signal (SIGINT, program_interrupt_signal);
-- signal (SIGQUIT, program_interrupt_signal);
-+ signal(SIGTERM, program_interrupt_signal);
-+ signal(SIGINT, program_interrupt_signal);
-+ signal(SIGQUIT, program_interrupt_signal);
-
- /******************* initialization ********************/
- prog_name = argv[0];
-@@ -996,102 +994,81 @@ int main(int argc,char *argv[])
-
- /*************** parse parameters **********************/
-
-- while (1)
-- {
-- rc=getopt_long(argc, argv, dasdfmt_getopt_string,
-- dasdfmt_getopt_long_options, &index);
-+ while (1) {
-+ rc = getopt_long(argc, argv, dasdfmt_getopt_string,
-+ dasdfmt_getopt_long_options, &index);
-
-- switch (rc)
-- {
-+ switch (rc) {
- case 'F':
-- info.force=1;
-+ info.force = 1;
- break;
--
-- case 'd' :
-- if (strncmp(optarg,"cdl",3)==0)
-- {
-+ case 'd':
-+ if (strncmp(optarg, "cdl", 3) == 0) {
- format_params.intensity |= DASD_FMT_INT_COMPAT;
-- if (info.writenolabel)
-- {
-- printf("WARNING: using the cdl " \
-- "format without writing a " \
-- "label doesn't make much " \
-+ if (info.writenolabel) {
-+ printf("WARNING: using the cdl "
-+ "format without writing a "
-+ "label doesn't make much "
- "sense!\n");
- exit(1);
- }
-- }
-- else if (strncmp(optarg,"ldl",3)==0)
-+ } else if (strncmp(optarg, "ldl", 3) == 0) {
- format_params.intensity &= ~DASD_FMT_INT_COMPAT;
-- else
-- {
-+ } else {
- printf("%s is not a valid option!\n", optarg);
- exit(1);
- }
-- break;
--
-+ break;
- case 'y':
-- info.withoutprompt=1;
-+ info.withoutprompt = 1;
- break;
--
- case 'z':
-- format_params.intensity |= DASD_FMT_INT_FMT_NOR0;
-+ format_params.intensity |= DASD_FMT_INT_FMT_NOR0;
- break;
--
- case 't':
-- info.testmode=1;
-+ info.testmode = 1;
- break;
--
- case 'p':
- if (!(info.print_hashmarks || info.print_percentage))
- info.print_progressbar = 1;
- break;
--
- case 'm':
-- if (!(info.print_progressbar || info.print_percentage))
-- {
-- hashstep_str=optarg;
-- info.print_hashmarks=1;
-+ if (!(info.print_progressbar || info.print_percentage)) {
-+ hashstep_str = optarg;
-+ info.print_hashmarks = 1;
- }
- break;
--
- case 'Q':
- if (!(info.print_hashmarks || info.print_progressbar))
- info.print_percentage = 1;
- break;
--
- case 'v':
-- info.verbosity=1;
-+ info.verbosity = 1;
- break;
--
- case 'h':
- exit_usage(0);
--
- case 'V':
- print_version();
- exit(0);
--
- case 'l':
-- strncpy(buf, optarg, 6);
-+ strncpy(buf, optarg, 6);
- if (check_volser(buf, 0) < 0)
- break;
-- vtoc_volume_label_set_volser(&vlabel,buf);
-- info.labelspec=1;
-+ vtoc_volume_label_set_volser(&vlabel, buf);
-+ info.labelspec = 1;
- break;
--
- case 'L':
-- if (format_params.intensity & DASD_FMT_INT_COMPAT)
-- {
-- printf("WARNING: using the cdl format " \
-- "without writing a label doesn't " \
-+ if (format_params.intensity & DASD_FMT_INT_COMPAT) {
-+ printf("WARNING: using the cdl format "
-+ "without writing a label doesn't "
- "make much sense!\n");
- exit(1);
- }
-- info.writenolabel=1;
-+ info.writenolabel = 1;
- break;
--
-- case 'b' :
-- blksize_param_str=optarg;
-- info.blksize_specified=1;
-+ case 'b':
-+ blksize_param_str = optarg;
-+ info.blksize_specified = 1;
- break;
- case 'r':
- reqsize_param_str = optarg;
-@@ -1102,7 +1079,7 @@ int main(int argc,char *argv[])
- ERRMSG_EXIT(EXIT_MISUSE,"%s: too many devices specified.\n",
- prog_name);
- dev_filename[dev_count++]=strdup(optarg);
-- info.node_specified=1;
-+ info.node_specified = 1;
- break;
- case 'Y' : /* YaST mode */
- info.yast_mode=1;
-@@ -1110,8 +1087,8 @@ int main(int argc,char *argv[])
- case 'P' : /* max parallel formatting processes */
- max_parallel=atoi(optarg);
- break;
-- case 'k' :
-- info.keep_volser=1;
-+ case 'k':
-+ info.keep_volser = 1;
- break;
- case 'C':
- info.force_host = 1;
-@@ -1121,11 +1098,12 @@ int main(int argc,char *argv[])
- info.device_id = optind;
- break;
- default:
-- ERRMSG_EXIT(EXIT_MISUSE,
-- "Try '%s --help' for more"
-- " information.\n",prog_name);
-+ ERRMSG_EXIT(EXIT_MISUSE, "Try '%s --help' for more"
-+ " information.\n", prog_name);
- }
-- if (rc==-1) break; // exit loop if finished
-+
-+ if (rc == -1)
-+ break; /* exit loop if finished */
- }
-
- CHECK_SPEC_MAX_ONCE(info.blksize_specified, "blocksize");
-@@ -1222,7 +1200,7 @@ int main(int argc,char *argv[])
- } else
- reqsize = DEFAULT_REQUESTSIZE;
- if (info.print_hashmarks)
-- PARSE_PARAM_INTO(info.hashstep, hashstep_str,10,"hashstep");
-+ PARSE_PARAM_INTO(info.hashstep, hashstep_str, 10, "hashstep");
-
- get_device_name(&info, dev_filename[i]);
-
-@@ -1230,22 +1208,24 @@ int main(int argc,char *argv[])
- format_params = ask_user_for_blksize(format_params);
-
- if (info.keep_volser) {
-- if(format_params.intensity == 0x00) {
-- printf("WARNING: VOLSER cannot be kept " \
-+ if (format_params.intensity == 0x00) {
-+ printf("WARNING: VOLSER cannot be kept "
- "when using the ldl format!\n");
- exit(1);
- }
-
-- if(dasdfmt_get_volser(info.devname, old_volser) == 0)
-+ if (dasdfmt_get_volser(info.devname, old_volser) == 0)
- vtoc_volume_label_set_volser(&vlabel, old_volser);
- else
-- ERRMSG_EXIT(EXIT_FAILURE,"%s: VOLSER not found on device %s\n",
-+ ERRMSG_EXIT(EXIT_FAILURE,
-+ "%s: VOLSER not found on device %s\n",
- prog_name, info.devname);
-
- }
-
-- if ((filedes = open(info.devname, O_RDWR)) == -1)
-- ERRMSG_EXIT(EXIT_FAILURE,"%s: Unable to open device %s: %s\n",
-+ filedes = open(info.devname, O_RDWR);
-+ if (filedes == -1)
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: Unable to open device %s: %s\n",
- prog_name, info.devname, strerror(errno));
-
- check_disk(&info);
diff --git a/s390-tools-sles12sp3-dasdfmt-04-Use-enhanced-DASD-information.patch b/s390-tools-sles12sp3-dasdfmt-04-Use-enhanced-DASD-information.patch
deleted file mode 100644
index 1116f55..0000000
--- a/s390-tools-sles12sp3-dasdfmt-04-Use-enhanced-DASD-information.patch
+++ /dev/null
@@ -1,399 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: -
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Use enhanced DASD information
-
- Currently dasdfmt uses BIODASDINFO as the IOCTL command to retrieve
- information about a DASD and stores them in dasd_information_t.
- However, BIODASDINFO2 offers a few more useful information (which are
- then stored in dasd_information2_t).
-
- - Replace dasd_information_t with dasd_information2_t.
- - Make the corresponding variable part of dasdfmt_info_t, since it is
- used in several functions anyway and we can save a few parameters and
- IOCTL calls.
- - Change the initialization of dasdfmt_info_t. A separate function isn't
- necessary for that.
- - Put the BIODASDINFO2 IOCTL call in a separate function which is then
- called only once.
-
- Signed-off-by: Jan Höppner
- Signed-off-by: Stefan Haberland
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.c | 132 ++++++++++++++++++------------------------------------
- dasdfmt/dasdfmt.h | 21 ++++++--
- 2 files changed, 61 insertions(+), 92 deletions(-)
-
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -237,30 +237,18 @@ static void get_device_name(dasdfmt_info
- }
-
- /*
-- * initialize the dasdfmt info structure
-+ * Retrieve DASD information
- */
--static void init_info(dasdfmt_info_t *info)
-+static void get_device_info(dasdfmt_info_t *info)
- {
-- info->devno = 0x0;
-- info->usage_count = 0;
-- info->testmode = 0;
-- info->verbosity = 0;
-- info->withoutprompt = 0;
-- info->print_progressbar = 0;
-- info->print_hashmarks = 0;
-- info->print_percentage = 0;
-- info->hashstep = 0;
-- info->force = 0;
-- info->writenolabel = 0;
-- info->labelspec = 0;
-- info->cdl_format = 0;
-- info->blksize_specified = 0;
-- info->reqsize_specified = 0;
-- info->node_specified = 0;
-- info->device_id = 0;
-- info->keep_volser = 0;
-- info->force_host = 0;
-- info->yast_mode = 0;
-+ dasd_information2_t dasd_info;
-+
-+ if (ioctl(filedes, BIODASDINFO2, &dasd_info))
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: the ioctl call to retrieve "
-+ "device information failed (%s).\n",
-+ prog_name, strerror(errno));
-+
-+ info->dasd_info = dasd_info;
- }
-
- /*
-@@ -268,7 +256,6 @@ static void init_info(dasdfmt_info_t *in
- */
- static void check_disk(dasdfmt_info_t *info)
- {
-- dasd_information_t dasd_info;
- int ro, errno_save, i = 0;
-
- if (ioctl(filedes, BLKROGET, &ro) != 0) {
-@@ -283,15 +270,6 @@ static void check_disk(dasdfmt_info_t *i
- ERRMSG_EXIT(EXIT_FAILURE, "Disk is read only!\n");
-
- retry:
-- if (ioctl(filedes, BIODASDINFO, &dasd_info) != 0) {
-- errno_save = errno;
-- close(filedes);
-- ERRMSG_EXIT(EXIT_FAILURE,
-- "%s: the ioctl call to retrieve device "
-- "information failed (%s)\n",
-- prog_name, strerror(errno_save));
-- }
--
- if (!info->force)
- /*
- * udev strikes again.
-@@ -302,7 +280,7 @@ retry:
- * And confusing the hell out ouf anyone else.
- * Bah.
- */
-- if (dasd_info.open_count > 1) {
-+ if (info->dasd_info.open_count > 1) {
- if (i < 5) {
- ++i;
- sleep(1);
-@@ -311,9 +289,7 @@ retry:
- ERRMSG_EXIT(EXIT_BUSY, "Disk in use!\n");
- }
-
-- info->usage_count = dasd_info.open_count;
-- info->devno = dasd_info.devno;
-- if (strncmp(dasd_info.type, "ECKD", 4) != 0) {
-+ if (strncmp(info->dasd_info.type, "ECKD", 4) != 0) {
- ERRMSG_EXIT(EXIT_FAILURE,
- "%s: Unsupported disk type\n%s is not an "
- "ECKD disk!\n", prog_name, info->devname);
-@@ -442,7 +418,7 @@ static void dasdfmt_print_info(dasdfmt_i
-
- printf("\nI am going to format the device ");
- printf("%s in the following way:\n", info->devname);
-- printf(" Device number of device : 0x%x\n", info->devno);
-+ printf(" Device number of device : 0x%x\n", info->dasd_info.devno);
- printf(" Labelling device : %s\n",
- (info->writenolabel) ? "no" : "yes");
-
-@@ -465,21 +441,16 @@ static void dasdfmt_print_info(dasdfmt_i
- /*
- * get volser
- */
--static int dasdfmt_get_volser(char *devname, char *volser)
-+static int dasdfmt_get_volser(dasdfmt_info_t *info, char *volser)
- {
-- dasd_information_t dasd_info;
- int blksize;
- int f;
- volume_label_t vlabel;
-
-- f = open(devname, O_RDONLY);
-+ f = open(info->devname, O_RDONLY);
- if (f == -1)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: Unable to open device %s: %s\n",
-- prog_name, devname, strerror(errno));
--
-- if (ioctl(f, BIODASDINFO, &dasd_info) != 0)
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: (label pos) IOCTL BIODASD"
-- "INFO failed (%s).\n", prog_name, strerror(errno));
-+ prog_name, info->devname, strerror(errno));
-
- if (ioctl(f, BLKSSZGET, &blksize) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (label pos) IOCTL BLKSSZGET "
-@@ -489,10 +460,12 @@ static int dasdfmt_get_volser(char *devn
- ERRMSG("%s: error during close: %s\ncontinuing...\n",
- prog_name, strerror(errno));
-
-- if ((strncmp(dasd_info.type, "ECKD", 4) == 0) &&
-- (!dasd_info.FBA_layout)) {
-+ if ((strncmp(info->dasd_info.type, "ECKD", 4) == 0) &&
-+ (!info->dasd_info.FBA_layout)) {
- /* OS/390 and zOS compatible disk layout */
-- vtoc_read_volume_label(devname, dasd_info.label_block * blksize, &vlabel);
-+ vtoc_read_volume_label(info->devname,
-+ info->dasd_info.label_block * blksize,
-+ &vlabel);
- vtoc_volume_label_get_volser(&vlabel, volser);
- return 0;
- } else {
-@@ -507,7 +480,6 @@ static void dasdfmt_write_labels(dasdfmt
- unsigned int cylinders, unsigned int heads)
- {
- int label_position;
-- dasd_information_t dasd_info;
- struct hd_geometry geo;
- format4_label_t f4;
- format5_label_t f5;
-@@ -520,11 +492,6 @@ static void dasdfmt_write_labels(dasdfmt
- if (info->verbosity > 0)
- printf("Retrieving dasd information... ");
-
-- if (ioctl(filedes, BIODASDINFO, &dasd_info) != 0)
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: (write labels) IOCTL "
-- "BIODASDINFO failed (%s).\n",
-- prog_name, strerror(errno));
--
- if (ioctl(filedes, BLKSSZGET, &blksize) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (write labels) IOCTL BLKSSZGET "
- "failed (%s).\n", prog_name, strerror(errno));
-@@ -586,14 +553,14 @@ static void dasdfmt_write_labels(dasdfmt
-
- /* write VTOC */
- vtoc_init_format4_label(&f4, geo.cylinders, cylinders, heads,
-- geo.sectors, blksize, dasd_info.dev_type);
-+ geo.sectors, blksize, info->dasd_info.dev_type);
-
- vtoc_init_format5_label(&f5);
- vtoc_init_format7_label(&f7);
- vtoc_set_freespace(&f4, &f5, &f7, '+', 0, FIRST_USABLE_TRK,
- (cylinders * heads - 1), cylinders, heads);
-
-- label_position = dasd_info.label_block * blksize;
-+ label_position = info->dasd_info.label_block * blksize;
-
- if (info->verbosity > 0)
- printf("Writing label...\n");
-@@ -831,7 +798,6 @@ static void do_format_dasd(dasdfmt_info_
- volume_label_t *vlabel)
- {
- char inp_buffer[5];
-- dasd_information_t dasd_info;
- struct dasd_eckd_characteristics *characteristics;
- unsigned int cylinders, heads;
- int count;
-@@ -839,13 +805,9 @@ static void do_format_dasd(dasdfmt_info_
- if (info->verbosity > 0)
- printf("Retrieving disk geometry...\n");
-
-- if (ioctl(filedes, BIODASDINFO, &dasd_info) != 0)
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: (retrieving disk information) "
-- "IOCTL BIODASDINFO failed (%s).\n",
-- prog_name, strerror(errno));
--
- characteristics =
-- (struct dasd_eckd_characteristics *) &dasd_info.characteristics;
-+ (struct dasd_eckd_characteristics *)
-+ &info->dasd_info.characteristics;
- if (characteristics->no_cyl == LV_COMPAT_CYL &&
- characteristics->long_no_cyl)
- cylinders = characteristics->long_no_cyl;
-@@ -878,8 +840,8 @@ static void do_format_dasd(dasdfmt_info_
- if (!info->labelspec && !info->keep_volser) {
- char buf[7];
-
-- sprintf(buf, "0X%04x", info->devno);
-- check_volser(buf, info->devno);
-+ sprintf(buf, "0X%04x", info->dasd_info.devno);
-+ check_volser(buf, info->dasd_info.devno);
- vtoc_volume_label_set_volser(vlabel, buf);
- }
-
-@@ -959,7 +921,10 @@ static void do_format_dasd(dasdfmt_info_
-
- int main(int argc, char *argv[])
- {
-- dasdfmt_info_t info;
-+ dasdfmt_info_t info = {
-+ .dasd_info = {0},
-+ {0}
-+ };
- volume_label_t vlabel;
- char old_volser[7];
-
-@@ -986,7 +951,6 @@ int main(int argc, char *argv[])
- prog_name = argv[0];
-
- /* set default values */
-- init_info(&info);
- vtoc_volume_label_init(&vlabel);
-
- format_params.blksize = DEFAULT_BLOCKSIZE;
-@@ -1137,30 +1101,24 @@ int main(int argc, char *argv[])
-
- if(info.yast_mode) {
- for(i=0;ino_cyl == LV_COMPAT_CYL &&
- characteristics->long_no_cyl)
- cylinders = characteristics->long_no_cyl;
-@@ -1204,6 +1162,13 @@ int main(int argc, char *argv[])
-
- get_device_name(&info, dev_filename[i]);
-
-+ filedes = open(info.devname, O_RDWR);
-+ if (filedes == -1)
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: Unable to open device %s: %s\n",
-+ prog_name, info.devname, strerror(errno));
-+
-+ get_device_info(&info);
-+
- if (!info.blksize_specified)
- format_params = ask_user_for_blksize(format_params);
-
-@@ -1214,7 +1179,7 @@ int main(int argc, char *argv[])
- exit(1);
- }
-
-- if (dasdfmt_get_volser(info.devname, old_volser) == 0)
-+ if (dasdfmt_get_volser(&info, old_volser) == 0)
- vtoc_volume_label_set_volser(&vlabel, old_volser);
- else
- ERRMSG_EXIT(EXIT_FAILURE,
-@@ -1223,11 +1188,6 @@ int main(int argc, char *argv[])
-
- }
-
-- filedes = open(info.devname, O_RDWR);
-- if (filedes == -1)
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: Unable to open device %s: %s\n",
-- prog_name, info.devname, strerror(errno));
--
- check_disk(&info);
-
- if (check_param(str, ERR_LENGTH, &format_params) < 0)
---- a/dasdfmt/dasdfmt.h
-+++ b/dasdfmt/dasdfmt.h
-@@ -32,11 +32,11 @@
- #define DASD_IOCTL_LETTER 'D'
-
- /*
-- * struct dasd_information_t
-+ * struct dasd_information2_t
- * represents any data about the device, which is visible to userspace.
- * including format and features.
- */
--typedef struct dasd_information_t {
-+typedef struct dasd_information2_t {
- unsigned int devno; /* S/390 devno */
- unsigned int real_devno; /* for aliases */
- unsigned int schid; /* S/390 subchannel identifier */
-@@ -56,7 +56,17 @@ typedef struct dasd_information_t {
- unsigned int confdata_size;
- char characteristics[64]; /* from read_device_characteristics */
- char configuration_data[256]; /* from read_configuration_data */
--} dasd_information_t;
-+ unsigned int format; /* format info like formatted/cdl/ldl/... */
-+ unsigned int features; /* dasd features like 'ro',... */
-+ unsigned int reserved0; /* reserved for further use ,... */
-+ unsigned int reserved1; /* reserved for further use ,... */
-+ unsigned int reserved2; /* reserved for further use ,... */
-+ unsigned int reserved3; /* reserved for further use ,... */
-+ unsigned int reserved4; /* reserved for further use ,... */
-+ unsigned int reserved5; /* reserved for further use ,... */
-+ unsigned int reserved6; /* reserved for further use ,... */
-+ unsigned int reserved7; /* reserved for further use ,... */
-+} dasd_information2_t;
-
-
- struct dasd_eckd_characteristics {
-@@ -165,7 +175,7 @@ typedef struct format_data_t {
- #define BIODASDENABLE _IO(DASD_IOCTL_LETTER,1)
-
- /* Get information on a dasd device (enhanced) */
--#define BIODASDINFO _IOR(DASD_IOCTL_LETTER,1,dasd_information_t)
-+#define BIODASDINFO2 _IOR(DASD_IOCTL_LETTER, 3, dasd_information2_t)
-
- /* #define BIODASDFORMAT _IOW(IOCTL_LETTER,0,format_data_t) , deprecated */
- #define BIODASDFMT _IOW(DASD_IOCTL_LETTER,1,format_data_t)
-@@ -251,9 +261,8 @@ typedef struct bootstrap2 {
- } __attribute__ ((packed)) bootstrap2_t;
-
- typedef struct dasdfmt_info {
-- int devno;
-+ dasd_information2_t dasd_info;
- char devname[PATH_MAX];
-- int usage_count;
- int verbosity;
- int testmode;
- int withoutprompt;
diff --git a/s390-tools-sles12sp3-dasdfmt-05-Refactor-do_format_dasd.patch b/s390-tools-sles12sp3-dasdfmt-05-Refactor-do_format_dasd.patch
deleted file mode 100644
index 46f6372..0000000
--- a/s390-tools-sles12sp3-dasdfmt-05-Refactor-do_format_dasd.patch
+++ /dev/null
@@ -1,210 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: -
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Refactor do_format_dasd
-
- In order to allow for new features to be integrated properly in the
- existing code, certain parts of do_format_dasd() must be moved to
- separate functions. This applies to setting VTOC information and
- geometrical information.
-
- Therefore, move the relevant bits to set_geo() and set_label().
-
- Signed-off-by: Jan Höppner
- Signed-off-by: Stefan Haberland
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.c | 132 +++++++++++++++++++++++++++++++-----------------------
- 1 file changed, 76 insertions(+), 56 deletions(-)
-
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -371,6 +371,73 @@ static int check_param(char *s, size_t b
- }
-
- /*
-+ * Retrieve disk information and set cylinders and heads accordingly.
-+ */
-+static void set_geo(dasdfmt_info_t *info, unsigned int *cylinders,
-+ unsigned int *heads)
-+{
-+ struct dasd_eckd_characteristics *characteristics;
-+
-+ if (info->verbosity > 0)
-+ printf("Retrieving disk geometry...\n");
-+
-+ characteristics = (struct dasd_eckd_characteristics *)
-+ &info->dasd_info.characteristics;
-+ if (characteristics->no_cyl == LV_COMPAT_CYL &&
-+ characteristics->long_no_cyl)
-+ *cylinders = characteristics->long_no_cyl;
-+ else
-+ *cylinders = characteristics->no_cyl;
-+ *heads = characteristics->trk_per_cyl;
-+}
-+
-+/*
-+ * Set VTOC label information
-+ */
-+static void set_label(dasdfmt_info_t *info, volume_label_t *vlabel,
-+ format_data_t *p, unsigned int cylinders)
-+{
-+ char inp_buffer[5];
-+
-+ if (info->writenolabel) {
-+ if (cylinders > LV_COMPAT_CYL && !info->withoutprompt) {
-+ printf("\n--->> ATTENTION! <<---\n");
-+ printf("You specified to write no labels to a"
-+ " volume with more then %u cylinders.\n"
-+ "Cylinders above this limit will not be"
-+ " accessible as a linux partition!\n"
-+ "Type \"yes\" to continue, no will leave"
-+ " the disk untouched: ", LV_COMPAT_CYL);
-+ if (fgets(inp_buffer, sizeof(inp_buffer), stdin) == NULL)
-+ return;
-+ if (strcasecmp(inp_buffer, "yes") &&
-+ strcasecmp(inp_buffer, "yes\n")) {
-+ printf("Omitting ioctl call (disk will "
-+ "NOT be formatted).\n");
-+ return;
-+ }
-+ }
-+ } else {
-+ if (!info->labelspec && !info->keep_volser) {
-+ char buf[7];
-+
-+ sprintf(buf, "0X%04x", info->dasd_info.devno);
-+ check_volser(buf, info->dasd_info.devno);
-+ vtoc_volume_label_set_volser(vlabel, buf);
-+ }
-+
-+ if (p->intensity & DASD_FMT_INT_COMPAT) {
-+ info->cdl_format = 1;
-+ vtoc_volume_label_set_label(vlabel, "VOL1");
-+ vtoc_volume_label_set_key(vlabel, "VOL1");
-+ vtoc_set_cchhb(&vlabel->vtoc, 0x0000, 0x0001, 0x01);
-+ } else {
-+ vtoc_volume_label_set_label(vlabel, "LNX1");
-+ }
-+ }
-+}
-+
-+/*
- * ask the user to specify a blocksize
- */
- static format_data_t ask_user_for_blksize(format_data_t params)
-@@ -452,7 +519,7 @@ static int dasdfmt_get_volser(dasdfmt_in
- ERRMSG_EXIT(EXIT_FAILURE, "%s: Unable to open device %s: %s\n",
- prog_name, info->devname, strerror(errno));
-
-- if (ioctl(f, BLKSSZGET, &blksize) != 0)
-+ if (ioctl(filedes, BLKSSZGET, &blksize) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (label pos) IOCTL BLKSSZGET "
- "failed (%s).\n", prog_name, strerror(errno));
-
-@@ -794,67 +861,16 @@ static void dasdfmt_prepare_and_format(d
- disk_disabled = 0;
- }
-
--static void do_format_dasd(dasdfmt_info_t *info, format_data_t *p,
-- volume_label_t *vlabel)
-+static void do_format_dasd(dasdfmt_info_t *info, volume_label_t *vlabel,
-+ format_data_t *p, unsigned int cylinders,
-+ unsigned int heads)
- {
- char inp_buffer[5];
-- struct dasd_eckd_characteristics *characteristics;
-- unsigned int cylinders, heads;
- int count;
-
-- if (info->verbosity > 0)
-- printf("Retrieving disk geometry...\n");
--
-- characteristics =
-- (struct dasd_eckd_characteristics *)
-- &info->dasd_info.characteristics;
-- if (characteristics->no_cyl == LV_COMPAT_CYL &&
-- characteristics->long_no_cyl)
-- cylinders = characteristics->long_no_cyl;
-- else
-- cylinders = characteristics->no_cyl;
-- heads = characteristics->trk_per_cyl;
--
- p->start_unit = 0;
- p->stop_unit = (cylinders * heads) - 1;
-
-- if (info->writenolabel) {
-- if (cylinders > LV_COMPAT_CYL && !info->withoutprompt) {
-- printf("\n--->> ATTENTION! <<---\n");
-- printf("You specified to write no labels to a"
-- " volume with more then %u cylinders.\n"
-- "Cylinders above this limit will not be"
-- " accessible as a linux partition!\n"
-- "Type \"yes\" to continue, no will leave"
-- " the disk untouched: ", LV_COMPAT_CYL);
-- if (fgets(inp_buffer, sizeof(inp_buffer), stdin) == NULL)
-- return;
-- if (strcasecmp(inp_buffer, "yes") &&
-- strcasecmp(inp_buffer, "yes\n")) {
-- printf("Omitting ioctl call (disk will "
-- "NOT be formatted).\n");
-- return;
-- }
-- }
-- } else {
-- if (!info->labelspec && !info->keep_volser) {
-- char buf[7];
--
-- sprintf(buf, "0X%04x", info->dasd_info.devno);
-- check_volser(buf, info->dasd_info.devno);
-- vtoc_volume_label_set_volser(vlabel, buf);
-- }
--
-- if (p->intensity & DASD_FMT_INT_COMPAT) {
-- info->cdl_format = 1;
-- vtoc_volume_label_set_label(vlabel, "VOL1");
-- vtoc_volume_label_set_key(vlabel, "VOL1");
-- vtoc_set_cchhb(&vlabel->vtoc, 0x0000, 0x0001, 0x01);
-- } else {
-- vtoc_volume_label_set_label(vlabel, "LNX1");
-- }
-- }
--
- if ((info->verbosity > 0) || !info->withoutprompt || info->testmode)
- dasdfmt_print_info(info, vlabel, cylinders, heads, p);
-
-@@ -938,6 +954,7 @@ int main(int argc, char *argv[])
- char *hashstep_str = NULL;
-
- int rc, index, i;
-+ unsigned int cylinders, heads;
-
- int max_parallel=1;
- int running=0;
-@@ -1193,7 +1210,10 @@ int main(int argc, char *argv[])
- if (check_param(str, ERR_LENGTH, &format_params) < 0)
- ERRMSG_EXIT(EXIT_MISUSE, "%s: %s\n", prog_name, str);
-
-- do_format_dasd(&info, &format_params, &vlabel);
-+ set_geo(&info, &cylinders, &heads);
-+ set_label(&info, &vlabel, &format_params, cylinders);
-+ do_format_dasd(&info, &vlabel, &format_params,
-+ cylinders, heads);
-
- if (close(filedes) != 0)
- ERRMSG("%s: error during close: %s\ncontinuing...\n",
diff --git a/s390-tools-sles12sp3-dasdfmt-06-Make-the-IOCTL-BLKSSZGET-reusable.patch b/s390-tools-sles12sp3-dasdfmt-06-Make-the-IOCTL-BLKSSZGET-reusable.patch
deleted file mode 100644
index b078fcf..0000000
--- a/s390-tools-sles12sp3-dasdfmt-06-Make-the-IOCTL-BLKSSZGET-reusable.patch
+++ /dev/null
@@ -1,93 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: -
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Make the IOCTL BLKSSZGET reusable
-
- The IOCTL BLKSSZGET is used to get blocksize information of a device.
- Since it is used several times already and could be used for new
- features as well, move the call to a separate function and make it
- reusable.
-
- Signed-off-by: Jan Höppner
- Signed-off-by: Stefan Haberland
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.c | 24 ++++++++++++++++--------
- 1 file changed, 16 insertions(+), 8 deletions(-)
-
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -252,6 +252,17 @@ static void get_device_info(dasdfmt_info
- }
-
- /*
-+ * Retrieve blocksize of device
-+ */
-+static void get_blocksize(unsigned int *blksize)
-+{
-+ if (ioctl(filedes, BLKSSZGET, blksize) != 0)
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: the ioctl to get the blocksize "
-+ "of the device failed (%s).\n", prog_name,
-+ strerror(errno));
-+}
-+
-+/*
- * check for disk type and set some variables (e.g. usage count)
- */
- static void check_disk(dasdfmt_info_t *info)
-@@ -510,7 +521,7 @@ static void dasdfmt_print_info(dasdfmt_i
- */
- static int dasdfmt_get_volser(dasdfmt_info_t *info, char *volser)
- {
-- int blksize;
-+ unsigned int blksize;
- int f;
- volume_label_t vlabel;
-
-@@ -519,9 +530,7 @@ static int dasdfmt_get_volser(dasdfmt_in
- ERRMSG_EXIT(EXIT_FAILURE, "%s: Unable to open device %s: %s\n",
- prog_name, info->devname, strerror(errno));
-
-- if (ioctl(filedes, BLKSSZGET, &blksize) != 0)
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: (label pos) IOCTL BLKSSZGET "
-- "failed (%s).\n", prog_name, strerror(errno));
-+ get_blocksize(&blksize);
-
- if (close(f) != 0)
- ERRMSG("%s: error during close: %s\ncontinuing...\n",
-@@ -551,7 +560,8 @@ static void dasdfmt_write_labels(dasdfmt
- format4_label_t f4;
- format5_label_t f5;
- format7_label_t f7;
-- int rc, blksize;
-+ unsigned int blksize;
-+ int rc;
- void *ipl1_record, *ipl2_record;
- int ipl1_record_len, ipl2_record_len;
-
-@@ -559,9 +569,7 @@ static void dasdfmt_write_labels(dasdfmt
- if (info->verbosity > 0)
- printf("Retrieving dasd information... ");
-
-- if (ioctl(filedes, BLKSSZGET, &blksize) != 0)
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: (write labels) IOCTL BLKSSZGET "
-- "failed (%s).\n", prog_name, strerror(errno));
-+ get_blocksize(&blksize);
-
- /*
- * Don't rely on the cylinders returned by HDIO_GETGEO, they might be
diff --git a/s390-tools-sles12sp3-dasdfmt-07-Add-quick-format-support.patch b/s390-tools-sles12sp3-dasdfmt-07-Add-quick-format-support.patch
deleted file mode 100644
index 555f4d2..0000000
--- a/s390-tools-sles12sp3-dasdfmt-07-Add-quick-format-support.patch
+++ /dev/null
@@ -1,555 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: -
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Add quick format support
-
- On occasion, a DASD device might have been formatted earlier and could
- potentially be re-initialized very easily. For device initalization it
- is necessary to write disk information (i.e. volume label (CDL or LDL)
- and VTOC) to the first two tracks. Currently you'd need to format the
- entire disk, which can take a long time and is totally unnecessary.
-
- Therefore, add a quick format mode which formats only the first two
- tracks and fills them with data accordingly.
- Before any formatting is done, several checks regarding the
- expected device format are performed.
-
- The mode can be specified with the new command line switch -M (--mode)
- . Where 'mode' can be either 'full' (default) or 'quick'.
-
- Document the switch -M (--mode) and its options full and quick in the
- man page.
-
- Signed-off-by: Jan Höppner
- Signed-off-by: Stefan Haberland
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.8 | 15 ++
- dasdfmt/dasdfmt.c | 290 ++++++++++++++++++++++++++++++++++++++++++++++++++++--
- dasdfmt/dasdfmt.h | 51 +++++++++
- 3 files changed, 344 insertions(+), 12 deletions(-)
-
---- a/dasdfmt/dasdfmt.8
-+++ b/dasdfmt/dasdfmt.8
-@@ -67,7 +67,7 @@ Print version number and exit.
-
- .TP
- \fB-F\fR or \fB--force\fR
--Formats the device without checking, if the device is in use.
-+Formats the device without performing sanity checking.
-
- .TP
- \fB-C\fR or \fB--check_host_count\fR
-@@ -108,6 +108,19 @@ The value will be at least as big as the
- .br
-
- .TP
-+\fB-M\fR \fImode\fR or \fB--mode\fR=\fImode\fR
-+Specify the \fImode\fR to be used to format the device. Valid modes are:
-+.RS
-+.IP full
-+Format the entire disk with the specified blocksize. (default)
-+.IP quick
-+Format the first two tracks and write label and partition information. Only use
-+this option if you are sure that the target DASD already contains a regular
-+format with the specified blocksize. A blocksize can optionally be specified
-+using \fB-b\fR (\fB--blocksize\fR).
-+.RE
-+
-+.TP
- \fB-r\fR \fIcylindercount\fR or \fB--requestsize\fR=\fIcylindercount\fR
- Number of cylinders to be processed in one formatting step.
- The value must be an integer in the range 1 - 255.
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -30,6 +30,7 @@ static const char copyright_notice[] = "
- static int filedes;
- static int disk_disabled;
- static format_data_t format_params;
-+static format_mode_t mode;
- char *prog_name;
- volatile sig_atomic_t program_interrupt_in_progress;
- int reqsize;
-@@ -53,6 +54,7 @@ static void exit_usage(int exitcode)
- " [-b | --blocksize=]\n"
- " [-d | --disk_layout=]\n"
- " [-r | --requestsize=]\n"
-+ " [-M | --mode=]\n"
- " \n\n", prog_name);
-
- printf(" -t or --test means testmode\n"
-@@ -65,7 +67,7 @@ static void exit_usage(int exitcode)
- " -r x or --requestsize=x means use x cylinders in one "
- "format step\n"
- " -v means verbose mode\n"
-- " -F means don't check if the device is in use\n"
-+ " -F means format without performing sanity checking\n"
- " -k means keep volume serial\n"
- " -C or --check_host_count means force dasdfmt to check\n"
- " the host access open count to ensure the device\n"
-@@ -79,7 +81,10 @@ static void exit_usage(int exitcode)
- " is either\n"
- " 'cdl' for compatible disk layout (default) or\n"
- " 'ldl' for linux disk layout\n"
-- " device node of the device to format\n");
-+ " device node of the device to format\n"
-+ " is either\n"
-+ " 'full' to fully format the device (default)\n"
-+ " 'quick' to format only the first two tracks\n");
- exit(exitcode);
- }
-
-@@ -102,6 +107,131 @@ static int reread_partition_table(void)
- }
-
- /*
-+ * Helper function for recs_per_track.
-+ */
-+static inline unsigned int ceil_quot(unsigned int d1, unsigned int d2)
-+{
-+ return (d1 + (d2 - 1)) / d2;
-+}
-+
-+/*
-+ * Calculate records per track depending on the device characteristics.
-+ */
-+static unsigned int recs_per_track(struct dasd_eckd_characteristics *rdc,
-+ unsigned int kl, unsigned int dl)
-+{
-+ int dn, kn;
-+
-+ switch (rdc->dev_type) {
-+ case 0x3380:
-+ if (kl)
-+ return 1499 / (15 + 7 + ceil_quot(kl + 12, 32) +
-+ ceil_quot(dl + 12, 32));
-+ else
-+ return 1499 / (15 + ceil_quot(dl + 12, 32));
-+ case 0x3390:
-+ dn = ceil_quot(dl + 6, 232) + 1;
-+ if (kl) {
-+ kn = ceil_quot(kl + 6, 232) + 1;
-+ return 1729 / (10 + 9 + ceil_quot(kl + 6 * kn, 34) +
-+ 9 + ceil_quot(dl + 6 * dn, 34));
-+ } else
-+ return 1729 / (10 + 9 + ceil_quot(dl + 6 * dn, 34));
-+ case 0x9345:
-+ dn = ceil_quot(dl + 6, 232) + 1;
-+ if (kl) {
-+ kn = ceil_quot(kl + 6, 232) + 1;
-+ return 1420 / (18 + 7 + ceil_quot(kl + 6 * kn, 34) +
-+ ceil_quot(dl + 6 * dn, 34));
-+ } else
-+ return 1420 / (18 + 7 + ceil_quot(dl + 6 * dn, 34));
-+ }
-+ return 0;
-+}
-+
-+/*
-+ * Evaluate errors recognized by format checks and print appropriate error
-+ * messages depending on the content of cdata.
-+ */
-+static void evaluate_format_error(dasdfmt_info_t *info, format_check_t *cdata,
-+ unsigned int heads)
-+{
-+ struct dasd_eckd_characteristics *rdc;
-+ /* Special blocksize values of the first 3 records of trk 0 of cyl 0 */
-+ const int blksizes_trk0[] = { 24, 144, 80 };
-+ /* Special blocksize value of trk 1 cyl 0 */
-+ const int blksize_trk1 = 96;
-+ unsigned int cyl;
-+ unsigned int head;
-+ unsigned int rpt;
-+ unsigned int kl = 0;
-+ int blksize = cdata->expect.blksize;
-+
-+ /*
-+ * Reading record zero will never happen. If the record in error is 0
-+ * nonetheless, the device is not formatted at all!
-+ */
-+ if (cdata->rec == 0) {
-+ ERRMSG("WARNING: The specified device is not "
-+ "formatted at all.\n");
-+ return;
-+ }
-+
-+ cyl = cdata->unit / heads;
-+ head = cyl >> 16;
-+ head <<= 4;
-+ head |= cdata->unit % heads;
-+
-+ /*
-+ * Set special expected values for the first 3 records of trk 0 of cyl 0
-+ * and trk 1 of cyl 0 when checking a CDL formatted device.
-+ */
-+ if ((cdata->expect.intensity & DASD_FMT_INT_COMPAT) &&
-+ cyl == 0 && head == 0 && cdata->rec < 4) {
-+ kl = 4;
-+ blksize = blksizes_trk0[cdata->rec - 1];
-+ }
-+ if ((cdata->expect.intensity & DASD_FMT_INT_COMPAT) &&
-+ cyl == 0 && head == 1) {
-+ kl = 44;
-+ blksize = blksize_trk1;
-+ }
-+
-+ rdc = (struct dasd_eckd_characteristics *)
-+ &info->dasd_info.characteristics;
-+
-+ rpt = recs_per_track(rdc, kl, cdata->expect.blksize);
-+
-+ ERRMSG("WARNING: The specified device is not formatted as expected.\n");
-+ switch (cdata->result) {
-+ case DASD_FMT_ERR_TOO_FEW_RECORDS:
-+ ERRMSG("Too few records (found %d, expected %d) "
-+ "at cyl: %d trk: %d rec: %d.\n",
-+ cdata->num_records, rpt, cyl, head, cdata->rec);
-+ break;
-+ case DASD_FMT_ERR_TOO_MANY_RECORDS:
-+ ERRMSG("Too many records (found %d, expected %d) "
-+ "at cyl: %d trk: %d rec: %d.\n",
-+ cdata->num_records, rpt, cyl, head, cdata->rec);
-+ break;
-+ case DASD_FMT_ERR_BLKSIZE:
-+ ERRMSG("Invalid blocksize (found %d, expected %d) "
-+ "at cyl: %d trk: %d rec: %d.\n",
-+ cdata->blksize, blksize, cyl, head, cdata->rec);
-+ break;
-+ case DASD_FMT_ERR_RECORD_ID:
-+ ERRMSG("Invalid record ID at cyl: %d trk: %d rec: %d.\n",
-+ cyl, head, cdata->rec);
-+ break;
-+ case DASD_FMT_ERR_KEY_LENGTH:
-+ ERRMSG("Invalid key length (found %d, expected %d) "
-+ "at cyl: %d trk: %d rec: %d.\n",
-+ cdata->key_length, kl, cyl, head, cdata->rec);
-+ break;
-+ }
-+}
-+
-+/*
- * signal handler:
- * enables the disk again in case of SIGTERM, SIGINT and SIGQUIT
- */
-@@ -263,6 +393,25 @@ static void get_blocksize(unsigned int *
- }
-
- /*
-+ * Check whether a specified blocksize matches the blocksize of the device
-+ */
-+static void check_blocksize(dasdfmt_info_t *info, unsigned int blksize)
-+{
-+ unsigned int dev_blksize;
-+
-+ if (!info->blksize_specified ||
-+ info->dasd_info.format == DASD_FORMAT_NONE)
-+ return;
-+
-+ get_blocksize(&dev_blksize);
-+ if (dev_blksize != blksize) {
-+ ERRMSG_EXIT(EXIT_FAILURE, "WARNING: Device is formatted with a "
-+ "different blocksize (%d).\nUse --mode=full to "
-+ "perform a clean format.\n", dev_blksize);
-+ }
-+}
-+
-+/*
- * check for disk type and set some variables (e.g. usage count)
- */
- static void check_disk(dasdfmt_info_t *info)
-@@ -449,6 +598,35 @@ static void set_label(dasdfmt_info_t *in
- }
-
- /*
-+ * This function checks whether a range of tracks is in regular format
-+ * with the specified block size.
-+ */
-+static format_check_t check_track_format(format_data_t *p)
-+{
-+ format_check_t cdata = {
-+ .expect = {
-+ .blksize = p->blksize,
-+ .intensity = p->intensity,
-+ .start_unit = p->start_unit,
-+ .stop_unit = p->stop_unit
-+ }, 0
-+ };
-+
-+ if (ioctl(filedes, BIODASDCHECKFMT, &cdata)) {
-+ if (errno == ENOTTY) {
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: Missing kernel support "
-+ "for format checking (--force to "
-+ "override)\n", prog_name);
-+ }
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: Could no check format: %s\n",
-+ prog_name, strerror(errno));
-+ }
-+
-+ return cdata;
-+}
-+
-+
-+/*
- * ask the user to specify a blocksize
- */
- static format_data_t ask_user_for_blksize(format_data_t params)
-@@ -511,6 +689,8 @@ static void dasdfmt_print_info(dasdfmt_i
- printf(" Compatible Disk Layout : %s\n",
- (p->intensity & DASD_FMT_INT_COMPAT) ? "yes" : "no");
- printf(" Blocksize : %d\n", p->blksize);
-+ printf(" Mode : %s\n",
-+ (mode == FULL) ? "Full" : "Quick");
-
- if (info->testmode)
- printf("Test mode active, omitting ioctl.\n");
-@@ -869,6 +1049,62 @@ static void dasdfmt_prepare_and_format(d
- disk_disabled = 0;
- }
-
-+/*
-+ * This function will only format the first two tracks of a DASD.
-+ * The rest of the DASD is untouched and left as is.
-+ */
-+static void dasdfmt_quick_format(dasdfmt_info_t *info, unsigned int cylinders,
-+ unsigned int heads, format_data_t *p)
-+{
-+ format_check_t cdata = { .expect = {0}, 0 };
-+ format_data_t tmp = *p;
-+
-+ if (info->force) {
-+ printf("Skipping format check due to --force.\n");
-+ } else {
-+ check_blocksize(info, p->blksize);
-+
-+ printf("Checking the format of selected tracks...\n");
-+
-+ /* Check device format on the first and last 3 regular tracks */
-+ tmp.start_unit = 2;
-+ tmp.stop_unit = 4;
-+ cdata = check_track_format(&tmp);
-+ if (!cdata.result) {
-+ tmp.start_unit = (cylinders * heads) - 3;
-+ tmp.stop_unit = (cylinders * heads) - 1;
-+ cdata = check_track_format(&tmp);
-+ }
-+ if (cdata.result) {
-+ evaluate_format_error(info, &cdata, heads);
-+ ERRMSG_EXIT(EXIT_FAILURE, "Use --mode=full to perform "
-+ "a clean format.\n");
-+ } else {
-+ printf("Done. Disk seems fine.\n");
-+ }
-+ }
-+
-+ if (!((info->withoutprompt) && (info->verbosity < 1)))
-+ printf("Formatting the first two tracks of the device.\n");
-+
-+ /* Disable the device before we do anything */
-+ if (ioctl(filedes, BIODASDDISABLE, p))
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: the ioctl to disable the device "
-+ "failed. (%s)\n", prog_name, strerror(errno));
-+ disk_disabled = 1;
-+
-+ /* Now do the actual formatting of our first two tracks */
-+ if (ioctl(filedes, BIODASDFMT, p))
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: the ioctl to format the device "
-+ "failed. (%s)\n", prog_name, strerror(errno));
-+
-+ /* Re-Enable the device so that we can continue working with it */
-+ if (ioctl(filedes, BIODASDENABLE, p))
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: the ioctl to enable the device "
-+ "failed. (%s)\n", prog_name, strerror(errno));
-+ disk_disabled = 0;
-+}
-+
- static void do_format_dasd(dasdfmt_info_t *info, volume_label_t *vlabel,
- format_data_t *p, unsigned int cylinders,
- unsigned int heads)
-@@ -877,7 +1113,15 @@ static void do_format_dasd(dasdfmt_info_
- int count;
-
- p->start_unit = 0;
-- p->stop_unit = (cylinders * heads) - 1;
-+
-+ switch (mode) {
-+ case FULL: /* all tracks */
-+ p->stop_unit = (cylinders * heads) - 1;
-+ break;
-+ case QUICK: /* just the first two */
-+ p->stop_unit = 1;
-+ break;
-+ }
-
- if ((info->verbosity > 0) || !info->withoutprompt || info->testmode)
- dasdfmt_print_info(info, vlabel, cylinders, heads, p);
-@@ -919,11 +1163,17 @@ static void do_format_dasd(dasdfmt_info_
- }
- }
-
-- if (!((info->withoutprompt) && (info->verbosity < 1)))
-- printf("Formatting the device. This may take a "
-- "while (get yourself a coffee).\n");
--
-- dasdfmt_prepare_and_format(info, cylinders, heads, p);
-+ switch (mode) {
-+ case FULL:
-+ if (!((info->withoutprompt) && (info->verbosity < 1)))
-+ printf("Formatting the device. This may take a "
-+ "while (get yourself a coffee).\n");
-+ dasdfmt_prepare_and_format(info, cylinders, heads, p);
-+ break;
-+ case QUICK:
-+ dasdfmt_quick_format(info, cylinders, heads, p);
-+ break;
-+ }
-
- if (!info->yast_mode)
- printf("Finished formatting the device.\n");
-@@ -981,6 +1231,8 @@ int main(int argc, char *argv[])
- format_params.blksize = DEFAULT_BLOCKSIZE;
- format_params.intensity = DASD_FMT_INT_COMPAT;
-
-+ mode = FULL;
-+
- /*************** parse parameters **********************/
-
- while (1) {
-@@ -1082,6 +1334,18 @@ int main(int argc, char *argv[])
- case 'C':
- info.force_host = 1;
- break;
-+ case 'M':
-+ if (strcasecmp(optarg, "full") == 0)
-+ mode = FULL;
-+ else if (strcasecmp(optarg, "quick") == 0)
-+ mode = QUICK;
-+ else
-+ ERRMSG_EXIT(EXIT_FAILURE,
-+ "%s: The specified mode '%s' is "
-+ "invalid. Consult the man page for "
-+ "more information.\n",
-+ prog_name, optarg);
-+ break;
- case -1:
- /* End of options string - start of devices list */
- info.device_id = optind;
-@@ -1194,8 +1458,14 @@ int main(int argc, char *argv[])
-
- get_device_info(&info);
-
-- if (!info.blksize_specified)
-- format_params = ask_user_for_blksize(format_params);
-+ /* Either let the user specify the blksize or get it from the kernel */
-+ if (!info.blksize_specified) {
-+ if (!(mode == FULL ||
-+ info.dasd_info.format == DASD_FORMAT_NONE))
-+ get_blocksize(&format_params.blksize);
-+ else
-+ format_params = ask_user_for_blksize(format_params);
-+ }
-
- if (info.keep_volser) {
- if (format_params.intensity == 0x00) {
---- a/dasdfmt/dasdfmt.h
-+++ b/dasdfmt/dasdfmt.h
-@@ -143,6 +143,14 @@ struct dasd_eckd_characteristics {
- unsigned int long_no_cyl;
- } __attribute__ ((packed));
-
-+/*
-+ * Represents possible format modes that can be specified when formatting
-+ * a DASD.
-+ */
-+typedef enum format_mode_t {
-+ FULL, /* default mode */
-+ QUICK, /* format only the first 2 tracks */
-+} format_mode_t;
-
- /*
- * struct format_data_t
-@@ -168,6 +176,43 @@ typedef struct format_data_t {
- #define DASD_FMT_INT_COMPAT 8 /* use OS/390 compatible disk layout */
- #define DASD_FMT_INT_FMT_NOR0 16 /* remove permission to write record zero */
-
-+/*
-+ * struct format_check_t
-+ * represents all data necessary to evaluate the format of
-+ * different tracks of a dasd
-+ */
-+typedef struct format_check_t {
-+ /* Input */
-+ struct format_data_t expect;
-+
-+ /* Output */
-+ unsigned int result; /* Error indication (DASD_FMT_ERR_*) */
-+ unsigned int unit; /* Track that is in error */
-+ unsigned int rec; /* Record that is in error */
-+ unsigned int num_records; /* Records in the track in error */
-+ unsigned int blksize; /* Block-size of first record in error */
-+ unsigned int key_length; /* Key length of first record in error */
-+} format_check_t;
-+
-+/*
-+ * values to be used in format_check_t for indicating
-+ * possible format errors
-+ */
-+#define DASD_FMT_ERR_TOO_FEW_RECORDS 1
-+#define DASD_FMT_ERR_TOO_MANY_RECORDS 2
-+#define DASD_FMT_ERR_BLKSIZE 3
-+#define DASD_FMT_ERR_RECORD_ID 4
-+#define DASD_FMT_ERR_KEY_LENGTH 5
-+
-+/*
-+ * values to be used for dasd_information2_t.format
-+ * 0x00: NOT formatted
-+ * 0x01: Linux disc layout
-+ * 0x02: Common disc layout
-+ */
-+#define DASD_FORMAT_NONE 0
-+#define DASD_FORMAT_LDL 1
-+#define DASD_FORMAT_CDL 2
-
- /* Disable the volume (for Linux) */
- #define BIODASDDISABLE _IO(DASD_IOCTL_LETTER,0)
-@@ -180,6 +225,9 @@ typedef struct format_data_t {
- /* #define BIODASDFORMAT _IOW(IOCTL_LETTER,0,format_data_t) , deprecated */
- #define BIODASDFMT _IOW(DASD_IOCTL_LETTER,1,format_data_t)
-
-+/* Check device format according to format_data_t */
-+#define BIODASDCHECKFMT _IOWR(DASD_IOCTL_LETTER, 2, format_check_t)
-+
- /****************************************************************************
- * SECTION: Further IOCTL Definitions (see fs.h and hdreq.h ) *
- ****************************************************************************/
-@@ -225,7 +273,7 @@ typedef struct format_data_t {
- if (*endptr) ERRMSG_EXIT(EXIT_MISUSE,"%s: " str " " \
- "is in invalid format\n",prog_name);}
-
--#define dasdfmt_getopt_string "b:n:l:f:d:m:r:hpQLtyvVFkCYP:"
-+#define dasdfmt_getopt_string "b:n:l:f:d:m:M:r:hpQLtyvVFkCYP:"
-
- static struct option dasdfmt_getopt_long_options[]=
- {
-@@ -247,6 +295,7 @@ static struct option dasdfmt_getopt_long
- { "keep_volser", 0, 0, 'k'},
- { "norecordzero", 0, 0, 'z'},
- { "check_host_count", 0, 0, 'C'},
-+ { "mode", 1, 0, 'M'},
- {0, 0, 0, 0}
- };
-
diff --git a/s390-tools-sles12sp3-dasdfmt-08-Make-progress-output-reusable-and-add-ETR.patch b/s390-tools-sles12sp3-dasdfmt-08-Make-progress-output-reusable-and-add-ETR.patch
deleted file mode 100644
index 0063641..0000000
--- a/s390-tools-sles12sp3-dasdfmt-08-Make-progress-output-reusable-and-add-ETR.patch
+++ /dev/null
@@ -1,279 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: -
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Make progress output reusable and add ETR
-
- The progress indicator (in form of a progressbar, hashmarks or
- percentages) is created within the formatting loop.
-
- Put the drawing of the progress into a separate function to make it
- reusable. Clean up the formatting loop while at it and make it
- more readable. Also, add the estimated time remaining to the output.
-
- Signed-off-by: Jan Höppner
- Signed-off-by: Stefan Haberland
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.c | 198 +++++++++++++++++++++++++++++++++++++-----------------
- 1 file changed, 138 insertions(+), 60 deletions(-)
-
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -11,6 +11,7 @@
-
- #include
- #include
-+#include
-
- #include "zt_common.h"
- #include "dasdfmt.h"
-@@ -20,6 +21,8 @@
- #include
-
- #define BUSIDSIZE 8
-+#define SEC_PER_DAY (60 * 60 * 24)
-+#define SEC_PER_HOUR (60 * 60)
-
- /* Full tool name */
- static const char tool_name[] = "dasdfmt: zSeries DASD format program";
-@@ -88,6 +91,120 @@ static void exit_usage(int exitcode)
- exit(exitcode);
- }
-
-+/*
-+ * Helper function to calculate the days, hours, minutes, and seconds
-+ * for a given timestamp in seconds
-+ */
-+static void calc_time(time_t time, int *d, int *h, int *m, int *s)
-+{
-+ *d = time / SEC_PER_DAY;
-+ time %= SEC_PER_DAY;
-+ *h = time / SEC_PER_HOUR;
-+ time %= SEC_PER_HOUR;
-+ *m = time / 60;
-+ *s = time % 60;
-+}
-+
-+/*
-+ * This function calculates and prints the estimated time remaining.
-+ */
-+static void print_etr(int p_new, int started)
-+{
-+ static struct timeval start;
-+ struct timeval now;
-+ time_t time_elapsed;
-+ time_t time_end;
-+ int d, h, m, s;
-+ static int p_init;
-+ int p;
-+
-+ if (!started) {
-+ gettimeofday(&start, NULL);
-+ p_init = p_new;
-+ }
-+ gettimeofday(&now, NULL);
-+ time_elapsed = now.tv_sec - start.tv_sec;
-+
-+ /*
-+ * We might start somewhere in the middle with an initial percentage
-+ * value of i.e. 60%. Therefore we need to calculate the relative
-+ * percentage of p_new within that remaining range (100 - 60) in order
-+ * to correctly estimate the remaining time.
-+ */
-+ if (p_init == 100)
-+ p = p_init;
-+ else
-+ p = 100 * (p_new - p_init) / (100 - p_init);
-+
-+ if (p == 0)
-+ time_end = time_elapsed;
-+ else
-+ time_end = time_elapsed * (100 - p) / p;
-+
-+ /* Calculate days, hours, minutes, and seconds */
-+ calc_time(time_end, &d, &h, &m, &s);
-+ if (p_new == 100)
-+ calc_time(time_elapsed, &d, &h, &m, &s);
-+
-+ /* Avoid printing leading zeros */
-+ if (d > 0)
-+ printf(" [%dd %dh %dm %ds%-4s", d, h, m, s, "]");
-+ else if (h > 0)
-+ printf(" [%dh %dm %ds%-7s", h, m, s, "]");
-+ else if (m > 0)
-+ printf(" [%dm %ds%-6s", m, s, "]");
-+ else if (s > 0 || p > 50)
-+ printf(" [%ds%-5s", s, "]");
-+ else
-+ printf(" [--%-1s", "]");
-+}
-+
-+/*
-+ * Draw the progress indicator depending on what command line argument is set.
-+ * This can either be a progressbar, hashmarks, or percentage.
-+ */
-+static void draw_progress(dasdfmt_info_t *info, int cyl, unsigned int cylinders)
-+{
-+ static int hashcount;
-+ static int started;
-+ static int p_old;
-+ int p_new = 0;
-+ int barlength;
-+ int i;
-+
-+ if (info->print_progressbar) {
-+ printf("cyl %7d of %7d |", cyl, cylinders);
-+ p_new = cyl * 100 / cylinders;
-+ if (p_new != p_old || !started) {
-+ /* percent value has changed */
-+ p_old = p_new;
-+ barlength = cyl * 33 / cylinders;
-+ for (i = 1; i <= barlength; i++)
-+ printf("#");
-+ for (i = barlength + 1; i <= 33; i++)
-+ printf("-");
-+ printf("|%3d%%", p_new);
-+ print_etr(p_new, started);
-+ started = 1;
-+ }
-+ printf("\r");
-+ fflush(stdout);
-+ }
-+
-+ if (info->print_hashmarks &&
-+ (cyl / info->hashstep - hashcount) != 0) {
-+ printf("%d|", info->procnum);
-+ fflush(stdout);
-+ hashcount++;
-+ }
-+
-+ if (info->print_percentage) {
-+ printf("cyl %7d of %7d |%3d%%\n", cyl, cylinders,
-+ cyl * 100 / cylinders);
-+ fflush(stdout);
-+ }
-+}
-+
- static int reread_partition_table(void)
- {
- int i = 0;
-@@ -906,9 +1023,10 @@ static void dasdfmt_write_labels(dasdfmt
- static void dasdfmt_format(dasdfmt_info_t *info, unsigned int cylinders,
- unsigned int heads, format_data_t *format_params)
- {
-- format_data_t format_step;
-- int j, cyl, tmp, p1, p2, hashcount = 0;
-- unsigned int k;
-+ unsigned int step_value;
-+ unsigned long cur_trk;
-+ format_data_t step;
-+ int cyl = 0;
-
- if (info->print_hashmarks) {
- if (info->hashstep < reqsize)
-@@ -923,73 +1041,33 @@ static void dasdfmt_format(dasdfmt_info_
- info->hashstep);
- }
-
-- format_step.blksize = format_params->blksize;
-- format_step.intensity = format_params->intensity;
--
-- k = 0;
-- cyl = 1;
-- if ((info->print_progressbar || info->print_hashmarks) && !info->yast_mode)
-- printf("\n");
--
-- while (1) {
-- p1 = -1;
-- p2 = 0;
-- if (k + heads * reqsize >= format_params->stop_unit)
-- reqsize = 1;
-- format_step.start_unit = k;
-- format_step.stop_unit = k + reqsize * heads - 1;
-+ step = *format_params;
-+ cur_trk = format_params->start_unit;
-
-- if (cyl == 1)
-- format_step.start_unit += 1;
-+ while (cur_trk < format_params->stop_unit) {
-+ step_value = reqsize * heads - (cur_trk % heads);
-+ step.start_unit = cur_trk;
-+ if (cur_trk + reqsize * heads >= format_params->stop_unit)
-+ step.stop_unit = format_params->stop_unit;
-+ else
-+ step.stop_unit = cur_trk + step_value - 1;
-
-- if (ioctl(filedes, BIODASDFMT, &format_step) != 0)
-+ if (ioctl(filedes, BIODASDFMT, &step) != 0)
- ERRMSG_EXIT(EXIT_FAILURE, "%s: (format cylinder) IOCTL "
- "BIODASDFMT failed. (%s)\n",
- prog_name, strerror(errno));
-
-- if (info->print_progressbar) {
-- printf("cyl %7d of %7d |", cyl, cylinders);
-- p2 = p1;
-- p1 = cyl * 100 / cylinders;
-- if (p1 != p2) {
-- /* percent value has changed */
-- tmp = cyl * 50 / cylinders;
-- for (j = 1; j <= tmp; j++)
-- printf("#");
-- for (j = tmp + 1; j <= 50; j++)
-- printf("-");
-- printf("|%3d%%", p1);
-- }
-- printf("\r");
-- fflush(stdout);
-- }
--
-- if (info->print_hashmarks)
-- if ((cyl / info->hashstep - hashcount) != 0) {
-- printf("%d|",info->procnum);
-- fflush(stdout);
-- hashcount++;
-- }
--
-- if (info->print_percentage) {
-- printf("cyl %7d of %7d |%3d%%\n", cyl, cylinders,
-- cyl*100/cylinders);
-- fflush(stdout);
-- }
-+ cyl = cur_trk / heads + 1;
-+ draw_progress(info, cyl, cylinders);
-
-- if (k % heads == 0) {
-- k += reqsize * heads;
-- cyl += reqsize;
-- } else {
-- k += format_params->stop_unit % heads;
-- }
--
-- if (k > format_params->stop_unit)
-- break;
-+ cur_trk += step_value;
- }
-+ /* We're done, draw the 100% mark */
-+ cyl = step.stop_unit / heads + 1;
-+ draw_progress(info, cyl, cylinders);
-
- if ((info->print_progressbar || info->print_hashmarks) && !info->yast_mode)
-- printf("\n\n");
-+ printf("\n");
- }
-
- static void dasdfmt_prepare_and_format(dasdfmt_info_t *info,
diff --git a/s390-tools-sles12sp3-dasdfmt-09-Add-command-line-argument-check.patch b/s390-tools-sles12sp3-dasdfmt-09-Add-command-line-argument-check.patch
deleted file mode 100644
index d06bab5..0000000
--- a/s390-tools-sles12sp3-dasdfmt-09-Add-command-line-argument-check.patch
+++ /dev/null
@@ -1,453 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: -
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Add command line argument --check
-
- Provide a function to check a DASD volume for correct formatting. This
- can be useful in cases where only a few tracks of a volume are
- incorrectly formatted.
-
- All tracks on the device are checked. In an error case, the check
- process is stopped and appropriate messages are displayed.
-
- The --check command line argument can be combined with either -P,
- -p or -m to display progress on the console.
-
- Signed-off-by: Jan Höppner
- Signed-off-by: Stefan Haberland
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.8 | 5 +
- dasdfmt/dasdfmt.c | 244 ++++++++++++++++++++++++++++++++++++++++--------------
- dasdfmt/dasdfmt.h | 4
- 3 files changed, 190 insertions(+), 63 deletions(-)
-
---- a/dasdfmt/dasdfmt.8
-+++ b/dasdfmt/dasdfmt.8
-@@ -121,6 +121,11 @@ using \fB-b\fR (\fB--blocksize\fR).
- .RE
-
- .TP
-+\fB--check\fR
-+Perform a complete format check on a DASD volume. A blocksize can be specified
-+with \fB-b\fR (\fB--blocksize\fR).
-+
-+.TP
- \fB-r\fR \fIcylindercount\fR or \fB--requestsize\fR=\fIcylindercount\fR
- Number of cylinders to be processed in one formatting step.
- The value must be an integer in the range 1 - 255.
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -76,7 +76,8 @@ static void exit_usage(int exitcode)
- " the host access open count to ensure the device\n"
- " is not online on another operating system instance\n"
- " --norecordzero prevent storage server from modifying"
-- " record 0\n\n"
-+ " record 0\n"
-+ " --check perform complete format check on device\n\n"
- " is the volume identifier, which is converted\n"
- " to EBCDIC and written to disk.\n"
- " (6 characters, e.g. LNX001\n"
-@@ -163,7 +164,8 @@ static void print_etr(int p_new, int sta
- * Draw the progress indicator depending on what command line argument is set.
- * This can either be a progressbar, hashmarks, or percentage.
- */
--static void draw_progress(dasdfmt_info_t *info, int cyl, unsigned int cylinders)
-+static void draw_progress(dasdfmt_info_t *info, int cyl, unsigned int cylinders,
-+ int aborted)
- {
- static int hashcount;
- static int started;
-@@ -175,7 +177,7 @@ static void draw_progress(dasdfmt_info_t
- if (info->print_progressbar) {
- printf("cyl %7d of %7d |", cyl, cylinders);
- p_new = cyl * 100 / cylinders;
-- if (p_new != p_old || !started) {
-+ if (p_new != p_old || !started || aborted) {
- /* percent value has changed */
- p_old = p_new;
- barlength = cyl * 33 / cylinders;
-@@ -184,6 +186,8 @@ static void draw_progress(dasdfmt_info_t
- for (i = barlength + 1; i <= 33; i++)
- printf("-");
- printf("|%3d%%", p_new);
-+ if (aborted)
-+ p_new = 100;
- print_etr(p_new, started);
- started = 1;
- }
-@@ -284,11 +288,17 @@ static void evaluate_format_error(dasdfm
- unsigned int kl = 0;
- int blksize = cdata->expect.blksize;
-
-+ if (info->print_progressbar || info->print_hashmarks)
-+ printf("\n");
-+
- /*
-- * Reading record zero will never happen. If the record in error is 0
-- * nonetheless, the device is not formatted at all!
-+ * If mode is not QUICK and a device format couldn't be determined, the
-+ * device is considered not formatted.
-+ * Also, reading record zero will never happen. If the record in error
-+ * is 0 nonetheless, the device is not formatted at all as well!
- */
-- if (cdata->rec == 0) {
-+ if ((info->dasd_info.format == DASD_FORMAT_NONE && mode != QUICK) ||
-+ cdata->rec == 0) {
- ERRMSG("WARNING: The specified device is not "
- "formatted at all.\n");
- return;
-@@ -529,6 +539,35 @@ static void check_blocksize(dasdfmt_info
- }
-
- /*
-+ * Check whether a specified layout matches the layout
-+ * a device is formatted with.
-+ */
-+static void check_layout(dasdfmt_info_t *info, unsigned int intensity)
-+{
-+ char layout[4];
-+
-+ if (!info->layout_specified ||
-+ info->dasd_info.format == DASD_FORMAT_NONE)
-+ return;
-+
-+ if ((intensity & DASD_FMT_INT_COMPAT) &&
-+ info->dasd_info.format == DASD_FORMAT_CDL)
-+ return;
-+
-+ if (!(intensity & DASD_FMT_INT_COMPAT) &&
-+ info->dasd_info.format == DASD_FORMAT_LDL)
-+ return;
-+
-+ if (info->dasd_info.format == DASD_FORMAT_CDL)
-+ sprintf(layout, "CDL");
-+ if (info->dasd_info.format == DASD_FORMAT_LDL)
-+ sprintf(layout, "LDL");
-+
-+ ERRMSG_EXIT(EXIT_FAILURE, "WARNING: Device is formatted with a "
-+ "different layout (%s).\n", layout);
-+}
-+
-+/*
- * check for disk type and set some variables (e.g. usage count)
- */
- static void check_disk(dasdfmt_info_t *info)
-@@ -715,10 +754,29 @@ static void set_label(dasdfmt_info_t *in
- }
-
- /*
-+ * Check whether hashsteps are within the correct interval.
-+ */
-+static void check_hashmarks(dasdfmt_info_t *info)
-+{
-+ if (info->print_hashmarks) {
-+ if (info->hashstep < reqsize)
-+ info->hashstep = reqsize;
-+ if ((info->hashstep < 1) || (info->hashstep > 1000)) {
-+ printf("Hashmark increment is not in range <1,1000>, "
-+ "using the default.\n");
-+ info->hashstep = 10;
-+ }
-+
-+ printf("Printing hashmark every %d cylinders.\n",
-+ info->hashstep);
-+ }
-+}
-+
-+/*
- * This function checks whether a range of tracks is in regular format
- * with the specified block size.
- */
--static format_check_t check_track_format(format_data_t *p)
-+static format_check_t check_track_format(dasdfmt_info_t *info, format_data_t *p)
- {
- format_check_t cdata = {
- .expect = {
-@@ -731,9 +789,11 @@ static format_check_t check_track_format
-
- if (ioctl(filedes, BIODASDCHECKFMT, &cdata)) {
- if (errno == ENOTTY) {
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: Missing kernel support "
-- "for format checking (--force to "
-- "override)\n", prog_name);
-+ ERRMSG("%s: Missing kernel support for format checking",
-+ prog_name);
-+ if (!info->check)
-+ ERRMSG(" (--force to override)");
-+ ERRMSG_EXIT(EXIT_FAILURE, ".\n");
- }
- ERRMSG_EXIT(EXIT_FAILURE, "%s: Could no check format: %s\n",
- prog_name, strerror(errno));
-@@ -742,6 +802,97 @@ static format_check_t check_track_format
- return cdata;
- }
-
-+/*
-+ * Either do the actual format or check depending on the check-value.
-+ */
-+static int process_tracks(dasdfmt_info_t *info, unsigned int cylinders,
-+ unsigned int heads, format_data_t *format_params)
-+{
-+ format_check_t cdata = { .expect = {0}, 0};
-+ format_data_t step = *format_params;
-+ unsigned long step_value;
-+ unsigned long cur_trk;
-+ int cyl = 0;
-+
-+ check_hashmarks(info);
-+
-+ cur_trk = format_params->start_unit;
-+
-+ while (cur_trk < format_params->stop_unit) {
-+ step_value = reqsize * heads - (cur_trk % heads);
-+ step.start_unit = cur_trk;
-+ if (cur_trk + heads * reqsize >= format_params->stop_unit)
-+ step.stop_unit = format_params->stop_unit;
-+ else
-+ step.stop_unit = cur_trk + step_value - 1;
-+
-+ if (info->check) {
-+ cdata = check_track_format(info, &step);
-+ if (cdata.result) {
-+ cyl = cur_trk / heads + 1;
-+ draw_progress(info, cyl, cylinders, 1);
-+ evaluate_format_error(info, &cdata, heads);
-+ break;
-+ }
-+ } else {
-+ if (ioctl(filedes, BIODASDFMT, &step) != 0)
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: the ioctl call "
-+ "to format tracks failed. (%s)\n",
-+ prog_name, strerror(errno));
-+ }
-+
-+ cyl = cur_trk / heads + 1;
-+ draw_progress(info, cyl, cylinders, 0);
-+
-+ cur_trk += step_value;
-+ }
-+ /* We're done, draw the 100% mark */
-+ if (!cdata.result) {
-+ cyl = step.stop_unit / heads + 1;
-+ draw_progress(info, cyl, cylinders, 0);
-+ printf("\n");
-+ }
-+
-+ return cdata.result;
-+}
-+
-+/*
-+ * This function checks the format of the entire disk.
-+ */
-+static void check_disk_format(dasdfmt_info_t *info, unsigned int cylinders,
-+ unsigned int heads, format_data_t *check_params)
-+{
-+ check_params->start_unit = 0;
-+ check_params->stop_unit = (cylinders * heads) - 1;
-+
-+ printf("Checking format of the entire disk...\n");
-+
-+ if (info->testmode) {
-+ printf("Test mode active, omitting ioctl.\n");
-+ return;
-+ }
-+
-+ check_blocksize(info, check_params->blksize);
-+ check_layout(info, check_params->intensity);
-+
-+ /*
-+ * If no layout was specified, set the intensity
-+ * according to what the layout seems to be.
-+ */
-+ if (!info->layout_specified) {
-+ if (info->dasd_info.format == DASD_FORMAT_CDL)
-+ check_params->intensity |= DASD_FMT_INT_COMPAT;
-+ else if (info->dasd_info.format == DASD_FORMAT_LDL)
-+ check_params->intensity &= ~DASD_FMT_INT_COMPAT;
-+ }
-+
-+ if (process_tracks(info, cylinders, heads, check_params)) {
-+ ERRMSG_EXIT(EXIT_FAILURE, "Use --mode=full to perform a "
-+ "clean format.\n");
-+ }
-+
-+ printf("Done. Disk is fine.\n");
-+}
-
- /*
- * ask the user to specify a blocksize
-@@ -1023,51 +1174,7 @@ static void dasdfmt_write_labels(dasdfmt
- static void dasdfmt_format(dasdfmt_info_t *info, unsigned int cylinders,
- unsigned int heads, format_data_t *format_params)
- {
-- unsigned int step_value;
-- unsigned long cur_trk;
-- format_data_t step;
-- int cyl = 0;
--
-- if (info->print_hashmarks) {
-- if (info->hashstep < reqsize)
-- info->hashstep = reqsize;
-- if ((info->hashstep < 1) || (info->hashstep > 1000)) {
-- printf("Hashmark increment is not in range <1,1000>, "
-- "using the default.\n");
-- info->hashstep = 10;
-- }
--
-- if(!info->yast_mode) printf("Printing hashmark every %d cylinders.\n",
-- info->hashstep);
-- }
--
-- step = *format_params;
-- cur_trk = format_params->start_unit;
--
-- while (cur_trk < format_params->stop_unit) {
-- step_value = reqsize * heads - (cur_trk % heads);
-- step.start_unit = cur_trk;
-- if (cur_trk + reqsize * heads >= format_params->stop_unit)
-- step.stop_unit = format_params->stop_unit;
-- else
-- step.stop_unit = cur_trk + step_value - 1;
--
-- if (ioctl(filedes, BIODASDFMT, &step) != 0)
-- ERRMSG_EXIT(EXIT_FAILURE, "%s: (format cylinder) IOCTL "
-- "BIODASDFMT failed. (%s)\n",
-- prog_name, strerror(errno));
--
-- cyl = cur_trk / heads + 1;
-- draw_progress(info, cyl, cylinders);
--
-- cur_trk += step_value;
-- }
-- /* We're done, draw the 100% mark */
-- cyl = step.stop_unit / heads + 1;
-- draw_progress(info, cyl, cylinders);
--
-- if ((info->print_progressbar || info->print_hashmarks) && !info->yast_mode)
-- printf("\n");
-+ process_tracks(info, cylinders, heads, format_params);
- }
-
- static void dasdfmt_prepare_and_format(dasdfmt_info_t *info,
-@@ -1082,6 +1189,10 @@ static void dasdfmt_prepare_and_format(d
- | DASD_FMT_INT_INVAL)
- };
-
-+ if (!((info->withoutprompt) && (info->verbosity < 1)))
-+ printf("Formatting the device. This may take a while "
-+ "(get yourself a coffee).\n");
-+
- if (info->verbosity > 0)
- printf("Detaching the device...\n");
-
-@@ -1147,11 +1258,11 @@ static void dasdfmt_quick_format(dasdfmt
- /* Check device format on the first and last 3 regular tracks */
- tmp.start_unit = 2;
- tmp.stop_unit = 4;
-- cdata = check_track_format(&tmp);
-+ cdata = check_track_format(info, &tmp);
- if (!cdata.result) {
- tmp.start_unit = (cylinders * heads) - 3;
- tmp.stop_unit = (cylinders * heads) - 1;
-- cdata = check_track_format(&tmp);
-+ cdata = check_track_format(info, &tmp);
- }
- if (cdata.result) {
- evaluate_format_error(info, &cdata, heads);
-@@ -1243,9 +1354,6 @@ static void do_format_dasd(dasdfmt_info_
-
- switch (mode) {
- case FULL:
-- if (!((info->withoutprompt) && (info->verbosity < 1)))
-- printf("Formatting the device. This may take a "
-- "while (get yourself a coffee).\n");
- dasdfmt_prepare_and_format(info, cylinders, heads, p);
- break;
- case QUICK:
-@@ -1337,6 +1445,7 @@ int main(int argc, char *argv[])
- printf("%s is not a valid option!\n", optarg);
- exit(1);
- }
-+ info.layout_specified = 1;
- break;
- case 'y':
- info.withoutprompt = 1;
-@@ -1424,6 +1533,9 @@ int main(int argc, char *argv[])
- "more information.\n",
- prog_name, optarg);
- break;
-+ case OPT_CHECK:
-+ info.check = 1;
-+ break;
- case -1:
- /* End of options string - start of devices list */
- info.device_id = optind;
-@@ -1539,7 +1651,7 @@ int main(int argc, char *argv[])
- /* Either let the user specify the blksize or get it from the kernel */
- if (!info.blksize_specified) {
- if (!(mode == FULL ||
-- info.dasd_info.format == DASD_FORMAT_NONE))
-+ info.dasd_info.format == DASD_FORMAT_NONE) || info.check)
- get_blocksize(&format_params.blksize);
- else
- format_params = ask_user_for_blksize(format_params);
-@@ -1568,8 +1680,14 @@ int main(int argc, char *argv[])
-
- set_geo(&info, &cylinders, &heads);
- set_label(&info, &vlabel, &format_params, cylinders);
-- do_format_dasd(&info, &vlabel, &format_params,
-- cylinders, heads);
-+
-+ if (info.check) {
-+ check_disk_format(&info, cylinders, heads,
-+ &format_params);
-+ } else {
-+ do_format_dasd(&info, &vlabel, &format_params,
-+ cylinders, heads);
-+ }
-
- if (close(filedes) != 0)
- ERRMSG("%s: error during close: %s\ncontinuing...\n",
---- a/dasdfmt/dasdfmt.h
-+++ b/dasdfmt/dasdfmt.h
-@@ -273,6 +273,7 @@ typedef struct format_check_t {
- if (*endptr) ERRMSG_EXIT(EXIT_MISUSE,"%s: " str " " \
- "is in invalid format\n",prog_name);}
-
-+#define OPT_CHECK 128
- #define dasdfmt_getopt_string "b:n:l:f:d:m:M:r:hpQLtyvVFkCYP:"
-
- static struct option dasdfmt_getopt_long_options[]=
-@@ -296,6 +297,7 @@ static struct option dasdfmt_getopt_long
- { "norecordzero", 0, 0, 'z'},
- { "check_host_count", 0, 0, 'C'},
- { "mode", 1, 0, 'M'},
-+ { "check", 0, 0, OPT_CHECK},
- {0, 0, 0, 0}
- };
-
-@@ -328,6 +330,8 @@ typedef struct dasdfmt_info {
- int device_id;
- int keep_volser;
- int force_host;
-+ int layout_specified;
-+ int check;
- int yast_mode;
- int procnum;
- } dasdfmt_info_t;
diff --git a/s390-tools-sles12sp3-dasdfmt-10-Add-expand-format-mode.patch b/s390-tools-sles12sp3-dasdfmt-10-Add-expand-format-mode.patch
deleted file mode 100644
index 840edfb..0000000
--- a/s390-tools-sles12sp3-dasdfmt-10-Add-expand-format-mode.patch
+++ /dev/null
@@ -1,266 +0,0 @@
-Subject: [PATCH] [FEAT LS1501] dasdfmt: Add new formatting modes
-From: Jan Höppner
-
-Summary: dasdfmt: Add new formatting modes
-Description: Introduce new formatting modes 'quick' and 'expand' to either
- format an earlier formatted DASD that could potentially be
- re-initialized very easily or format unformatted tracks at the
- end of a device that was previously extended.
-
- Also add the command line argument --check to provide a function
- that checks a DASD volume for correct formatting.
-Upstream-ID: -
-Problem-ID: LS1501
-
-Upstream-Description:
-
- dasdfmt: Add expand format mode
-
- If a DASD was extended, the format of the first part of the disk
- can be assumed to be correct. Usually, a full format would be necessary
- to ensure a fully functional disk. However, this would also format
- the already formatted part resulting in the loss of the data stored in
- that part.
-
- Add an expand mode which will search for the start of the unformatted
- area and begin formatting from there.
-
- The mode is being specified by --mode=expand (or -M expand for short).
-
- Furthermore:
- - Add string array for the individual modes for output purposes
- - Avoid writing labels when EXPAND mode is active since they can be
- assumed to be correct
- - Add man page description
-
-
-Signed-off-by: Jan Höppner
----
- dasdfmt/dasdfmt.8 | 7 +++
- dasdfmt/dasdfmt.c | 120 +++++++++++++++++++++++++++++++++++++++++++++++++-----
- dasdfmt/dasdfmt.h | 5 ++
- 3 files changed, 123 insertions(+), 9 deletions(-)
-
---- a/dasdfmt/dasdfmt.8
-+++ b/dasdfmt/dasdfmt.8
-@@ -118,6 +118,13 @@ Format the first two tracks and write la
- this option if you are sure that the target DASD already contains a regular
- format with the specified blocksize. A blocksize can optionally be specified
- using \fB-b\fR (\fB--blocksize\fR).
-+.IP expand
-+Format all unformatted tracks at the end of the target DASD. This mode assumes
-+that tracks at the beginning of the DASD volume have already been correctly
-+formatted, while a consecutive set of tracks at the end are unformatted. You can
-+use this mode to make added space available for Linux use after dynamically
-+increasing the size of a DASD volume. A blocksize can optionally be specified
-+using \fB-b\fR (\fB--blocksize\fR).
- .RE
-
- .TP
---- a/dasdfmt/dasdfmt.c
-+++ b/dasdfmt/dasdfmt.c
-@@ -88,7 +88,9 @@ static void exit_usage(int exitcode)
- " device node of the device to format\n"
- " is either\n"
- " 'full' to fully format the device (default)\n"
-- " 'quick' to format only the first two tracks\n");
-+ " 'quick' to format only the first two tracks\n"
-+ " 'expand' to format only the unformatted end"
-+ " of a device\n");
- exit(exitcode);
- }
-
-@@ -791,8 +793,11 @@ static format_check_t check_track_format
- if (errno == ENOTTY) {
- ERRMSG("%s: Missing kernel support for format checking",
- prog_name);
-- if (!info->check)
-+ if (mode == EXPAND) {
-+ ERRMSG(". Mode 'expand' cannot be used");
-+ } else if (!info->check) {
- ERRMSG(" (--force to override)");
-+ }
- ERRMSG_EXIT(EXIT_FAILURE, ".\n");
- }
- ERRMSG_EXIT(EXIT_FAILURE, "%s: Could no check format: %s\n",
-@@ -957,8 +962,7 @@ static void dasdfmt_print_info(dasdfmt_i
- printf(" Compatible Disk Layout : %s\n",
- (p->intensity & DASD_FMT_INT_COMPAT) ? "yes" : "no");
- printf(" Blocksize : %d\n", p->blksize);
-- printf(" Mode : %s\n",
-- (mode == FULL) ? "Full" : "Quick");
-+ printf(" Mode : %s\n", mode_str[mode]);
-
- if (info->testmode)
- printf("Test mode active, omitting ioctl.\n");
-@@ -1169,6 +1173,60 @@ static void dasdfmt_write_labels(dasdfmt
- }
-
- /*
-+ * This function will search for the beginning of an unformatted area
-+ * on the device. It checks selected tracks beforehand and makes sure
-+ * that the device is formatted to a certain extent. Otherwise the
-+ * process is terminated.
-+ */
-+static void dasdfmt_find_start(dasdfmt_info_t *info, unsigned int cylinders,
-+ unsigned heads, format_data_t *format_params)
-+{
-+ format_check_t cdata;
-+ unsigned int middle;
-+ unsigned int left = 2;
-+ unsigned int right = (cylinders * heads) - 1;
-+ unsigned int first = left;
-+
-+ check_blocksize(info, format_params->blksize);
-+
-+ format_params->start_unit = 0;
-+ format_params->stop_unit = 4;
-+ cdata = check_track_format(info, format_params);
-+
-+ if (cdata.result) {
-+ evaluate_format_error(info, &cdata, heads);
-+ ERRMSG_EXIT(EXIT_FAILURE, "Use --mode=full to perform a "
-+ "clean format.\n");
-+ }
-+
-+ printf("Expansion mode active. Searching for starting position...\n");
-+
-+ while (left <= right) {
-+ /* new track number to look at */
-+ middle = left + ((right - left) / 2);
-+
-+ format_params->start_unit = middle;
-+ format_params->stop_unit = middle;
-+ cdata = check_track_format(info, format_params);
-+ if (cdata.blksize != format_params->blksize) {
-+ first = middle;
-+ right = middle - 1;
-+ } else {
-+ left = middle + 1;
-+ }
-+ }
-+
-+ if (first == 2 && cdata.blksize == format_params->blksize)
-+ ERRMSG_EXIT(EXIT_FAILURE,
-+ "No unformatted part found, aborting.\n");
-+
-+ printf("Done. Unformatted part starts at track %d.\n", first);
-+
-+ /* return format_params with start_unit set to the correct value */
-+ format_params->start_unit = first;
-+}
-+
-+/*
- * formats the disk cylinderwise
- */
- static void dasdfmt_format(dasdfmt_info_t *info, unsigned int cylinders,
-@@ -1239,6 +1297,40 @@ static void dasdfmt_prepare_and_format(d
- }
-
- /*
-+ * This function will start the expand format process.
-+ */
-+static void dasdfmt_expand_format(dasdfmt_info_t *info, unsigned int cylinders,
-+ unsigned int heads, format_data_t *p)
-+{
-+ if (!((info->withoutprompt) && (info->verbosity < 1)))
-+ printf("Formatting the device. This may take a while "
-+ "(get yourself a coffee).\n");
-+
-+ if (info->verbosity > 0)
-+ printf("Detaching the device...\n");
-+
-+ if (ioctl(filedes, BIODASDDISABLE, p) != 0)
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: (prepare device) IOCTL "
-+ "BIODASDDISABLE failed. (%s)\n", prog_name,
-+ strerror(errno));
-+ disk_disabled = 1;
-+
-+ dasdfmt_format(info, cylinders, heads, p);
-+
-+ if (info->verbosity > 0)
-+ printf("Formatting tracks complete...\n");
-+
-+ if (info->verbosity > 0)
-+ printf("Re-accessing the device...\n");
-+
-+ if (ioctl(filedes, BIODASDENABLE, p) != 0)
-+ ERRMSG_EXIT(EXIT_FAILURE, "%s: (prepare device) IOCTL "
-+ "BIODASDENABLE failed. (%s)\n", prog_name,
-+ strerror(errno));
-+ disk_disabled = 0;
-+}
-+
-+/*
- * This function will only format the first two tracks of a DASD.
- * The rest of the DASD is untouched and left as is.
- */
-@@ -1310,6 +1402,10 @@ static void do_format_dasd(dasdfmt_info_
- case QUICK: /* just the first two */
- p->stop_unit = 1;
- break;
-+ case EXPAND: /* only the end of the disk */
-+ dasdfmt_find_start(info, cylinders, heads, p);
-+ p->stop_unit = (cylinders * heads) - 1;
-+ break;
- }
-
- if ((info->verbosity > 0) || !info->withoutprompt || info->testmode)
-@@ -1338,10 +1434,12 @@ static void do_format_dasd(dasdfmt_info_
-
- if (!info->testmode) {
- if (!info->withoutprompt) {
-- printf("\n--->> ATTENTION! <<---\n");
-- printf("All data of that device will be lost.\nType "
-- "\"yes\" to continue, no will leave the disk "
-- "untouched: ");
-+ printf("\n");
-+ if (mode != EXPAND)
-+ printf("--->> ATTENTION! <<---\nAll data of "
-+ "that device will be lost.\n");
-+ printf("Type \"yes\" to continue, no will leave the "
-+ "disk untouched: ");
- if (fgets(inp_buffer, sizeof(inp_buffer), stdin) == NULL)
- return;
- if (strcasecmp(inp_buffer, "yes") &&
-@@ -1359,12 +1457,14 @@ static void do_format_dasd(dasdfmt_info_
- case QUICK:
- dasdfmt_quick_format(info, cylinders, heads, p);
- break;
-+ case EXPAND:
-+ dasdfmt_expand_format(info, cylinders, heads, p);
- }
-
- if (!info->yast_mode)
- printf("Finished formatting the device.\n");
-
-- if (!info->writenolabel)
-+ if (!(info->writenolabel || mode == EXPAND))
- dasdfmt_write_labels(info, vlabel, cylinders, heads);
-
- if (!info->yast_mode)
-@@ -1526,6 +1626,8 @@ int main(int argc, char *argv[])
- mode = FULL;
- else if (strcasecmp(optarg, "quick") == 0)
- mode = QUICK;
-+ else if (strcasecmp(optarg, "expand") == 0)
-+ mode = EXPAND;
- else
- ERRMSG_EXIT(EXIT_FAILURE,
- "%s: The specified mode '%s' is "
---- a/dasdfmt/dasdfmt.h
-+++ b/dasdfmt/dasdfmt.h
-@@ -150,8 +150,13 @@ struct dasd_eckd_characteristics {
- typedef enum format_mode_t {
- FULL, /* default mode */
- QUICK, /* format only the first 2 tracks */
-+ EXPAND, /* search for unformatted area and format only that part*/
- } format_mode_t;
-
-+static const char mode_str[3][10] = {
-+ "Full", "Quick", "Expand"
-+};
-+
- /*
- * struct format_data_t
- * represents all data necessary to format a dasd
diff --git a/s390-tools-sles12sp3-dbginfo-01-libutil-Add-utility-functions.patch b/s390-tools-sles12sp3-dbginfo-01-libutil-Add-utility-functions.patch
deleted file mode 100644
index 5fed129..0000000
--- a/s390-tools-sles12sp3-dbginfo-01-libutil-Add-utility-functions.patch
+++ /dev/null
@@ -1,1053 +0,0 @@
-Subject: [PATCH] [FEAT RTL1601] libutil: Add utility functions
-From: Peter Oberparleiter
-
-Summary: libutil: Add utility functions
-Description: Introduce utility functions for use by s390-tools.
-Upstream-ID: -
-Problem-ID: RTL1601
-
-Signed-off-by: Peter Oberparleiter
----
- include/util_base.h | 2
- include/util_libc.h | 125 +++++++++++++++++++
- include/util_opt.h | 82 ++++++++++++
- include/util_panic.h | 45 ++++++
- include/util_prg.h | 55 ++++++++
- libutil/Makefile | 11 +
- libutil/util_libc.c | 155 +++++++++++++++++++++++
- libutil/util_opt.c | 281 +++++++++++++++++++++++++++++++++++++++++++
- libutil/util_panic.c | 119 ++++++++++++++++++
- libutil/util_prg.c | 111 ++++++++++++++++
- 10 files changed, 985 insertions(+), 1 deletion(-)
-
---- a/include/util_base.h
-+++ b/include/util_base.h
-@@ -15,4 +15,6 @@ void util_hexdump(FILE *fh, const char *
- void util_hexdump_grp(FILE *fh, const char *tag, const void *data, int group,
- int cnt, int indent);
-
-+#define UTIL_ARRAY_SIZE(array) (sizeof(array) / sizeof((array)[0]))
-+
- #endif /* UTIL_BASE_H */
---- /dev/null
-+++ b/include/util_libc.h
-@@ -0,0 +1,125 @@
-+/**
-+ * @defgroup util_libc_h util_libc: Libc wrapper interface
-+ * @{
-+ * @brief Handle standard errors for libc functions
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef UTIL_LIBC_H
-+#define UTIL_LIBC_H
-+
-+#include
-+
-+/**
-+ * Allocate memory or panic in case of failure
-+ *
-+ * @param[in] size Number of bytes to be allocated
-+ *
-+ * @returns Pointer to allocated memory buffer
-+ */
-+#define util_malloc(size) \
-+ __util_malloc(__func__, __FILE__, __LINE__, size)
-+
-+void *__util_malloc(const char *func, const char *file, int line, size_t size);
-+
-+/**
-+ * Allocate zero-initialized memory or panic in case of failure
-+ *
-+ * @param[in] size Number of bytes to be allocated
-+ *
-+ * @returns Pointer to allocated memory buffer initialized with zeroes
-+ */
-+#define util_zalloc(size) \
-+ __util_zalloc(__func__, __FILE__, __LINE__, size)
-+
-+void *__util_zalloc(const char *func, const char *file, int line, size_t size);
-+
-+
-+/**
-+ * Re-allocate memory or exit in case of failure
-+ *
-+ * @param[in] ptr Pointer ot old memory buffer
-+ * @param[in] size Number of bytes to be allocated
-+ *
-+ * @returns Pointer to allocated memory buffer
-+ */
-+#define util_realloc(ptr, size) \
-+ __util_realloc(__func__, __FILE__, __LINE__, ptr, size)
-+
-+void *__util_realloc(const char *func, const char *file, int line,
-+ void *ptr, size_t size);
-+
-+/**
-+ * Duplicate a string buffer or exit in case of failure
-+ *
-+ * @param[in] str String to be duplicated
-+ *
-+ * @returns Pointer to newly allocated and copied string
-+ */
-+#define util_strdup(str) \
-+ __util_strdup(__func__, __FILE__, __LINE__, str)
-+
-+void *__util_strdup(const char *func, const char *file, int line,
-+ const char *str);
-+
-+/**
-+ * Print to allocated string or exit in case of failure
-+ *
-+ * @param[in] strp String to be allocated
-+ * @param[in] fmt Format string for generation of string
-+ * @param[in] ap Parameters for format string
-+ *
-+ * @returns num Number of formatted characters
-+ */
-+#define util_vasprintf(strp, fmt, ap) \
-+ __util_vasprintf(__func__, __FILE__, __LINE__, strp, fmt, ap)
-+
-+#define UTIL_VASPRINTF(strp, fmt, ap) \
-+do { \
-+ va_start(ap, fmt); \
-+ util_vasprintf(strp, fmt, ap); \
-+ va_end(ap); \
-+} while (0)
-+
-+int __util_vasprintf(const char *func, const char *file, int line,
-+ char **strp, const char *fmt, va_list ap);
-+
-+/**
-+ * Print to newly allocated string or exit in case of failure
-+ *
-+ * @param[in] strp String to be allocated
-+ * @param[in] ... Format string and parameters for format string
-+ *
-+ * @returns num Number of formatted characters
-+ */
-+#define util_asprintf(strp, ...) \
-+ __util_asprintf(__func__, __FILE__, __LINE__, strp, ##__VA_ARGS__)
-+
-+int __util_asprintf(const char *func, const char *file, int line,
-+ char **strp, const char *fmt, ...);
-+
-+
-+/**
-+ * Print to string buffer or exit in case of failure
-+ *
-+ * @param[in] str String buffer
-+ * @param[in] fmt Format string for generation of string
-+ * @param[in] ap Parameters for format string
-+ *
-+ * @returns num Number of formatted characters
-+ */
-+#define util_vsprintf(str, fmt, ap) \
-+ __util_vsprintf(__func__, __FILE__, __LINE__, str, fmt, ap)
-+
-+#define UTIL_VSPRINTF(str, fmt, ap) \
-+do { \
-+ va_start(ap, fmt); \
-+ util_vsprintf(str, fmt, ap); \
-+ va_end(ap); \
-+} while (0)
-+
-+int __util_vsprintf(const char *func, const char *file, int line,
-+ char *str, const char *fmt, va_list ap);
-+
-+#endif /** UTIL_LIBC_H @} */
---- /dev/null
-+++ b/include/util_opt.h
-@@ -0,0 +1,82 @@
-+/**
-+ * @defgroup util_opt_h util_opt: Command line options interface
-+ * @{
-+ * @brief Parse the command line options
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef UTIL_OPT_H
-+#define UTIL_OPT_H
-+
-+#include
-+#include
-+
-+/* Flag indicating that an option does not have a short form */
-+#define UTIL_OPT_FLAG_NOSHORT 1
-+
-+/* Flag indicating that an option does not have a long form */
-+#define UTIL_OPT_FLAG_NOLONG 2
-+
-+/* Flag indicating that this is a section heading */
-+#define UTIL_OPT_FLAG_SECTION 4
-+
-+/**
-+ * Command line option
-+ */
-+struct util_opt {
-+ /** Defined by getopt.h, see "man getopt_long" */
-+ struct option option;
-+ /** For options with arguments: Argument name */
-+ char *argument;
-+ /** Description displayed for --help */
-+ char *desc;
-+ /** Flags for this option */
-+ int flags;
-+};
-+
-+/**
-+ * Standard option: --help
-+ */
-+#define UTIL_OPT_HELP \
-+{ \
-+ .option = { "help", 0, NULL, 'h' }, \
-+ .desc = "Print this help, then exit", \
-+}
-+
-+/**
-+ * Standard option: --version
-+ */
-+#define UTIL_OPT_VERSION \
-+{ \
-+ .option = { "version", 0, NULL, 'v' }, \
-+ .desc = "Print version information, then exit", \
-+}
-+
-+/**
-+ * End-marker for the option pointer vector
-+ */
-+#define UTIL_OPT_END \
-+{ \
-+ .option = { NULL, 0, NULL, 0 }, \
-+}
-+
-+/**
-+ * Section header
-+ */
-+#define UTIL_OPT_SECTION(title) \
-+{ \
-+ .desc = (title), \
-+ .flags = UTIL_OPT_FLAG_SECTION, \
-+}
-+
-+/*
-+ * Option functions
-+ */
-+void util_opt_init(struct util_opt *opt_vec, const char *opt_prefix);
-+int util_opt_getopt_long(int argc, char *argv[]);
-+void util_opt_print_help(void);
-+void util_opt_print_indented(const char *opt, const char *desc);
-+void util_opt_print_parse_error(char opt, char *argv[]);
-+
-+#endif /** UTIL_OPT_H @} */
---- /dev/null
-+++ b/include/util_panic.h
-@@ -0,0 +1,45 @@
-+/**
-+ * @defgroup util_panic_h util_panic: Panic interface
-+ * @{
-+ * @brief Collect FFDC data for unexpected errors
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef UTIL_PANIC_H
-+#define UTIL_PANIC_H
-+
-+#include "zt_common.h"
-+
-+#ifndef __noreturn
-+#define __noreturn
-+#endif
-+
-+/**
-+ * Write message, print backtrace and then call the abort() function
-+ *
-+ * @param[in] ... Format string and parameters describing the panic reason
-+ */
-+#define util_panic(...) \
-+ __util_panic(__func__, __FILE__, __LINE__, ##__VA_ARGS__)
-+
-+void __util_panic(const char *func, const char *file, int line,
-+ const char *fmt, ...) __noreturn;
-+
-+/**
-+ * Ensure that assumption is not true, otherwise panic
-+ *
-+ * Example: util_assert(ptr == NULL, "The ptr must be NULL, but is %p", ptr)
-+ *
-+ * @param[in] assumption This assumption has to be true
-+ * @param[in] ... Format string and parameters describing the assumption
-+ */
-+#define util_assert(assumption, ...) \
-+ __util_assert(#assumption, __func__, __FILE__, __LINE__, \
-+ assumption, ##__VA_ARGS__)
-+
-+void __util_assert(const char *assertion_string,
-+ const char *func, const char *file, int line,
-+ int assumption, const char *fmt, ...);
-+
-+#endif /** UTIL_PANIC_H @} */
---- /dev/null
-+++ b/include/util_prg.h
-@@ -0,0 +1,55 @@
-+/**
-+ * @defgroup util_prg_h util_prg: Program interface
-+ * @{
-+ * @brief Print standard program messages
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef UTIL_PRG_H
-+#define UTIL_PRG_H
-+
-+#include
-+#include
-+#include
-+#include
-+#include
-+
-+/**
-+ * Copyright description
-+ */
-+struct util_prg_copyright {
-+ /** Name of the copyright owner, e.g. IBM */
-+ const char *owner;
-+ /** Year of first publishing */
-+ int pub_first;
-+ /** Year of last major changes */
-+ int pub_last;
-+};
-+
-+/**
-+ * @brief Coypright end marker
-+ */
-+#define UTIL_PRG_COPYRIGHT_END {NULL, 0, 0}
-+
-+/**
-+ * Program description
-+ */
-+struct util_prg {
-+ /** Description for help */
-+ const char *desc;
-+ /** Positional arguments */
-+ const char *args;
-+ /** Copyright list */
-+ struct util_prg_copyright copyright_vec[];
-+};
-+
-+void util_prg_init(const struct util_prg *prg);
-+void util_prg_print_parse_error(void);
-+void util_prg_print_required_arg(const char *option);
-+void util_prg_print_invalid_option(const char *option);
-+void util_prg_print_arg_error(const char *arg_name);
-+void util_prg_print_version(void);
-+void util_prg_print_help(void);
-+
-+#endif /** UTIL_PRG_H @} */
---- a/libutil/Makefile
-+++ b/libutil/Makefile
-@@ -2,12 +2,21 @@ include ../common.mak
-
- CPPFLAGS += -I../include
-
--all: util_base.o util_list.o util_part.o util_proc.o
-+all: util_base.o util_list.o util_part.o util_proc.o util_libc.o util_opt.o \
-+ util_panic.o util_prg.o
-
- util_list.o: util_list.c ../include/util.h
-
- util_proc.o: util_proc.c ../include/util_proc.h
-
-+util_libc.o: util_libc.c ../include/util_libc.h
-+
-+util_opt.o: util_opt.c ../include/util_opt.h
-+
-+util_panic.o: util_panic.c ../include/util_panic.h
-+
-+util_prg.o: util_prg.c ../include/util_prg.h
-+
- install: all
-
- clean:
---- /dev/null
-+++ b/libutil/util_libc.c
-@@ -0,0 +1,155 @@
-+/*
-+ * util - Utility function library
-+ *
-+ * Handle standard errors for libc functions
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#define _GNU_SOURCE /* for program_invocation_short_name */
-+
-+#include
-+#include
-+#include
-+#include
-+#include
-+
-+#include "util_base.h"
-+#include "util_panic.h"
-+#include "util_libc.h"
-+
-+/*
-+ * Return size as string of largest unit, e.g. 1025 = "1 KiB"
-+ */
-+static void format_size(char *str, size_t size)
-+{
-+ static const char * const unit_vec[] =
-+ {"byte", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB"};
-+ unsigned int i;
-+
-+ for (i = 0; i < UTIL_ARRAY_SIZE(unit_vec); i++) {
-+ if (size / 1024 == 0) {
-+ sprintf(str, "%zu %s", size, unit_vec[i]);
-+ return;
-+ }
-+ size /= 1024;
-+ }
-+ sprintf(str, "huge");
-+}
-+
-+static void __util_oom(const char *func, const char *file, int line,
-+ size_t size)
-+{
-+ char size_str[256];
-+
-+ fprintf(stderr, "%s: Failed to allocate memory",
-+ program_invocation_short_name);
-+ if (size > 0) {
-+ format_size(size_str, size);
-+ fprintf(stderr, " (%s)", size_str);
-+ }
-+ fprintf(stderr, " at %s:%d %s()\n", file, line, func);
-+ exit(EXIT_FAILURE);
-+}
-+
-+/*
-+ * Allocate memory or exit in case of failure
-+ */
-+void *__util_malloc(const char *func, const char *file, int line, size_t size)
-+{
-+ void *buf;
-+
-+ buf = malloc(size);
-+
-+ if (buf == NULL)
-+ __util_oom(func, file, line, size);
-+
-+ return buf;
-+}
-+
-+/*
-+ * Allocate zero-initialized memory or exit in case of failure
-+ */
-+void *__util_zalloc(const char *func, const char *file, int line, size_t size)
-+{
-+ void *buf = __util_malloc(func, file, line, size);
-+
-+ memset(buf, 0, size);
-+
-+ return buf;
-+}
-+
-+/*
-+ * Re-allocate memory or exit in case of failure
-+ */
-+void *__util_realloc(const char *func, const char *file, int line,
-+ void *ptr, size_t size)
-+{
-+ void *buf;
-+
-+ buf = realloc(ptr, size);
-+
-+ if (buf == NULL)
-+ __util_oom(func, file, line, size);
-+
-+ return buf;
-+}
-+
-+/*
-+ * Duplicate a string buffer or exit in case of failure
-+ */
-+void *__util_strdup(const char *func, const char *file, int line,
-+ const char *str)
-+{
-+ void *buf = strdup(str);
-+
-+ if (buf == NULL)
-+ __util_oom(func, file, line, strlen(str) + 1);
-+
-+ return buf;
-+}
-+
-+/*
-+ * Print to newly allocated string or exit in case of failure
-+ */
-+int __util_vasprintf(const char *func, const char *file, int line,
-+ char **strp, const char *fmt, va_list ap)
-+{
-+ int rc;
-+
-+ rc = vasprintf(strp, fmt, ap);
-+ if (rc == -1)
-+ __util_oom(func, file, line, 0);
-+
-+ return rc;
-+}
-+
-+/*
-+ * Print to newly allocated string or exit in case of failure
-+ */
-+int __util_asprintf(const char *func, const char *file, int line,
-+ char **strp, const char *fmt, ...)
-+{
-+ va_list ap;
-+ int rc;
-+
-+ va_start(ap, fmt);
-+ rc = __util_vasprintf(func, file, line, strp, fmt, ap);
-+ va_end(ap);
-+ return rc;
-+}
-+
-+/*
-+ * Print to string buffer or exit in case of failure
-+ */
-+int __util_vsprintf(const char *func, const char *file, int line,
-+ char *str, const char *fmt, va_list ap)
-+{
-+ int rc;
-+
-+ rc = vsprintf(str, fmt, ap);
-+ if (rc == -1)
-+ __util_assert("rc != -1", func, file, line,
-+ rc != -1, "Could not format string\n");
-+ return rc;
-+}
---- /dev/null
-+++ b/libutil/util_opt.c
-@@ -0,0 +1,281 @@
-+/*
-+ * util - Utility function library
-+ *
-+ * Parse the command line options
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#include
-+#include
-+#include
-+#include
-+#include
-+
-+#include "util_base.h"
-+#include "util_libc.h"
-+#include "util_opt.h"
-+#include "util_panic.h"
-+#include "util_prg.h"
-+
-+/*
-+ * Private data
-+ */
-+/// @cond
-+static struct util_opt_l {
-+ /* Option character string for getopt_long() */
-+ char *opt_str;
-+ /* Option array for getopt_long() */
-+ struct option *option_vec;
-+ /* Original util_opt array */
-+ struct util_opt *opt_vec;
-+ /* Length of longest option string */
-+ int opt_max;
-+} l;
-+
-+struct util_opt_l *util_opt_l = &l;
-+/// @endcond
-+
-+#define util_opt_iterate(opt) \
-+ for (opt = &l.opt_vec[0]; opt->desc != NULL; opt++)
-+
-+#define MAX(x, y) ((x) < (y) ? (y) : (x))
-+#define MAX_OPTLEN 256
-+
-+static int opt_max_len(void);
-+
-+/**
-+ * Initialize the command line options
-+ *
-+ * Build short option string and long option array to be used for getopt_long().
-+ * The ":" prefix is added to the short option string for handling of "missing
-+ * required arguments".
-+ *
-+ * @param[in] opt_vec Option array
-+ * @param[in] opt_prefix Optional option string prefix
-+ */
-+void util_opt_init(struct util_opt *opt_vec, const char *opt_prefix)
-+{
-+ int i, j, count;
-+ char *str;
-+ size_t prefix_len = opt_prefix ? strlen(opt_prefix) : 0;
-+
-+ opterr = 0;
-+ /* Get number of options */
-+ for (count = 0; opt_vec[count].desc != NULL; count++);
-+ /*
-+ * Allocate short option string for worst case when all options have
-+ * optional parameters e.g "x::" and long option string.
-+ */
-+ l.opt_str = util_malloc(sizeof(char) * count * 3 + 2 + prefix_len);
-+ l.option_vec = util_malloc(sizeof(struct option) * (count + 1));
-+ l.opt_vec = opt_vec;
-+
-+ str = l.opt_str;
-+ if (opt_prefix) {
-+ strcpy(str, opt_prefix);
-+ str += prefix_len;
-+ }
-+ /* Force getopt_long() to return ':' for missing required arguments */
-+ *str++ = ':';
-+ /* Construction of input structures for getopt_long() function. */
-+ for (i = 0, j = 0; i < count; i++) {
-+ if (opt_vec[i].flags & UTIL_OPT_FLAG_SECTION)
-+ continue;
-+ memcpy(&l.option_vec[j++], &opt_vec[i].option,
-+ sizeof(struct option));
-+ if (opt_vec[i].flags & UTIL_OPT_FLAG_NOSHORT)
-+ continue;
-+ *str++ = opt_vec[i].option.val;
-+ switch (opt_vec[i].option.has_arg) {
-+ case no_argument:
-+ break;
-+ case required_argument:
-+ *str++ = ':';
-+ break;
-+ case optional_argument:
-+ *str++ = ':';
-+ *str++ = ':';
-+ break;
-+ default:
-+ util_panic("Unexpected \"has_arg\" parameter: %d\n",
-+ opt_vec[i].option.has_arg);
-+ }
-+ }
-+ /* Add end marker to option array and short option string */
-+ memset(&l.option_vec[j], 0, sizeof(struct option));
-+ *str = '\0';
-+
-+ l.opt_max = opt_max_len();
-+}
-+
-+/**
-+ * Wrapper for getopt_long
-+ *
-+ * @param[in] argc Count of command line parameters
-+ * @param[in] argv Array of command line parameters
-+ */
-+int util_opt_getopt_long(int argc, char *argv[])
-+{
-+ return getopt_long(argc, argv, l.opt_str, l.option_vec, NULL);
-+}
-+
-+/*
-+ * Format option name: Add short, long option and argument (as applicable)
-+ */
-+static void format_opt(char *buf, size_t maxlen, const struct util_opt *opt)
-+{
-+ int has_arg, flags, rc;
-+ char val, *arg_str;
-+ const char *name;
-+
-+ has_arg = opt->option.has_arg;
-+ name = opt->option.name;
-+ val = opt->option.val;
-+ flags = opt->flags;
-+
-+ /* Prepare potential option argument string */
-+ if (has_arg == optional_argument) {
-+ if (flags & UTIL_OPT_FLAG_NOLONG)
-+ util_asprintf(&arg_str, "[%s]", opt->argument);
-+ else
-+ util_asprintf(&arg_str, "[=%s]", opt->argument);
-+ } else if (has_arg == required_argument) {
-+ util_asprintf(&arg_str, " %s", opt->argument);
-+ } else {
-+ util_asprintf(&arg_str, "");
-+ }
-+
-+ /* Format the option */
-+ if (flags & UTIL_OPT_FLAG_NOLONG)
-+ rc = snprintf(buf, maxlen, "-%c%s", val, arg_str);
-+ else if (flags & UTIL_OPT_FLAG_NOSHORT)
-+ rc = snprintf(buf, maxlen, " --%s%s", name, arg_str);
-+ else
-+ rc = snprintf(buf, maxlen, "-%c, --%s%s", val, name, arg_str);
-+
-+ util_assert(rc < (int)maxlen, "Option too long: %s\n", name);
-+ free(arg_str);
-+}
-+
-+/*
-+ * Return size of the longest formatted option
-+ */
-+static int opt_max_len(void)
-+{
-+ const struct util_opt *opt;
-+ unsigned int max = 0;
-+ char opt_str[MAX_OPTLEN];
-+
-+ util_opt_iterate(opt) {
-+ if (opt->flags & UTIL_OPT_FLAG_SECTION)
-+ continue;
-+ format_opt(opt_str, MAX_OPTLEN, opt);
-+ max = MAX(max, strlen(opt_str));
-+ }
-+ return max;
-+}
-+
-+/*
-+ * Print option description with indentation
-+ */
-+static void print_opt_description(const char *desc_in, int indent)
-+{
-+ char *word, *line, *desc, *desc_ptr;
-+ int word_len, pos = indent;
-+
-+ desc = desc_ptr = util_strdup(desc_in);
-+ line = strsep(&desc, "\n");
-+ while (line) {
-+ word = strsep(&line, " ");
-+ pos = indent;
-+ while (word) {
-+ word_len = strlen(word);
-+ if (pos + word_len + 1 > 80) {
-+ printf("\n%*s", indent, " ");
-+ pos = indent;
-+ }
-+ printf(" %s", word);
-+ pos += word_len + 1;
-+ word = strsep(&line, " ");
-+ }
-+ if (desc)
-+ printf("\n%*s", indent, " ");
-+ line = strsep(&desc, "\n");
-+ }
-+ printf("\n");
-+ free(desc_ptr);
-+}
-+
-+/**
-+ * Print an option name, followed by a description indented to fit the
-+ * longest option name
-+ */
-+void util_opt_print_indented(const char *opt, const char *desc)
-+{
-+ printf(" %-*s ", l.opt_max, opt);
-+ print_opt_description(desc, 2 + l.opt_max);
-+}
-+
-+/**
-+ * Print the usage of the command line options to the console
-+ */
-+void util_opt_print_help(void)
-+{
-+ char opt_str[MAX_OPTLEN];
-+ struct util_opt *opt;
-+ int first = 1;
-+
-+ /*
-+ * Create format string: " -%c, --%-s %s"
-+ *
-+ * Example:
-+ *
-+ * -p, --print STRING Print STRING to console
-+ */
-+ util_opt_iterate(opt) {
-+ if (opt->flags & UTIL_OPT_FLAG_SECTION) {
-+ printf("%s%s\n", first ? "" : "\n", opt->desc);
-+ first = 0;
-+ continue;
-+ }
-+ format_opt(opt_str, MAX_OPTLEN, opt);
-+ util_opt_print_indented(opt_str, opt->desc);
-+ }
-+}
-+
-+/**
-+ * Print option parsing error message
-+ *
-+ * This function should be used when the return code of the
-+ * util_opt_getopt_long() function returns a character that does
-+ * not match any of the expected options.
-+ *
-+ * @param[in] opt Short option returned by getopt_long()
-+ * @param[in] argv Option array
-+ */
-+void util_opt_print_parse_error(char opt, char *argv[])
-+{
-+ char optopt_str[3];
-+
-+ switch (opt) {
-+ case ':':
-+ /* A required option argument has not been specified */
-+ util_prg_print_required_arg(argv[optind - 1]);
-+ break;
-+ case '?':
-+ /* An invalid option has been specified */
-+ if (optopt) {
-+ /* Short option */
-+ sprintf(optopt_str, "-%c", optopt);
-+ util_prg_print_invalid_option(optopt_str);
-+ } else {
-+ /* Long option */
-+ util_prg_print_invalid_option(argv[optind - 1]);
-+ }
-+ break;
-+ default:
-+ util_panic("Option '%c' should not be handled here\n", opt);
-+ }
-+}
-+
---- /dev/null
-+++ b/libutil/util_panic.c
-@@ -0,0 +1,119 @@
-+/*
-+ * util - Utility function library
-+ *
-+ * Collect FFDC data for unexpected errors
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#include
-+#include
-+
-+#include
-+#include
-+#include
-+#include
-+
-+#include "util_base.h"
-+#include "util_panic.h"
-+
-+/*
-+ * Obtain a backtrace and print it to stderr
-+ *
-+ * To get symbols, compile the code with "-rdynamic".
-+ */
-+static void print_backtrace(void)
-+{
-+ void *array[256];
-+ size_t i, size;
-+ char **strings;
-+
-+ fprintf(stderr, "Backtrace:\n\n");
-+ size = backtrace(array, UTIL_ARRAY_SIZE(array));
-+ strings = backtrace_symbols(array, size);
-+ if (strings == NULL) {
-+ fprintf(stderr, " Could not obtain backtrace (ENOMEM)\n");
-+ return;
-+ }
-+ for (i = 0; i < size; i++)
-+ fprintf(stderr, " %s\n", strings[i]);
-+
-+ free(strings);
-+}
-+
-+/*
-+ * Check for core ulimit
-+ */
-+static void ulimit_core_check(void)
-+{
-+ struct rlimit limit;
-+
-+ if (getrlimit(RLIMIT_CORE, &limit) != 0)
-+ return;
-+ if (limit.rlim_cur != 0)
-+ return;
-+ fprintf(stderr, "Core dump size is zero. To get a full core dump use 'ulimit -c unlimited'.\n");
-+}
-+
-+/*
-+ * Print FFDC data and then abort
-+ */
-+static void panic_finish(const char *func, const char *file, int line,
-+ const char *fmt, va_list ap)
-+{
-+ /* Write panic error string */
-+ fprintf(stderr, "\n");
-+ fprintf(stderr, "Error string:\n");
-+ fprintf(stderr, "\n");
-+ fprintf(stderr, " ");
-+ vfprintf(stderr, fmt, ap);
-+ fprintf(stderr, "\n");
-+ /* Write file, line number, and function name */
-+ fprintf(stderr, "Location:\n\n");
-+ fprintf(stderr, " %s:%d: %s()\n", file, line, func);
-+ fprintf(stderr, "\n");
-+
-+ /* Print the function backtrace */
-+ print_backtrace();
-+ fprintf(stderr, "\n");
-+
-+ ulimit_core_check();
-+ fprintf(stderr, "----------------------------------------------------------------------->8-----\n");
-+ abort();
-+}
-+
-+/*
-+ * Do panic processing if the assumption is not true
-+ */
-+void __util_assert(const char *assertion_str,
-+ const char *func, const char *file, int line,
-+ int assumption, const char *fmt, ...)
-+{
-+ va_list ap;
-+
-+ if (assumption)
-+ return;
-+ va_start(ap, fmt);
-+ fprintf(stderr, "---8<-------------------------------------------------------------------------\n");
-+ fprintf(stderr, "ASSERTION FAILED: The application terminated due to an internal or OS error\n");
-+ fprintf(stderr, "\n");
-+ fprintf(stderr, "The following assumption was *not* true:\n");
-+ fprintf(stderr, "\n");
-+ fprintf(stderr, " %s\n", assertion_str);
-+ panic_finish(func, file, line, fmt, ap);
-+}
-+
-+/*
-+ * Do panic processing
-+ */
-+void __util_panic(const char *func, const char *file, int line,
-+ const char *fmt, ...)
-+{
-+ va_list ap;
-+
-+ va_start(ap, fmt);
-+ fprintf(stderr, "---8<-------------------------------------------------------------------------\n");
-+ fprintf(stderr, "PANIC: The application terminated due to an unrecoverable error\n");
-+ panic_finish(func, file, line, fmt, ap);
-+ while(1);
-+}
---- /dev/null
-+++ b/libutil/util_prg.c
-@@ -0,0 +1,111 @@
-+/*
-+ * util - Utility function library
-+ *
-+ * Print standard program messages
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#define _GNU_SOURCE /* for program_invocation_short_name */
-+
-+#include
-+#include
-+
-+#include "util_prg.h"
-+#include "zt_common.h"
-+
-+/*
-+ * Private data
-+ */
-+static struct util_prg_l {
-+ const struct util_prg *prg;
-+} l;
-+
-+struct util_prg_l *util_prg_l = &l;
-+
-+/**
-+ * Print program usage information for the --help option
-+ */
-+void util_prg_print_help(void)
-+{
-+ printf("Usage: %s [OPTIONS]", program_invocation_short_name);
-+ if (l.prg->args)
-+ printf(" %s", l.prg->args);
-+ printf("\n\n%s\n\n", l.prg->desc);
-+}
-+
-+/**
-+ * Print program version information for the --version option
-+ */
-+void util_prg_print_version(void)
-+{
-+ const struct util_prg_copyright *copyright;
-+
-+ printf("%s version %s\n", program_invocation_short_name,
-+ RELEASE_STRING);
-+ copyright = l.prg->copyright_vec;
-+ while (copyright->owner) {
-+ if (copyright->pub_first == copyright->pub_last)
-+ printf("Copyright %s %d\n", copyright->owner,
-+ copyright->pub_first);
-+ else
-+ printf("Copyright %s %d, %d\n", copyright->owner,
-+ copyright->pub_first, copyright->pub_last);
-+ copyright++;
-+ }
-+}
-+
-+/*
-+ * Ask user to use the --help option
-+ */
-+void util_prg_print_parse_error(void)
-+{
-+ fprintf(stderr, "Try '%s --help' for more information.\n",
-+ program_invocation_short_name);
-+}
-+
-+/**
-+ * An option has been specified that is not supported
-+ *
-+ * @param[in] option Option string (short or long)
-+ */
-+void util_prg_print_invalid_option(const char *opt_name)
-+{
-+ fprintf(stderr, "%s: Invalid option '%s'\n",
-+ program_invocation_short_name, opt_name);
-+ util_prg_print_parse_error();
-+}
-+
-+/**
-+ * A required argument for an option is missing
-+ *
-+ * @param[in] option Option string
-+ */
-+void util_prg_print_required_arg(const char *opt_name)
-+{
-+ fprintf(stderr, "%s: Option '%s' requires an argument\n",
-+ program_invocation_short_name, opt_name);
-+ util_prg_print_parse_error();
-+}
-+
-+/**
-+ * A superfluous invalid positional argument has been specified
-+ *
-+ * @param[in] arg_name Name of the invalid argument
-+ */
-+void util_prg_print_arg_error(const char *arg_name)
-+{
-+ fprintf(stderr, "%s: Invalid argument '%s'\n",
-+ program_invocation_short_name, arg_name);
-+ util_prg_print_parse_error();
-+}
-+
-+/**
-+ * Initialize the program module
-+ *
-+ * @param[in] prg Program description
-+ */
-+void util_prg_init(const struct util_prg *prg)
-+{
-+ l.prg = prg;
-+}
diff --git a/s390-tools-sles12sp3-dbginfo-02-dump2tar-Add-sysfs-collection-helper-for-dbginfo.sh-v2.patch b/s390-tools-sles12sp3-dbginfo-02-dump2tar-Add-sysfs-collection-helper-for-dbginfo.sh-v2.patch
deleted file mode 100644
index e0c486d..0000000
--- a/s390-tools-sles12sp3-dbginfo-02-dump2tar-Add-sysfs-collection-helper-for-dbginfo.sh-v2.patch
+++ /dev/null
@@ -1,4687 +0,0 @@
-Subject: [PATCH] [FEAT RTL1601] dump2tar: Add sysfs collection helper for dbginfo.sh
-From: Peter Oberparleiter
-
-Summary: dump2tar: Add sysfs collection helper for dbginfo.sh
-Description: Certain files in virtual filesystems such as sysfs,
- debugfs and procfs do not correctly report their size.
- As a result, tools like tar, cp or rsync cannot be easily
- used to collect these files.
-
- This patch adds a tool to efficiently dump such files
- into a compressed tar archive. It is intended to be used
- by the dbginfo.sh script to significantly speed up data
- collection.
-Upstream-ID: -
-Problem-ID: RTL1601
-
-Changelog:
-
- - v2: Fix compiler warning and missing DESTDIR logic in Makefiles
-
-Signed-off-by: Peter Oberparleiter
----
- Makefile | 2
- README | 5
- dump2tar/Makefile | 12
- dump2tar/include/buffer.h | 47 +
- dump2tar/include/dref.h | 26
- dump2tar/include/dump.h | 60 +
- dump2tar/include/global.h | 20
- dump2tar/include/idcache.h | 23
- dump2tar/include/misc.h | 98 ++
- dump2tar/include/strarray.h | 23
- dump2tar/include/tar.h | 41
- dump2tar/man/Makefile | 12
- dump2tar/man/dump2tar.1 | 454 ++++++++++
- dump2tar/src/Makefile | 36
- dump2tar/src/buffer.c | 271 ++++++
- dump2tar/src/dref.c | 92 ++
- dump2tar/src/dump.c | 1850 ++++++++++++++++++++++++++++++++++++++++++++
- dump2tar/src/dump2tar.c | 474 +++++++++++
- dump2tar/src/global.c | 17
- dump2tar/src/idcache.c | 153 +++
- dump2tar/src/misc.c | 492 +++++++++++
- dump2tar/src/strarray.c | 81 +
- dump2tar/src/tar.c | 270 ++++++
- 23 files changed, 4558 insertions(+), 1 deletion(-)
-
---- a/Makefile
-+++ b/Makefile
-@@ -8,7 +8,7 @@ SUB_DIRS = $(LIB_DIRS) zipl zdump fdasd
- tape390 osasnmpd qetharp ip_watcher qethconf scripts zconf \
- vmconvert vmcp man mon_tools dasdinfo vmur cpuplugd ipl_tools \
- ziomon iucvterm hyptop cmsfs-fuse qethqoat zfcpdump zdsfs cpumf \
-- systemd hmcdrvfs cpacfstats zdev
-+ systemd hmcdrvfs cpacfstats zdev dump2tar
-
- all: subdirs_make
-
---- a/README
-+++ b/README
-@@ -210,6 +210,11 @@ s390-tools (1.34.0)
- configuration of devices and device drivers which are specific to the s390
- platform.
-
-+ * dump2tar:
-+ dump2tar is a tool for creating a tar archive from the contents of
-+ arbitrary files. It works even when the size of the actual file content
-+ is not known beforehand (e.g. FIFOs, sysfs files).
-+
- For more information refer to the following publications:
- * "Device Drivers, Features, and Commands" chapter "Useful Linux commands"
- * "Using the dump tools"
---- /dev/null
-+++ b/dump2tar/Makefile
-@@ -0,0 +1,12 @@
-+# Common definitions
-+include ../common.mak
-+
-+all:
-+ $(MAKE) -C src
-+
-+install: all
-+ $(MAKE) -C src install
-+ $(MAKE) -C man install
-+
-+clean:
-+ $(MAKE) -C src clean
---- /dev/null
-+++ b/dump2tar/include/buffer.h
-@@ -0,0 +1,47 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Data buffering functions
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef BUFFER_H
-+#define BUFFER_H
-+
-+#include
-+#include
-+#include
-+
-+/* Buffers for building tar file entries */
-+struct buffer {
-+ size_t total; /* Total number of bytes in buffer */
-+ size_t off; /* Current offset to next free byte in memory buffer */
-+ size_t size; /* Memory buffer size */
-+ char *addr; /* Memory buffer address */
-+ bool fd_open; /* Has fd been openend yet? */
-+ FILE *file; /* FILE * of file containing previous buffer data */
-+ int fd; /* Handle of file containing previous buffer data */
-+};
-+
-+void buffer_init(struct buffer *buffer, size_t size);
-+struct buffer *buffer_alloc(size_t size);
-+void buffer_reset(struct buffer *buffer);
-+void buffer_close(struct buffer *buffer);
-+void buffer_free(struct buffer *buffer, bool dyn);
-+int buffer_open(struct buffer *buffer);
-+int buffer_flush(struct buffer *buffer);
-+ssize_t buffer_make_room(struct buffer *buffer, size_t size, bool usefile,
-+ size_t max_buffer_size);
-+int buffer_truncate(struct buffer *buffer, size_t len);
-+
-+ssize_t buffer_read_fd(struct buffer *buffer, int fd, size_t chunk,
-+ bool usefile, size_t max_buffer_size);
-+int buffer_add_data(struct buffer *buffer, char *addr, size_t len,
-+ bool usefile, size_t max_buffer_size);
-+
-+typedef int (*buffer_cb_t)(void *data, void *addr, size_t len);
-+int buffer_iterate(struct buffer *buffer, buffer_cb_t cb, void *data);
-+void buffer_print(struct buffer *buffer);
-+
-+#endif /* BUFFER_H */
---- /dev/null
-+++ b/dump2tar/include/dref.h
-@@ -0,0 +1,26 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Reference counting for directory handles
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef DREF_H
-+#define DREF_H
-+
-+#include
-+#include
-+
-+/* Multiple jobs may refer to an open DIR * - need reference counting */
-+struct dref {
-+ DIR *dd;
-+ int dirfd;
-+ unsigned int count;
-+};
-+
-+struct dref *dref_create(const char *dirname);
-+struct dref *dref_get(struct dref *dref);
-+void dref_put(struct dref *dref);
-+
-+#endif /* DREF_H */
---- /dev/null
-+++ b/dump2tar/include/dump.h
-@@ -0,0 +1,60 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Main dump logic
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef DUMP_H
-+#define DUMP_H
-+
-+#include
-+#include
-+#include
-+
-+#include "strarray.h"
-+
-+#define NUM_EXCLUDE_TYPES 7
-+
-+struct dump_spec {
-+ char *inname;
-+ char *outname;
-+ bool is_cmd;
-+};
-+
-+struct dump_opts {
-+ bool add_cmd_status;
-+ bool append;
-+ bool dereference;
-+ bool exclude_type[NUM_EXCLUDE_TYPES];
-+ bool gzip;
-+ bool ignore_failed_read;
-+ bool no_eof;
-+ bool quiet;
-+ bool recursive;
-+ bool threaded;
-+ bool verbose;
-+ const char *output_file;
-+ int file_timeout;
-+ int timeout;
-+ long jobs;
-+ long jobs_per_cpu;
-+ size_t file_max_size;
-+ size_t max_buffer_size;
-+ size_t max_size;
-+ size_t read_chunk_size;
-+ struct strarray exclude;
-+ struct dump_spec *specs;
-+ unsigned int num_specs;
-+};
-+
-+struct dump_opts *dump_opts_new(void);
-+int dump_opts_set_type_excluded(struct dump_opts *opts, char c);
-+void dump_opts_add_spec(struct dump_opts *opts, char *inname, char *outname,
-+ bool is_cmd);
-+void dump_opts_free(struct dump_opts *opts);
-+
-+int dump_to_tar(struct dump_opts *opts);
-+
-+#endif /* DUMP_H */
---- /dev/null
-+++ b/dump2tar/include/global.h
-@@ -0,0 +1,20 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Global variables
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef GLOBAL_H
-+#define GLOBAL_H
-+
-+#include
-+
-+extern bool global_threaded;
-+extern bool global_debug;
-+extern bool global_verbose;
-+extern bool global_quiet;
-+extern bool global_timestamps;
-+
-+#endif /* GLOBAL_H */
---- /dev/null
-+++ b/dump2tar/include/idcache.h
-@@ -0,0 +1,23 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Caches for user and group ID lookups
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef IDCACHE_H
-+#define IDCACHE_H
-+
-+#include
-+#include
-+
-+/* Buffer sizes for getpwuid_r and getgid_r calls (bytes) */
-+#define PWD_BUFFER_SIZE 4096
-+#define GRP_BUFFER_SIZE 4096
-+
-+void uid_to_name(uid_t uid, char *name, size_t len);
-+void gid_to_name(gid_t gid, char *name, size_t len);
-+void idcache_cleanup(void);
-+
-+#endif /* IDCACHE_H */
---- /dev/null
-+++ b/dump2tar/include/misc.h
-@@ -0,0 +1,98 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Helper functions
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef MISC_H
-+#define MISC_H
-+
-+#include
-+#include
-+#include
-+#include
-+
-+#include "global.h"
-+
-+#include "util_libc.h"
-+
-+#define MSG_LEN 256
-+
-+#define DBG(...) \
-+ do { \
-+ if (global_debug) \
-+ debug(__FILE__, __LINE__, ##__VA_ARGS__); \
-+ } while (0)
-+
-+#define mwarn(fmt, ...) _mwarn(true, (fmt), ##__VA_ARGS__)
-+#define mwarnx(fmt, ...) _mwarn(false, (fmt), ##__VA_ARGS__)
-+
-+/* Helper macro for constructing messages in variables */
-+#define HANDLE_RC(rc, max, off, label) \
-+ do { \
-+ if ((rc) > 0) \
-+ (off) += (rc); \
-+ if ((off) > (max)) \
-+ goto label; \
-+ } while (0)
-+
-+/* Program exit codes */
-+#define EXIT_OK 0
-+#define EXIT_RUNTIME 1
-+#define EXIT_USAGE 2
-+
-+/* Number of nanoseconds in a second */
-+#define NSEC_PER_SEC 1000000000L
-+#define NSEC_PER_MSEC 1000000L
-+#define NSEC_PER_USEC 1000L
-+
-+extern struct timespec main_start_ts;
-+struct dref;
-+
-+int misc_write_data(int fd, char *addr, size_t len);
-+ssize_t misc_read_data(int fd, char *addr, size_t len);
-+void inc_timespec(struct timespec *ts, time_t sec, long nsec);
-+void set_timespec(struct timespec *ts, time_t sec, long nsec);
-+bool ts_before(struct timespec *a, struct timespec *b);
-+int snprintf_duration(char *buff, size_t len, struct timespec *start,
-+ struct timespec *end);
-+char *get_threadname(void);
-+void debug(const char *file, unsigned long line, const char *format, ...);
-+void _mwarn(bool print_errno, const char *format, ...);
-+void verb(const char *format, ...);
-+void info(const char *format, ...);
-+#define mmalloc(len) util_zalloc(len)
-+#define mcalloc(n, len) util_zalloc((n) * (len))
-+#define mrealloc(ptr, len) util_realloc((ptr), (len))
-+#define mstrdup(str) util_strdup(str)
-+#define masprintf(fmt, ...) __masprintf(__func__, __FILE__, __LINE__, \
-+ (fmt), ##__VA_ARGS__)
-+char *__masprintf(const char *func, const char *file, int line,
-+ const char *fmt, ...);
-+#define set_threadname(fmt, ...) __set_threadname(__func__, __FILE__, \
-+ __LINE__, (fmt), \
-+ ##__VA_ARGS__)
-+void __set_threadname(const char *func, const char *file, int line,
-+ const char *fmt, ...);
-+
-+void clear_threadname(void);
-+void chomp(char *str, char *c);
-+void lchomp(char *str, char *c);
-+void remove_double_slashes(char *str);
-+int stat_file(bool dereference, const char *abs, const char *rel,
-+ struct dref *dref, struct stat *st);
-+void set_dummy_stat(struct stat *st);
-+bool starts_with(const char *str, const char *prefix);
-+bool ends_with(const char *str, const char *suffix);
-+
-+int cmd_child(int fd, char *cmd);
-+int cmd_open(char *cmd, pid_t *pid_ptr);
-+int cmd_close(int fd, pid_t pid, int *status_ptr);
-+
-+void misc_init(void);
-+void misc_cleanup(void);
-+void set_stdout_data(void);
-+
-+#endif /* MISC_H */
---- /dev/null
-+++ b/dump2tar/include/strarray.h
-@@ -0,0 +1,23 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Dynamically growing string arrays
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef STRARRAY_H
-+#define STRARRAY_H
-+
-+/* A string array that can grow in size */
-+struct strarray {
-+ unsigned int num;
-+ char **str;
-+};
-+
-+void free_strarray(struct strarray *array);
-+void add_str_to_strarray(struct strarray *array, const char *str);
-+void add_vstr_to_strarray(struct strarray *array, const char *fmt, ...);
-+int add_file_to_strarray(struct strarray *array, const char *filename);
-+
-+#endif /* STRARRAY_H */
---- /dev/null
-+++ b/dump2tar/include/tar.h
-@@ -0,0 +1,41 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * TAR file generation
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#ifndef TAR_H
-+#define TAR_H
-+
-+#include
-+#include
-+#include
-+
-+#define TYPE_REGULAR '0'
-+#define TYPE_LINK '2'
-+#define TYPE_DIR '5'
-+
-+#define TAR_BLOCKSIZE 512
-+
-+struct buffer;
-+
-+/* emit_cb_t - Callback used for emitting chunks of a byte stream
-+ * @data: Arbitrary pointer passed via the @data parameter of the
-+ * tar_emit_file_* functions
-+ * @addr: Pointer to data
-+ * @len: Size of data
-+ * Return %0 on success. Returning non-zero will indicate failure and abort
-+ * further data emission. */
-+typedef int (*emit_cb_t)(void *data, void *addr, size_t len);
-+
-+int tar_emit_file_from_buffer(char *filename, char *link, size_t len,
-+ struct stat *stat, char type,
-+ struct buffer *content, emit_cb_t emit_cb,
-+ void *data);
-+int tar_emit_file_from_data(char *filename, char *link, size_t len,
-+ struct stat *stat, char type, void *addr,
-+ emit_cb_t emit_cb, void *data);
-+
-+#endif /* TAR_H */
---- /dev/null
-+++ b/dump2tar/man/Makefile
-@@ -0,0 +1,12 @@
-+# Common definitions
-+include ../../common.mak
-+
-+all:
-+
-+install:
-+ $(INSTALL) -d -m 755 $(DESTDIR)$(MANDIR)/man1
-+ $(INSTALL) -m 644 -c dump2tar.1 $(DESTDIR)$(MANDIR)/man1
-+
-+clean:
-+
-+.PHONY: all clean
---- /dev/null
-+++ b/dump2tar/man/dump2tar.1
-@@ -0,0 +1,454 @@
-+.\" Macro for inserting an option description prologue.
-+.\" .OD [] [args]
-+.de OD
-+. ds args "
-+. if !'\\$3'' .as args \fI\\$3\fP
-+. if !'\\$4'' .as args \\$4
-+. if !'\\$5'' .as args \fI\\$5\fP
-+. if !'\\$6'' .as args \\$6
-+. if !'\\$7'' .as args \fI\\$7\fP
-+. PD 0
-+. if !'\\$2'' .IP "\fB\-\\$2\fP \\*[args]" 4
-+. if !'\\$1'' .IP "\fB\-\-\\$1\fP \\*[args]" 4
-+. PD
-+..
-+.\" Macro for inserting code line.
-+.\" .CL
-+.de CL
-+. ds pfont \\n[.f]
-+. nh
-+. na
-+. ft CW
-+\\$*
-+. ft \\*[pfont]
-+. ad
-+. hy
-+. br
-+..
-+.\" Macro for inserting a man page reference.
-+.\" .MP man-page section [suffix]
-+.de MP
-+. nh
-+. na
-+. BR \\$1 (\\$2)\\$3
-+. ad
-+. hy
-+..
-+.
-+.TH "dump2tar" "1" "2016\-09\-02" "" ""
-+.
-+.SH "NAME"
-+dump2tar - Gather file contents and command output into a tar archive
-+.
-+.
-+.SH "SYNOPSIS"
-+.B "dump2tar "
-+.RI "[" "OPTIONS" "] " "SPECS"
-+.
-+.
-+.SH "DESCRIPTION"
-+.B dump2tar
-+creates a tar archive from the contents of any files, including files of
-+unknown size.
-+
-+Examples for files of unknown size are:
-+.IP \(bu 3
-+Named pipes (FIFOs)
-+.PP
-+.IP \(bu 3
-+Particular Linux kernel debugfs or sysfs files
-+.PP
-+.IP \(bu 3
-+Character or block devices
-+.PP
-+
-+When adding such a file,
-+.B dump2tar
-+first reads all available data until an end-of-file indication is found. From
-+this data, it then creates a regular file entry in the resulting tar archive.
-+By default, symbolic links and directories are preserved in the archive in
-+their original form.
-+
-+.B dump2tar
-+can also:
-+.IP \(bu 3
-+Add files under a different name
-+.PP
-+.IP \(bu 3
-+Run arbitrary commands and add the resulting command output as a
-+regular file
-+.PP
-+.
-+.
-+.SH "FILE SPECIFICATIONS"
-+.
-+This section describes the format of the
-+.I SPECS
-+argument mentioned in the command synopsis.
-+Use the following command line syntax to identify data sources and
-+to specify file names within the archive:
-+.PP
-+
-+.TP
-+.I "PATH"
-+Adds the contents of the file system subtree at file system location
-+.I PATH
-+(with possible exceptions described by options) in the archive under the same
-+file name as on the file system.
-+.PP
-+.
-+.
-+.TP
-+.IR "FILENAME" ":=" "PATH"
-+Adds the contents of the file at file system location
-+.I PATH
-+in the archive under the name specified by
-+.IR FILENAME .
-+.PP
-+.
-+.
-+.TP
-+.IR "FILENAME" "|=" "CMDLINE"
-+Runs the command
-+.IR CMDLINE
-+and captures both the resulting standard output and standard error streams.
-+Adds the collected output as a regular file named
-+.I FILENAME
-+in the resulting archive. You can also include the resulting program exit code
-+by using option \-\-add\-cmd\-status.
-+.PP
-+.
-+You can also specify "\-\-". All specifications that follow are interpreted as
-+simple file names. This is useful for archiving files that contain ":=" or "|=".
-+.PP
-+.
-+.
-+.SH "OUTPUT OPTIONS"
-+.
-+.OD "output\-file" "o" "TARFILE"
-+Writes the resulting tar archive to
-+.IR TARFILE .
-+An existing file at the specified file system location is overwritten.
-+
-+If this option is omitted or if "\-" is specified for
-+.IR TARFILE ,
-+the archive is written to the standard output stream.
-+.PP
-+.
-+.
-+.OD "gzip" "z" ""
-+Compresses the resulting tar archive using gzip.
-+.PP
-+.
-+.
-+.OD "max\-size" "m" "VALUE"
-+Sets an upper size limit, in bytes, for the resulting archive. If this limit
-+is exceeded after adding a file, no further files are added.
-+.PP
-+.
-+.
-+.OD "timeout" "t" "VALUE"
-+Sets an upper time limit, in seconds, for the archiving process. If this limit
-+is exceeded while adding a file, that file is truncated and no
-+further files are added.
-+.PP
-+.
-+.
-+.OD "no-eof" "" ""
-+Does not write an end-of-file marker.
-+
-+Use this option if you want to create an archive that can be extended by
-+appending additional tar archive data.
-+
-+Note: Do not use this option for the final data to be added.
-+A valid tar archive requires a trailing end-of-file marker.
-+.PP
-+.
-+.
-+.OD "append" "" ""
-+Appends data to the end of the archive.
-+
-+Use this option to incrementally build a tar file by repeatedly calling
-+.BR dump2tar .
-+You must specify the \-\-no\-eof option for each but the final call of
-+.BR dump2tar .
-+.PP
-+.
-+.
-+.OD "add-cmd-status" "" ""
-+Adds a separate file named
-+.RI \(dq FILENAME .cmdstatus\(dq
-+for each command output added through the
-+.RI \(dq FILENAME |= CMDLINE \(dq
-+notation (see FILE SPECIFICATIONS).
-+This file contains information about the exit status of the
-+process that executed the command:
-+.
-+.RS 8
-+.TP
-+.RI EXITSTATUS= VALUE
-+Unless
-+.I VALUE
-+is -1, the process ended normally with the specified exit value.
-+.PP
-+.
-+.TP
-+.RI TERMSIG= VALUE
-+Unless
-+.I VALUE
-+is -1, the process was stopped by a signal of the specified number.
-+.PP
-+.
-+.TP
-+.RI WAITPID_ERRNO= VALUE
-+Unless
-+.I VALUE
-+is -1, an attempt to obtain the status of the process failed with the
-+specified error.
-+.PP
-+.RE
-+.
-+.
-+.
-+.SH "INPUT OPTIONS"
-+.
-+.OD "files\-from" "F" "FILENAME"
-+Reads input data specifications (see FILE SPECIFICATIONS) from
-+.IR FILENAME ,
-+one specification per line. Each line contains either a file name or a
-+.IR FILENAME := PATH
-+or
-+.IR FILENAME |= CMDLINE
-+specification. Empty lines are ignored.
-+
-+A line can also consist of only "\-\-". All lines following this specification
-+are interpreted as simple file names. This is useful for archiving files that
-+contain ":=" or "|=".
-+.PP
-+.
-+.
-+.OD "ignore\-failed\-read" "i" ""
-+Continues after read errors.
-+
-+By default,
-+.B dump2tar
-+stops processing after encountering errors while reading an input file.
-+With this option,
-+.B dump2tar
-+prints a warning message and adds an empty entry for the erroneous file in
-+the archive.
-+.PP
-+.
-+.
-+.OD "buffer\-size" "b" "VALUE"
-+Reads data from input files in chunks of
-+.I VALUE
-+bytes. Large values can accelerate the archiving process for large files
-+at the cost of increased memory usage. The default value is 1048576.
-+.PP
-+.
-+.
-+.OD "file\-timeout" "T" "VALUE"
-+Sets an upper time limit, in seconds, for reading an input file.
-+
-+.B dump2tar
-+stops processing a file when the time limit is exceeded. Archive entries for
-+such files are truncated to the amount of data that is collected by the time
-+the limit is reached.
-+.PP
-+.
-+.
-+.OD "file\-max\-size" "M" "N"
-+Sets an upper size limit, in bytes, for an input file.
-+
-+.B dump2tar
-+stops processing a file when the size limit is exceeded. Archive entries for
-+such files are truncated to the specified size.
-+.PP
-+.
-+.
-+.OD "jobs" "j" "N"
-+By default,
-+.B dump2tar
-+processes one file at a time. With this option,
-+.B dump2tar
-+processes
-+.I N
-+files in parallel.
-+
-+Parallel processing can accelerate the archiving process,
-+especially if input files are located on slow devices, or when output from
-+multiple commands is added to the archive.
-+
-+Note: Use
-+.B tar
-+option \-\-delay\-directory\-restore when extracting files from an archive
-+created with \-\-jobs to prevent conflicts with directory permissions and
-+modification times.
-+.PP
-+.
-+.
-+.OD "jobs\-per\-cpu" "J" "N"
-+Processes
-+.I N
-+files for each online CPU in parallel.
-+
-+Parallel processing can accelerate the
-+archiving process, especially if input files are located on slow devices, or
-+when output from multiple commands is added to the archive.
-+
-+Note: Use
-+.B tar
-+option \-\-delay\-directory\-restore when extracting files from an archive
-+created with \-\-jobs\-per\-cpu to prevent conflicts with directory permissions
-+and modification times.
-+.PP
-+.
-+.
-+.OD "exclude" "x" "PATTERN"
-+Does not add files to the archive if their file names match
-+.IR PATTERN .
-+.I PATTERN
-+is an expression that uses the shell wildcards.
-+.PP
-+.
-+.
-+.OD "exclude\-from" "X" "FILENAME"
-+Does not add files to the archive if their names match at least one of the
-+patterns listed in the pattern file with name
-+.IR FILENAME .
-+In the pattern file, each line specifies an expression that uses the
-+shell wildcards.
-+.PP
-+.
-+.
-+.OD "exclude\-type" "" "TYPE"
-+Does not add files to the archive if they match at least one of the file types
-+specified with
-+.IR TYPE .
-+.I TYPE
-+uses one or more of the characters "fdcbpls", where:
-+
-+.RS 8
-+.IP f 3
-+regular files
-+.PP
-+.IP d 3
-+directories
-+.PP
-+.IP c 3
-+character devices
-+.PP
-+.IP b 3
-+block devices
-+.PP
-+.IP p 3
-+named pipes (FIFOs)
-+.PP
-+.IP l 3
-+symbolic links
-+.PP
-+.IP s 3
-+sockets
-+.PP
-+.RE
-+.
-+.PP
-+.
-+.
-+.OD "dereference" "" ""
-+Adds the content of link targets instead of symbolic links.
-+.PP
-+.
-+.
-+.OD "no\-recursion" "" ""
-+Does not add files from sub\-directories.
-+
-+By default,
-+.B dump2tar
-+adds archive entries for specified directories, and for the files within these
-+directories. With this option, a specified directory results in a single entry
-+for the directory. Any contained files to be included must be specified
-+explicitly.
-+.PP
-+.
-+.
-+.SH "MISC OPTIONS"
-+.
-+.OD "help" "h" ""
-+Prints an overview of available options, then exits.
-+.PP
-+.
-+.
-+.OD "verbose" "V" ""
-+Prints additional informational output.
-+.PP
-+.
-+.
-+.OD "quiet" "q" ""
-+Suppresses printing of informational output.
-+.PP
-+.
-+.
-+.
-+.SH "EXAMPLES"
-+.
-+.\fB
-+.CL # dump2tar a b \-o archive.tar
-+.\fR
-+
-+.RS 4
-+Creates a tar archive named archive.tar containing files a and b.
-+.RE
-+.PP
-+.
-+.\fB
-+.CL # dump2tar /proc \-o procdump.tar.gz \-z \-i \-T 1 \-M 1048576
-+.\fR
-+
-+.RS 4
-+Creates a gzip compressed tar archive named procdump.tar.gz that contains
-+all procfs files. Unreadable files are ignored. Files are truncated when the
-+first of the two limiting conditions is reached, either 1048576 bytes of
-+content or the reading time of 1 second.
-+.RE
-+.PP
-+.
-+.\fB
-+.CL # dump2tar '|=dmesg' '|=lspci' \-o data.tar
-+.\fR
-+
-+.RS 4
-+Creates a tar archive named data.tar containing the output of the 'dmesg'
-+and 'lspci' commands.
-+.RE
-+.PP
-+.
-+.\fB
-+.CL # dump2tar /sys/kernel/debug/ -x '*/tracing/*' -o debug.tar -i
-+.\fR
-+
-+.RS 4
-+Creates a tar archive named debug.tar containing the contents of directory
-+/sys/kernel/debug/ while excluding any file that is located in a sub-directory
-+named 'tracing'.
-+.RE
-+.PP
-+.
-+.
-+.SH "EXIT CODES"
-+.TP
-+.B 0
-+The program finished successfully
-+.TP
-+.B 1
-+A run-time error occurred
-+.TP
-+.B 2
-+The specified command was not valid
-+.PP
-+.
-+.
-+.SH "SEE ALSO"
-+.MP dump2tar 1 ,
-+.MP tar 1
---- /dev/null
-+++ b/dump2tar/src/Makefile
-@@ -0,0 +1,36 @@
-+# Common definitions
-+include ../../common.mak
-+
-+CPPFLAGS += -I ../../include -I../include -std=gnu99 -Wno-unused-parameter
-+LDLIBS += -lpthread -lrt
-+ifneq ($(HAVE_ZLIB),0)
-+CPPFLAGS += -DHAVE_ZLIB
-+LDLIBS += -lz
-+endif
-+
-+core_objects = buffer.o dref.o global.o dump.o idcache.o misc.o strarray.o tar.o
-+libs = $(rootdir)/libutil/util_libc.o $(rootdir)/libutil/util_opt.o \
-+ $(rootdir)/libutil/util_prg.o $(rootdir)/libutil/util_panic.o
-+
-+check_dep_zlib:
-+ $(call check_dep, \
-+ "dump2tar", \
-+ "zlib.h", \
-+ "zlib-devel", \
-+ "HAVE_ZLIB=0")
-+
-+all: check_dep_zlib dump2tar
-+
-+dump2tar: $(core_objects) dump2tar.o $(libs)
-+
-+install: dump2tar
-+ $(INSTALL) -c dump2tar $(DESTDIR)$(USRBINDIR)
-+
-+clean:
-+ @rm -f dump2tar *.o
-+
-+.PHONY: all install clean
-+
-+# Additional manual dependencies
-+../../libutil/%.o:
-+ make -C ../../libutil $<
---- /dev/null
-+++ b/dump2tar/src/buffer.c
-@@ -0,0 +1,271 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Data buffering functions
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#include "buffer.h"
-+
-+#include
-+#include
-+#include
-+#include
-+#include
-+
-+#include "misc.h"
-+
-+void buffer_print(struct buffer *buffer)
-+{
-+ fprintf(stderr, "DEBUG: buffer at %p\n", (void *) buffer);
-+ if (!buffer)
-+ return;
-+ fprintf(stderr, "DEBUG: total=%zu\n", buffer->total);
-+ fprintf(stderr, "DEBUG: off=%zu\n", buffer->off);
-+ fprintf(stderr, "DEBUG: size=%zu\n", buffer->size);
-+ fprintf(stderr, "DEBUG: addr=%p\n", (void *) buffer->addr);
-+ fprintf(stderr, "DEBUG: fd_open=%d\n", buffer->fd_open);
-+ fprintf(stderr, "DEBUG: fd=%d\n", buffer->fd);
-+ if (buffer->fd_open) {
-+ fprintf(stderr, "DEBUG: fd->pos=%zu\n",
-+ lseek(buffer->fd, 0, SEEK_CUR));
-+ }
-+}
-+
-+/* Initialize @buffer to hold @size bytes in memory */
-+void buffer_init(struct buffer *buffer, size_t size)
-+{
-+ memset(buffer, 0, sizeof(struct buffer));
-+ buffer->addr = mmalloc(size);
-+ buffer->size = size;
-+}
-+
-+/* Allocate a new buffer for holding @size bytes in memory */
-+struct buffer *buffer_alloc(size_t size)
-+{
-+ struct buffer *buffer;
-+
-+ buffer = mmalloc(sizeof(struct buffer));
-+ buffer_init(buffer, size);
-+
-+ return buffer;
-+}
-+
-+/* Forget about any data stored in @buffer */
-+void buffer_reset(struct buffer *buffer)
-+{
-+ buffer->total = 0;
-+ buffer->off = 0;
-+ if (buffer->fd_open) {
-+ if (ftruncate(buffer->fd, 0))
-+ mwarn("Cannot truncate temporary file");
-+ if (lseek(buffer->fd, 0, SEEK_SET) == (off_t) -1)
-+ mwarn("Cannot seek in temporary file");
-+ }
-+}
-+
-+/* Close buffer file associated with @buffer */
-+void buffer_close(struct buffer *buffer)
-+{
-+ if (!buffer->fd_open)
-+ return;
-+
-+ fclose(buffer->file);
-+ buffer->fd = 0;
-+ buffer->fd_open = false;
-+}
-+
-+/* Release all resources associated with @buffer. If @dyn is %true, also free
-+ * @buffer itself. */
-+void buffer_free(struct buffer *buffer, bool dyn)
-+{
-+ if (!buffer)
-+ return;
-+ buffer_reset(buffer);
-+ buffer_close(buffer);
-+ free(buffer->addr);
-+ if (dyn)
-+ free(buffer);
-+}
-+
-+/* Open a buffer file for @buffer. Return %EXIT_OK on success, %EXIT_RUNTIME
-+ * otherwise. */
-+int buffer_open(struct buffer *buffer)
-+{
-+ if (buffer->fd_open)
-+ return EXIT_OK;
-+
-+ buffer->file = tmpfile();
-+ if (!buffer->file) {
-+ mwarn("Could not create temporary file");
-+ return EXIT_RUNTIME;
-+ }
-+
-+ buffer->fd = fileno(buffer->file);
-+ buffer->fd_open = true;
-+
-+ return EXIT_OK;
-+}
-+
-+/* Write data in memory of @buffer to buffer file. Return %EXIT_OK on success,
-+ * %EXIT_RUNTIME otherwise. */
-+int buffer_flush(struct buffer *buffer)
-+{
-+ if (buffer->off == 0)
-+ return EXIT_OK;
-+ if (buffer_open(buffer))
-+ return EXIT_RUNTIME;
-+ if (misc_write_data(buffer->fd, buffer->addr, buffer->off)) {
-+ mwarn("Could not write to temporary file");
-+ return EXIT_RUNTIME;
-+ }
-+ buffer->off = 0;
-+
-+ return EXIT_OK;
-+}
-+
-+/* Try to ensure that at least @size bytes are available at
-+ * @buffer->addr[buffer->off]. Return the actual number of bytes available or
-+ * @-1 on error. If @usefile is %true, make use of a buffer file if
-+ * the total buffer size exceeds @max_buffer_size. */
-+ssize_t buffer_make_room(struct buffer *buffer, size_t size, bool usefile,
-+ size_t max_buffer_size)
-+{
-+ size_t needsize;
-+
-+ if (size > max_buffer_size && usefile)
-+ size = max_buffer_size;
-+
-+ needsize = buffer->off + size;
-+ if (needsize <= buffer->size) {
-+ /* Room available */
-+ return size;
-+ }
-+
-+ if (needsize > max_buffer_size && usefile) {
-+ /* Need to write out memory buffer to buffer file */
-+ if (buffer_flush(buffer))
-+ return -1;
-+ if (size <= buffer->size)
-+ return size;
-+ needsize = size;
-+ }
-+
-+ /* Need to increase memory buffer size */
-+ buffer->size = needsize;
-+ buffer->addr = mrealloc(buffer->addr, buffer->size);
-+
-+ return size;
-+}
-+
-+/* Try to read @chunk bytes from @fd to @buffer. Return the number of bytes
-+ * read on success, %0 on EOF or %-1 on error. */
-+ssize_t buffer_read_fd(struct buffer *buffer, int fd, size_t chunk,
-+ bool usefile, size_t max_buffer_size)
-+{
-+ ssize_t c = buffer_make_room(buffer, chunk, usefile, max_buffer_size);
-+
-+ DBG("buffer_read_fd wanted %zd got %zd", chunk, c);
-+ if (c < 0)
-+ return c;
-+
-+ c = read(fd, buffer->addr + buffer->off, c);
-+ if (c > 0) {
-+ buffer->total += c;
-+ buffer->off += c;
-+ }
-+
-+ return c;
-+}
-+
-+/* Add @len bytes at @addr to @buffer. If @addr is %NULL, add zeroes. Return
-+ * %EXIT_OK on success, %EXIT_RUNTIME otherwise. */
-+int buffer_add_data(struct buffer *buffer, char *addr, size_t len, bool usefile,
-+ size_t max_buffer_size)
-+{
-+ ssize_t c;
-+
-+ while (len > 0) {
-+ c = buffer_make_room(buffer, len, usefile, max_buffer_size);
-+ if (c < 0)
-+ return EXIT_RUNTIME;
-+ if (addr) {
-+ memcpy(buffer->addr + buffer->off, addr, c);
-+ addr += c;
-+ } else {
-+ memset(buffer->addr + buffer->off, 0, c);
-+ }
-+ buffer->total += c;
-+ buffer->off += c;
-+
-+ len -= c;
-+ }
-+
-+ return EXIT_OK;
-+}
-+
-+/* Call @cb for all chunks of data in @buffer. @data is passed to @cb. */
-+int buffer_iterate(struct buffer *buffer, buffer_cb_t cb, void *data)
-+{
-+ int rc;
-+ ssize_t r;
-+
-+ if (buffer->total == 0)
-+ return EXIT_OK;
-+
-+ if (!buffer->fd_open)
-+ return cb(data, buffer->addr, buffer->off);
-+
-+ /* Free memory buffer to be used as copy buffer */
-+ if (buffer_flush(buffer))
-+ return EXIT_RUNTIME;
-+ if (lseek(buffer->fd, 0, SEEK_SET) == (off_t) -1) {
-+ mwarn("Cannot seek in temporary file");
-+ return EXIT_RUNTIME;
-+ }
-+
-+ /* Copy data from temporary file to target file */
-+ while ((r = misc_read_data(buffer->fd, buffer->addr,
-+ buffer->size)) != 0) {
-+ if (r < 0) {
-+ mwarn("Cannot read from temporary file");
-+ return EXIT_RUNTIME;
-+ }
-+ rc = cb(data, buffer->addr, r);
-+ if (rc)
-+ return rc;
-+ }
-+
-+ return EXIT_OK;
-+}
-+
-+/* Truncate @buffer to at most @len bytes */
-+int buffer_truncate(struct buffer *buffer, size_t len)
-+{
-+ size_t delta;
-+
-+ if (buffer->total <= len)
-+ return EXIT_OK;
-+
-+ delta = buffer->total - len;
-+
-+ buffer->total = len;
-+ if (buffer->fd_open && delta > buffer->off) {
-+ /* All of memory and some of file buffer is truncated */
-+ buffer->off = 0;
-+ if (ftruncate(buffer->fd, len)) {
-+ mwarn("Cannot truncate temporary file");
-+ return EXIT_RUNTIME;
-+ }
-+ if (lseek(buffer->fd, len, SEEK_SET) == (off_t) -1) {
-+ mwarn("Cannot seek in temporary file");
-+ return EXIT_RUNTIME;
-+ }
-+ } else {
-+ /* Only memory buffer is truncated */
-+ buffer->off -= delta;
-+ }
-+
-+ return EXIT_OK;
-+}
---- /dev/null
-+++ b/dump2tar/src/dref.c
-@@ -0,0 +1,92 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Reference counting for directory handles
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#include "dref.h"
-+
-+#include
-+#include
-+#include
-+
-+#include "global.h"
-+#include "misc.h"
-+
-+/* dref_mutex serializes access to drefs */
-+static pthread_mutex_t dref_mutex = PTHREAD_MUTEX_INITIALIZER;
-+
-+static unsigned long num_open_dirs;
-+static unsigned long num_open_dirs_max;
-+
-+/* Lock dref mutex */
-+static void dref_lock(void)
-+{
-+ if (!global_threaded)
-+ return;
-+ pthread_mutex_lock(&dref_mutex);
-+}
-+
-+/* Unlock dref mutex */
-+static void dref_unlock(void)
-+{
-+ if (!global_threaded)
-+ return;
-+ pthread_mutex_unlock(&dref_mutex);
-+}
-+
-+/* Create a reference count managed directory handle for @dirname */
-+struct dref *dref_create(const char *dirname)
-+{
-+ struct dref *dref;
-+ DIR *dd;
-+
-+ dd = opendir(dirname);
-+ DBG("opendir(%s)=%p (total=%lu)", dirname, dd, ++num_open_dirs);
-+ if (!dd) {
-+ num_open_dirs--;
-+ return NULL;
-+ }
-+
-+ if (num_open_dirs > num_open_dirs_max)
-+ num_open_dirs_max = num_open_dirs;
-+
-+ dref = mmalloc(sizeof(struct dref));
-+ dref->dd = dd;
-+ dref->dirfd = dirfd(dd);
-+ dref->count = 1;
-+
-+ return dref;
-+}
-+
-+/* Obtain a reference to @dref */
-+struct dref *dref_get(struct dref *dref)
-+{
-+ if (dref) {
-+ dref_lock();
-+ dref->count++;
-+ dref_unlock();
-+ }
-+
-+ return dref;
-+}
-+
-+/* Release a reference to @dref. If this was the last reference, lose the
-+ * associated directory handle and free @dref. */
-+void dref_put(struct dref *dref)
-+{
-+ if (dref) {
-+ dref_lock();
-+ dref->count--;
-+ if (dref->count == 0) {
-+ num_open_dirs--;
-+ DBG("closedir(%p) (total=%lu, max=%lu)", dref->dd,
-+ num_open_dirs, num_open_dirs_max);
-+ closedir(dref->dd);
-+ free(dref);
-+ }
-+ dref_unlock();
-+ }
-+}
---- /dev/null
-+++ b/dump2tar/src/dump.c
-@@ -0,0 +1,1850 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Main dump logic
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#include "dump.h"
-+
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+
-+#ifdef HAVE_ZLIB
-+#include
-+#endif /* HAVE_ZLIB */
-+
-+#include "buffer.h"
-+#include "dref.h"
-+#include "global.h"
-+#include "idcache.h"
-+#include "misc.h"
-+#include "tar.h"
-+
-+/* Default input file read size (bytes) */
-+#define DEFAULT_READ_CHUNK_SIZE (512 * 1024)
-+#define DEFAULT_MAX_BUFFER_SIZE (2 * 1024 * 1024)
-+
-+#define _SET_ABORTED(task) _set_aborted((task), __func__, __LINE__)
-+#define SET_ABORTED(task) set_aborted((task), __func__, __LINE__)
-+
-+#define read_error(task, filename, fmt, ...) \
-+ do { \
-+ if (!(task)->opts->ignore_failed_read) \
-+ SET_ABORTED((task)); \
-+ _mwarn(true, "%s: " fmt, (filename), ##__VA_ARGS__); \
-+ } while (0)
-+
-+#define write_error(task, fmt, ...) \
-+ do { \
-+ SET_ABORTED((task)); \
-+ _mwarn(true, "%s: " fmt, (task)->opts->output_file, \
-+ ##__VA_ARGS__); \
-+ } while (0)
-+
-+#define tverb(fmt, ...) \
-+ do { \
-+ if (task->opts->verbose) \
-+ verb((fmt), ##__VA_ARGS__); \
-+ } while (0)
-+
-+
-+/* Jobs representing a file or command output to add */
-+struct job {
-+ struct job *next_job;
-+ enum job_type {
-+ JOB_INIT, /* Initialization work */
-+ JOB_FILE, /* Add a regular file */
-+ JOB_LINK, /* Add a symbolic link */
-+ JOB_DIR, /* Add a directory */
-+ JOB_CMD, /* Add command output */
-+ } type;
-+ enum job_status {
-+ JOB_QUEUED, /* Transient: Job processing has not started */
-+ JOB_IN_PROGRESS,/* Transient: Job processing has started */
-+ JOB_EXCLUDED, /* Final: File was excluded */
-+ JOB_FAILED, /* Final: Data could not be obtained */
-+ JOB_DONE, /* Final: All data was obtained */
-+ JOB_PARTIAL, /* Final: Only some data was obtained */
-+ } status;
-+ char *outname;
-+ char *inname;
-+ char *relname;
-+ struct stat stat;
-+ bool timed;
-+ struct timespec deadline;
-+ struct dref *dref;
-+ int cmd_status;
-+ struct buffer *content;
-+};
-+
-+/* Run-time statistics */
-+struct stats {
-+ unsigned long num_done;
-+ unsigned long num_excluded;
-+ unsigned long num_failed;
-+ unsigned long num_partial;
-+};
-+
-+/* Information specific to a single dump task */
-+struct task {
-+ /* Input */
-+ struct dump_opts *opts;
-+
-+ /* State */
-+
-+ /* mutex serializes access to global data */
-+ pthread_mutex_t mutex;
-+ pthread_cond_t worker_cond;
-+ pthread_cond_t cond;
-+ unsigned long num_jobs_active;
-+ struct job *jobs_head;
-+ struct job *jobs_tail;
-+ bool aborted;
-+
-+ /* output_mutex serializes access to output file */
-+ pthread_mutex_t output_mutex;
-+ int output_fd;
-+ size_t output_written;
-+#ifdef HAVE_ZLIB
-+ gzFile output_gzfd;
-+#endif /* HAVE_ZLIB */
-+ unsigned long output_num_files;
-+
-+ /* No protection needed (only accessed in single-threaded mode) */
-+ struct stats stats;
-+ struct timespec start_ts;
-+};
-+
-+/* Per thread management data */
-+struct per_thread {
-+ long num;
-+ pthread_t thread;
-+ bool running;
-+ bool timed_out;
-+ struct stats stats;
-+ struct job *job;
-+ struct buffer buffer;
-+ struct task *task;
-+};
-+
-+static const struct {
-+ mode_t mode;
-+ char c;
-+} exclude_types[NUM_EXCLUDE_TYPES] = {
-+ { S_IFREG, 'f' },
-+ { S_IFDIR, 'd' },
-+ { S_IFCHR, 'c' },
-+ { S_IFBLK, 'b' },
-+ { S_IFIFO, 'p' },
-+ { S_IFLNK, 'l' },
-+ { S_IFSOCK, 's' },
-+};
-+
-+/* Lock main mutex */
-+static void main_lock(struct task *task)
-+{
-+ if (!global_threaded)
-+ return;
-+ DBG("main lock");
-+ pthread_mutex_lock(&task->mutex);
-+}
-+
-+/* Unlock main mutex */
-+static void main_unlock(struct task *task)
-+{
-+ if (!global_threaded)
-+ return;
-+ DBG("main unlock");
-+ pthread_mutex_unlock(&task->mutex);
-+}
-+
-+/* Lock output mutex */
-+static void output_lock(struct task *task)
-+{
-+ if (!global_threaded)
-+ return;
-+ pthread_mutex_lock(&task->output_mutex);
-+}
-+
-+/* Unlock output mutex */
-+static void output_unlock(struct task *task)
-+{
-+ if (!global_threaded)
-+ return;
-+ pthread_mutex_unlock(&task->output_mutex);
-+}
-+
-+/* Wake up all waiting workers */
-+static void _worker_wakeup_all(struct task *task)
-+{
-+ if (!global_threaded)
-+ return;
-+ DBG("waking up all worker threads");
-+ pthread_cond_broadcast(&task->worker_cond);
-+}
-+
-+/* Wake up one waiting worker */
-+static void _worker_wakeup_one(struct task *task)
-+{
-+ if (!global_threaded)
-+ return;
-+ DBG("waking up one worker thread");
-+ pthread_cond_signal(&task->worker_cond);
-+}
-+
-+/* Wait for a signal to a worker */
-+static int _worker_wait(struct task *task)
-+{
-+ int rc;
-+
-+ DBG("waiting for signal to worker");
-+ rc = pthread_cond_wait(&task->worker_cond, &task->mutex);
-+ DBG("waiting for signal to worker done (rc=%d)", rc);
-+
-+ return rc;
-+}
-+
-+/* Wake up main thread */
-+static void _main_wakeup(struct task *task)
-+{
-+ if (!global_threaded)
-+ return;
-+ DBG("waking up main thread");
-+ pthread_cond_broadcast(&task->cond);
-+}
-+
-+/* Wait for a signal to the main thread */
-+static int _main_wait(struct task *task)
-+{
-+ int rc;
-+
-+ DBG("waiting for status change");
-+ rc = pthread_cond_wait(&task->cond, &task->mutex);
-+ DBG("waiting for status change done (rc=%d)", rc);
-+
-+ return rc;
-+}
-+
-+/* Wait for a signal to the main thread. Abort waiting after @deadline */
-+static int _main_wait_timed(struct task *task, struct timespec *deadline)
-+{
-+ int rc;
-+
-+ DBG("timed waiting for status change");
-+ rc = pthread_cond_timedwait(&task->cond, &task->mutex, deadline);
-+ DBG("timed waiting for status change done (rc=%d)", rc);
-+
-+ return rc;
-+}
-+
-+/* Allow thread to be canceled */
-+static void cancel_enable(void)
-+{
-+ if (!global_threaded)
-+ return;
-+ if (pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL) != 0)
-+ mwarn("pthread_setcancelstate");
-+}
-+
-+/* Prevent thread from being canceled */
-+static void cancel_disable(void)
-+{
-+ if (!global_threaded)
-+ return;
-+ if (pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, NULL) != 0)
-+ mwarn("pthread_setcancelstate");
-+}
-+
-+/* Abort processing and inform all threads to shutdown. Must be called with
-+ * task->mutex locked */
-+static void _set_aborted(struct task *task, const char *func, unsigned int line)
-+{
-+ DBG("set aborted at %s:%u", func, line);
-+ task->aborted = true;
-+ _worker_wakeup_all(task);
-+ _main_wakeup(task);
-+}
-+
-+/* Abort processing and inform all threads to shutdown */
-+static void set_aborted(struct task *task, const char *func, unsigned int line)
-+{
-+ main_lock(task);
-+ _set_aborted(task, func, line);
-+ main_unlock(task);
-+}
-+
-+/* Check if abort processing has been initiated */
-+static bool is_aborted(struct task *task)
-+{
-+ bool result;
-+
-+ main_lock(task);
-+ result = task->aborted;
-+ main_unlock(task);
-+
-+ return result;
-+}
-+
-+/* Release resources associated with @job */
-+static void free_job(struct task *task, struct job *job)
-+{
-+ DBG("free job %p (%s)", job, job->inname);
-+ if (!job)
-+ return;
-+ free(job->inname);
-+ free(job->outname);
-+ free(job->relname);
-+ dref_put(job->dref);
-+ free(job);
-+}
-+
-+/* Check if file type specified by mode @m was marked as excluded */
-+static bool is_type_excluded(struct dump_opts *opts, mode_t m)
-+{
-+ int i;
-+
-+ m &= S_IFMT;
-+ for (i = 0; i < NUM_EXCLUDE_TYPES; i++) {
-+ if (exclude_types[i].mode == m)
-+ return opts->exclude_type[i];
-+ }
-+ return false;
-+}
-+
-+/* Replace all '/' characters in @filename with '_' */
-+static void escape_filename(char *filename)
-+{
-+ for (; *filename; filename++) {
-+ if (*filename == '/')
-+ *filename = '_';
-+ }
-+}
-+
-+/* Determine filename in archive from original filename @inname and
-+ * requested new filename @outname and depending on @type. */
-+static void set_outname(char **result_ptr, const char *outname,
-+ const char *inname, enum job_type type)
-+{
-+ const char *prefix = "", *name, *suffix;
-+ char *result, *end;
-+ size_t olen = outname ? strlen(outname) : 0, plen, nlen;
-+
-+ if (olen == 0) {
-+ /* No output name specified: outname = inname */
-+ name = inname;
-+ } else if (outname[olen - 1] == '/') {
-+ /* Output name is a directory: outname = outname/inname */
-+ prefix = outname;
-+ name = inname;
-+ } else {
-+ /* Output name is a filename: outname = inname */
-+ name = outname;
-+ }
-+
-+ if (type == JOB_DIR)
-+ suffix = "/";
-+ else
-+ suffix = "";
-+
-+ plen = strlen(prefix);
-+ nlen = strlen(name);
-+
-+ result = mmalloc(plen + nlen + strlen(suffix) + /* NUL */ 1);
-+
-+ /* Add prefix */
-+ strcpy(result, prefix);
-+
-+ /* Add name */
-+ end = result + plen;
-+ strcpy(end, name);
-+ if (type == JOB_CMD)
-+ escape_filename(end);
-+
-+ /* Add suffix */
-+ end = end + nlen;
-+ strcpy(end, suffix);
-+
-+ remove_double_slashes(result);
-+
-+ *result_ptr = result;
-+}
-+
-+static void sanitize_dirname(char **name_ptr)
-+{
-+ char *name;
-+
-+ name = mmalloc(strlen(*name_ptr) + /* Slash */ 1 + /* NUL */ 1);
-+ strcpy(name, *name_ptr);
-+ remove_double_slashes(name);
-+ chomp(name, "/");
-+ strcat(name, "/");
-+ free(*name_ptr);
-+ *name_ptr = name;
-+}
-+
-+/* Allocate and initialize a new job representation to add an entry according
-+ * to the specified parameters. @relname and @dref are used for opening files
-+ * more efficiently using *at() functions if specified. @is_cmd specifies if
-+ * the specified inname is a command line. */
-+static struct job *create_job(struct task *task, const char *inname,
-+ const char *outname, bool is_cmd,
-+ const char *relname, struct dref *dref,
-+ struct stats *stats)
-+{
-+ struct job *job = mmalloc(sizeof(struct job));
-+ int rc;
-+
-+ DBG("create job inname=%s outname=%s is_cmd=%d relname=%s dref=%p",
-+ inname, outname, is_cmd, relname, dref);
-+
-+ job->status = JOB_QUEUED;
-+
-+ if (!inname) {
-+ job->type = JOB_INIT;
-+ return job;
-+ }
-+
-+ job->inname = mstrdup(inname);
-+
-+ if (is_cmd) {
-+ /* Special case - read from command output */
-+ job->type = JOB_CMD;
-+ set_dummy_stat(&job->stat);
-+ goto out;
-+ }
-+
-+ if (!relname && strcmp(job->inname, "-") == 0) {
-+ /* Special case - read from standard input */
-+ job->type = JOB_FILE;
-+ set_dummy_stat(&job->stat);
-+ goto out;
-+ }
-+
-+ rc = stat_file(task->opts->dereference, job->inname, relname, dref,
-+ &job->stat);
-+
-+ if (rc < 0) {
-+ read_error(task, job->inname, "Cannot stat file");
-+ free_job(task, job);
-+ stats->num_failed++;
-+ return NULL;
-+ }
-+
-+ if (is_type_excluded(task->opts, job->stat.st_mode)) {
-+ free_job(task, job);
-+ stats->num_excluded++;
-+ return NULL;
-+ }
-+
-+ if (S_ISLNK(job->stat.st_mode)) {
-+ job->type = JOB_LINK;
-+ } else if (S_ISDIR(job->stat.st_mode)) {
-+ job->type = JOB_DIR;
-+ sanitize_dirname(&job->inname);
-+
-+ /* No need to keep parent directory open */
-+ relname = NULL;
-+ dref = NULL;
-+ } else {
-+ job->type = JOB_FILE;
-+ }
-+
-+ if (relname)
-+ job->relname = mstrdup(relname);
-+ job->dref = dref_get(dref);
-+
-+out:
-+ set_outname(&job->outname, outname, inname, job->type);
-+
-+ return job;
-+}
-+
-+void job_print(struct job *job)
-+{
-+ printf("DEBUG: job_print at %p\n", job);
-+ printf("DEBUG: next_job=%p\n", job->next_job);
-+ printf("DEBUG: type=%d\n", job->type);
-+ printf("DEBUG: status==%d\n", job->status);
-+ printf("DEBUG: outname=%s\n", job->outname);
-+ printf("DEBUG: inname=%s\n", job->inname);
-+ printf("DEBUG: relname=%s\n", job->relname);
-+ printf("DEBUG: timed=%d\n", job->timed);
-+ printf("DEBUG: dref=%p\n", job->dref);
-+ printf("DEBUG: cmd_status=%d\n", job->cmd_status);
-+ printf("DEBUG: content=%p\n", job->content);
-+}
-+
-+/* Return the number of bytes written to the output file */
-+static size_t get_output_size(struct task *task)
-+{
-+#ifdef HAVE_ZLIB
-+ if (task->opts->gzip) {
-+ gzflush(task->output_gzfd, Z_SYNC_FLUSH);
-+ return gztell(task->output_gzfd);
-+ }
-+#endif /* HAVE_ZLIB */
-+ return task->output_written;
-+}
-+
-+/* Write @len bytes at address @ptr to the output file */
-+static int write_output(struct task *task, const char *ptr, size_t len)
-+{
-+ size_t todo = len;
-+ ssize_t w;
-+
-+#ifdef HAVE_ZLIB
-+ if (task->opts->gzip) {
-+ if (gzwrite(task->output_gzfd, ptr, len) == 0)
-+ goto err_write;
-+ task->output_written += len;
-+
-+ return EXIT_OK;
-+ }
-+#endif /* HAVE_ZLIB */
-+
-+ while (todo > 0) {
-+ w = write(task->output_fd, ptr, todo);
-+ if (w < 0)
-+ goto err_write;
-+ todo -= w;
-+ ptr += w;
-+ }
-+ task->output_written += len;
-+
-+ return EXIT_OK;
-+
-+err_write:
-+ write_error(task, "Cannot write output");
-+
-+ return EXIT_RUNTIME;
-+}
-+
-+/* Write an end-of-file marker to the output file */
-+static void write_eof(struct task *task)
-+{
-+ char zeroes[TAR_BLOCKSIZE];
-+
-+ memset(zeroes, 0, sizeof(zeroes));
-+ write_output(task, zeroes, TAR_BLOCKSIZE);
-+ write_output(task, zeroes, TAR_BLOCKSIZE);
-+}
-+
-+/* Callback for writing out chunks of job data */
-+static int _write_job_data_cb(void *data, void *addr, size_t len)
-+{
-+ struct task *task = data;
-+
-+ return write_output(task, addr, len);
-+}
-+
-+/* Write tar entry for a file containing the exit status of the process that
-+ * ran command job @job */
-+static int write_job_status_file(struct task *task, struct job *job)
-+{
-+ char *name, *content;
-+ size_t len;
-+ struct stat st;
-+ int rc, status = job->cmd_status, exitstatus = -1, termsig = -1,
-+ waitpid_errno = -1;
-+
-+ name = masprintf("%s.cmdstatus", job->outname);
-+ if (status < 0)
-+ waitpid_errno = -status;
-+ else if (WIFEXITED(status))
-+ exitstatus = WEXITSTATUS(status);
-+ else if (WIFSIGNALED(status))
-+ termsig = WTERMSIG(status);
-+
-+ content = masprintf("EXITSTATUS=%d\n"
-+ "TERMSIG=%d\n"
-+ "WAITPID_ERRNO=%d\n", exitstatus, termsig,
-+ waitpid_errno);
-+
-+ len = strlen(content);
-+ set_dummy_stat(&st);
-+ rc = tar_emit_file_from_data(name, NULL, len, &st, TYPE_REGULAR,
-+ content, _write_job_data_cb, task);
-+ free(name);
-+ free(content);
-+
-+ return rc;
-+}
-+
-+/* Write tar entry for data in @job to output. Must be called with output_lock
-+ * held. */
-+static void _write_job_data(struct task *task, struct job *job)
-+{
-+ struct buffer *buffer = job->content;
-+
-+ switch (job->status) {
-+ case JOB_DONE:
-+ case JOB_PARTIAL:
-+ break;
-+ case JOB_FAILED:
-+ /* Create empty entries for failed reads */
-+ if (task->opts->ignore_failed_read)
-+ break;
-+ return;
-+ default:
-+ return;
-+ }
-+
-+ switch (job->type) {
-+ case JOB_CMD:
-+ tar_emit_file_from_buffer(job->outname, NULL, buffer->total,
-+ &job->stat, TYPE_REGULAR, buffer,
-+ _write_job_data_cb, task);
-+ task->output_num_files++;
-+ if (task->opts->add_cmd_status) {
-+ write_job_status_file(task, job);
-+ task->output_num_files++;
-+ }
-+ break;
-+ case JOB_FILE:
-+ tar_emit_file_from_buffer(job->outname, NULL, buffer->total,
-+ &job->stat, TYPE_REGULAR, buffer,
-+ _write_job_data_cb, task);
-+ task->output_num_files++;
-+ break;
-+ case JOB_LINK:
-+ tar_emit_file_from_buffer(job->outname, buffer->addr, 0,
-+ &job->stat, TYPE_LINK, NULL,
-+ _write_job_data_cb, task);
-+ task->output_num_files++;
-+ break;
-+ case JOB_DIR:
-+ tar_emit_file_from_buffer(job->outname, NULL, 0, &job->stat,
-+ TYPE_DIR, NULL, _write_job_data_cb,
-+ task);
-+ task->output_num_files++;
-+ break;
-+ default:
-+ break;
-+ }
-+
-+ if (task->opts->max_size > 0 &&
-+ get_output_size(task) > task->opts->max_size) {
-+ mwarnx("Archive size exceeds maximum of %ld bytes - aborting",
-+ task->opts->max_size);
-+ SET_ABORTED(task);
-+ }
-+}
-+
-+/* Read the contents of the symbolic link at @filename. On success, the
-+ * contents is returned in @buffer and the return value is %EXIT_OK.
-+ * If @relname is non-null it points to the name of the file relative
-+ * to its parent directory for which @dirfd is an open file handle. */
-+static int read_symlink(struct task *task, const char *filename,
-+ const char *relname, int dirfd, struct buffer *buffer)
-+{
-+ ssize_t actual = 0;
-+ size_t currlen = buffer->size ? buffer->size :
-+ task->opts->read_chunk_size;
-+ int rc = EXIT_OK;
-+
-+ while (!is_aborted(task)) {
-+ buffer_make_room(buffer, currlen, false,
-+ task->opts->max_buffer_size);
-+
-+ cancel_enable();
-+ if (relname)
-+ actual = readlinkat(dirfd, relname, buffer->addr,
-+ buffer->size);
-+ else
-+ actual = readlink(filename, buffer->addr, buffer->size);
-+ cancel_disable();
-+
-+ if (actual == -1) {
-+ read_error(task, filename, "Cannot read link");
-+ rc = EXIT_RUNTIME;
-+ /* Reset actual counter to get an empty buffer */
-+ actual = 0;
-+ break;
-+ }
-+
-+ /* Ensure that content doesn't exceed --file-max-size limit */
-+ if (task->opts->file_max_size > 0 &&
-+ (size_t) actual > task->opts->file_max_size) {
-+ actual = task->opts->file_max_size;/* Don't count NUL */
-+ mwarnx("%s: Warning: Data exceeds maximum size of %ld "
-+ "bytes - truncating", filename,
-+ task->opts->file_max_size);
-+ break;
-+ }
-+
-+ if ((size_t) actual < buffer->size)
-+ break;
-+
-+ currlen += task->opts->read_chunk_size;
-+ }
-+
-+ if (rc == EXIT_OK && is_aborted(task))
-+ rc = EXIT_RUNTIME;
-+
-+ buffer->addr[actual] = 0;
-+ buffer->total = actual + 1;
-+
-+ return rc;
-+}
-+
-+/* Read data from the file descriptor @fd until an end-of-file condition is
-+ * encountered. On success, *@done bytes in @buffer contain the read data
-+ * and the return value is %EXIT_OK. */
-+static int read_fd(struct task *task, const char *name, int fd,
-+ struct buffer *buffer)
-+{
-+ ssize_t rc = 0;
-+ size_t c = buffer->size ? buffer->size : task->opts->read_chunk_size;
-+
-+ while (!is_aborted(task)) {
-+ cancel_enable();
-+ rc = buffer_read_fd(buffer, fd, c, true,
-+ task->opts->max_buffer_size);
-+ cancel_disable();
-+
-+ if (rc <= 0)
-+ break;
-+
-+ /* Ensure that content doesn't exceed --file-max-size limit */
-+ if (task->opts->file_max_size > 0 &&
-+ buffer->total >= task->opts->file_max_size) {
-+ buffer_truncate(buffer, task->opts->file_max_size);
-+ rc = 0;
-+ mwarnx("%s: Warning: Data exceeds maximum size of %ld "
-+ "bytes - truncating", name,
-+ task->opts->file_max_size);
-+ break;
-+ }
-+
-+ c = buffer->size - buffer->off;
-+ if (c > 0) {
-+ /* Read to memory */
-+ } else if (buffer->size + task->opts->read_chunk_size <
-+ task->opts->max_buffer_size) {
-+ /* Enlarge memory buffer */
-+ c = task->opts->read_chunk_size;
-+ } else {
-+ /* Use full memory buffer size */
-+ c = task->opts->max_buffer_size;
-+ }
-+ }
-+
-+ if (is_aborted(task) || rc != 0)
-+ return EXIT_RUNTIME;
-+
-+ return EXIT_OK;
-+}
-+
-+/* Read data from the file at @filename until an end-of-file condition is
-+ * encountered. On success, @buffer contains the data read and the return
-+ * value is %EXIT_OK. If @relname is non-null it points to the name of the
-+ * file relative to its parent directory for which @dirfd is an open file
-+ * handle. */
-+static int read_regular(struct task *task, const char *filename,
-+ const char *relname, int dirfd, struct buffer *buffer)
-+{
-+ int fd, rc = EXIT_OK;
-+ bool need_close = true;
-+
-+ /* Opening a named pipe can block when peer is not ready */
-+ cancel_enable();
-+ if (strcmp(filename, "-") == 0) {
-+ fd = STDIN_FILENO;
-+ need_close = false;
-+ filename = "Standard input";
-+ } else if (relname)
-+ fd = openat(dirfd, relname, O_RDONLY);
-+ else
-+ fd = open(filename, O_RDONLY);
-+ cancel_disable();
-+
-+ if (fd < 0) {
-+ read_error(task, filename, "Cannot open file");
-+ return EXIT_RUNTIME;
-+ }
-+
-+ rc = read_fd(task, filename, fd, buffer);
-+ if (rc) {
-+ if (is_aborted(task))
-+ mwarnx("%s: Read aborted", filename);
-+ else
-+ read_error(task, filename, "Cannot read file");
-+ }
-+
-+ if (need_close)
-+ close(fd);
-+
-+ return rc;
-+}
-+
-+/* Read the output of command @cmd until an end-of-file condition is
-+ * encountered. On success, @buffer contain the output and the return value
-+ * is %EXIT_OK. When not %NULL, use @status_ptr to store the resulting process
-+ * status. */
-+static int read_cmd_output(struct task *task, char *cmd, struct buffer *buffer,
-+ int *status_ptr)
-+{
-+ int fd, rc = EXIT_RUNTIME;
-+ pid_t pid;
-+
-+ fd = cmd_open(cmd, &pid);
-+ if (fd < 0) {
-+ read_error(task, cmd, "Cannot run command");
-+ return rc;
-+ }
-+
-+ if (read_fd(task, cmd, fd, buffer)) {
-+ if (is_aborted(task))
-+ mwarnx("%s: Command aborted", cmd);
-+ else
-+ read_error(task, cmd, "Cannot read command output");
-+ } else
-+ rc = EXIT_OK;
-+
-+ cmd_close(fd, pid, status_ptr);
-+
-+ return rc;
-+
-+}
-+
-+/* Check the exclude patterns in @task->opts->exclude for a match of @filename.
-+ * If found, return the matching pattern string, otherwise return %NULL. */
-+static const char *get_exclude_match(struct task *task, const char *filename)
-+{
-+ unsigned int i;
-+ int mode = FNM_PERIOD | FNM_NOESCAPE;
-+
-+ for (i = 0; i < task->opts->exclude.num; i++) {
-+ if (fnmatch(task->opts->exclude.str[i], filename, mode) == 0)
-+ return task->opts->exclude.str[i];
-+ }
-+
-+ return NULL;
-+}
-+
-+/* Add the specified @job to the start of the job queue */
-+static void _queue_job_head(struct task *task, struct job *job)
-+{
-+ DBG("queue job type=%d inname=%s at head", job->type, job->inname);
-+ job->next_job = task->jobs_head;
-+ task->jobs_head = job;
-+ if (!task->jobs_tail)
-+ task->jobs_tail = job;
-+}
-+
-+/* Add the specified @job to the end of the job queue */
-+static void _queue_job_tail(struct task *task, struct job *job)
-+{
-+ DBG("queue job type=%d inname=%s at tail", job->type, job->inname);
-+ if (task->jobs_tail)
-+ task->jobs_tail->next_job = job;
-+ else
-+ task->jobs_head = job;
-+ task->jobs_tail = job;
-+}
-+
-+/* Add the specified @job to the job queue and trigger processing.
-+ * If @head is %true, the new job is inserted at the start of the job queue,
-+ * otherwise at the end. */
-+static void queue_job(struct task *task, struct job *job, bool head)
-+{
-+ main_lock(task);
-+ task->num_jobs_active++;
-+ if (head)
-+ _queue_job_head(task, job);
-+ else
-+ _queue_job_tail(task, job);
-+ _worker_wakeup_one(task);
-+ main_unlock(task);
-+}
-+
-+/* Add the specified list of jobs starting with @first up to @last to the start
-+ * of the job queue and trigger processing */
-+static void queue_jobs(struct task *task, struct job *first, struct job *last,
-+ int num)
-+{
-+ main_lock(task);
-+ last->next_job = task->jobs_head;
-+ task->jobs_head = first;
-+ task->num_jobs_active += num;
-+ _worker_wakeup_all(task);
-+ main_unlock(task);
-+}
-+
-+/* Remove the head of the job queue and return it to the caller */
-+static struct job *_dequeue_job(struct task *task)
-+{
-+ struct job *job = NULL;
-+
-+ if (task->jobs_head) {
-+ job = task->jobs_head;
-+ task->jobs_head = job->next_job;
-+ job->next_job = NULL;
-+ if (job == task->jobs_tail)
-+ task->jobs_tail = NULL;
-+ DBG("dequeueing job type=%d inname=%s", job->type, job->inname);
-+ job->status = JOB_IN_PROGRESS;
-+ } else {
-+ DBG("no job to dequeue");
-+ }
-+
-+ return job;
-+}
-+
-+/* Create and queue job for file at @filename */
-+static void queue_file(struct task *task, const char *inname,
-+ const char *outname, bool is_cmd,
-+ const char *relname, struct dref *dref,
-+ struct stats *stats, bool head)
-+{
-+ struct job *job;
-+
-+ job = create_job(task, inname, outname, is_cmd, relname, dref, stats);
-+ if (job)
-+ queue_job(task, job, head);
-+}
-+
-+/* Queue initial job */
-+static void init_queue(struct task *task)
-+{
-+ queue_file(task, NULL, NULL, false, NULL, NULL, NULL, true);
-+}
-+
-+/* Create and queue jobs for all files found in @dirname */
-+static void queue_dir(struct task *task, const char *dirname,
-+ const char *outname, struct stats *stats)
-+{
-+ struct dirent *de;
-+ char *inpath, *outpath;
-+ struct dref *dref;
-+ struct job *job, *first = NULL, *last = NULL;
-+ int num = 0;
-+
-+ dref = dref_create(dirname);
-+ if (!dref) {
-+ read_error(task, dirname, "Cannot read directory");
-+ return;
-+ }
-+
-+ while ((de = readdir(dref->dd))) {
-+ if (de->d_name[0] == '.') {
-+ if (de->d_name[1] == 0)
-+ continue;
-+ if (de->d_name[1] == '.' && de->d_name[2] == 0)
-+ continue;
-+ }
-+ DBG("next file %s", de->d_name);
-+ inpath = masprintf("%s%s", dirname, de->d_name);
-+ outpath = masprintf("%s%s", outname, de->d_name);
-+ job = create_job(task, inpath, outpath, false, de->d_name, dref,
-+ stats);
-+ if (job) {
-+ if (last) {
-+ last->next_job = job;
-+ last = job;
-+ } else {
-+ first = job;
-+ last = job;
-+ }
-+ num++;
-+ }
-+ free(inpath);
-+ free(outpath);
-+ }
-+
-+ if (first)
-+ queue_jobs(task, first, last, num);
-+
-+ dref_put(dref);
-+}
-+
-+/* Create and queue jobs for all files specified on the command line */
-+static void queue_jobs_from_opts(struct task *task, struct stats *stats)
-+{
-+ struct dump_opts *opts = task->opts;
-+ unsigned int i;
-+
-+ /* Queue directly specified entries */
-+ for (i = 0; i < opts->num_specs && !is_aborted(task); i++) {
-+ queue_file(task, opts->specs[i].inname, opts->specs[i].outname,
-+ opts->specs[i].is_cmd, NULL, NULL, stats, false);
-+ }
-+}
-+
-+/* Prepare output stream */
-+static int open_output(struct task *task)
-+{
-+ bool to_stdout = !task->opts->output_file ||
-+ strcmp(task->opts->output_file, "-") == 0;
-+ int rc = EXIT_OK;
-+
-+ if (to_stdout) {
-+ set_stdout_data();
-+ task->opts->output_file = "Standard output";
-+ }
-+
-+ cancel_enable();
-+#ifdef HAVE_ZLIB
-+ if (task->opts->gzip) {
-+ if (to_stdout) {
-+ task->output_gzfd =
-+ gzdopen(STDOUT_FILENO,
-+ task->opts->append ? "ab" : "wb");
-+ } else {
-+ task->output_gzfd =
-+ gzopen(task->opts->output_file,
-+ task->opts->append ? "ab" : "wb");
-+ }
-+
-+ if (!task->output_gzfd)
-+ rc = EXIT_RUNTIME;
-+ goto out;
-+ }
-+#endif /* HAVE_ZLIB */
-+
-+ if (to_stdout) {
-+ task->output_fd = STDOUT_FILENO;
-+ } else {
-+ task->output_fd =
-+ open(task->opts->output_file, O_WRONLY | O_CREAT |
-+ (task->opts->append ? O_APPEND : 0), 0666);
-+ }
-+
-+ if (task->output_fd < 0)
-+ rc = EXIT_RUNTIME;
-+ else if (!task->opts->append && ftruncate(task->output_fd, 0) == -1)
-+ rc = EXIT_RUNTIME;
-+
-+#ifdef HAVE_ZLIB
-+out:
-+#endif /* HAVE_ZLIB */
-+ cancel_disable();
-+
-+ if (rc != EXIT_OK) {
-+ mwarn("%s: Cannot open output file", task->opts->output_file);
-+ return rc;
-+ }
-+
-+ return EXIT_OK;
-+}
-+
-+/* Determine if the specified @job should be excluded from archiving */
-+static bool is_job_excluded(struct task *task, struct job *job)
-+{
-+ const char *pat;
-+
-+ if (job->type == JOB_INIT || job->type == JOB_CMD)
-+ return false;
-+
-+ pat = get_exclude_match(task, job->inname);
-+ if (!pat)
-+ return false;
-+
-+ tverb("Excluding '%s' due to exclude pattern '%s'\n", job->inname, pat);
-+
-+ return true;
-+}
-+
-+/* Perform all actions necessary to process @job and add resulting tar
-+ * data buffers to the buffer list of @thread. */
-+static void process_job(struct per_thread *thread, struct job *job)
-+{
-+ struct task *task = thread->task;
-+ const char *relname = job->dref ? job->relname : NULL;
-+ int dirfd = job->dref ? job->dref->dirfd : -1;
-+ struct buffer *buffer = &thread->buffer;
-+ enum job_status status = JOB_DONE;
-+
-+ DBG("processing job type=%d inname=%s", job->type, job->inname);
-+
-+ if (is_job_excluded(task, job)) {
-+ status = JOB_EXCLUDED;
-+ goto out;
-+ }
-+
-+ switch (job->type) {
-+ case JOB_INIT: /* Perform initial setup steps */
-+ if (open_output(task)) {
-+ SET_ABORTED(task);
-+ status = JOB_FAILED;
-+ goto out;
-+ }
-+ queue_jobs_from_opts(task, &thread->stats);
-+ break;
-+ case JOB_CMD: /* Capture command output */
-+ tverb("Dumping command output '%s'\n", job->inname);
-+
-+ set_dummy_stat(&job->stat);
-+ if (read_cmd_output(task, job->inname, buffer,
-+ &job->cmd_status))
-+ status = JOB_FAILED;
-+
-+ break;
-+ case JOB_LINK: /* Read symbolic link */
-+ tverb("Dumping link '%s'\n", job->inname);
-+
-+ if (read_symlink(task, job->inname, relname, dirfd, buffer))
-+ status = JOB_FAILED;
-+
-+ break;
-+ case JOB_DIR: /* Read directory contents */
-+ tverb("Dumping directory '%s'\n", job->inname);
-+
-+ if (task->opts->recursive) {
-+ queue_dir(task, job->inname, job->outname,
-+ &thread->stats);
-+ }
-+ break;
-+ case JOB_FILE: /* Read file contents */
-+ tverb("Dumping file '%s'\n", job->inname);
-+
-+ if (read_regular(task, job->inname, relname, dirfd, buffer))
-+ status = JOB_FAILED;
-+
-+ break;
-+ default:
-+ break;
-+ }
-+
-+out:
-+ job->status = status;
-+ DBG("processing done status=%d", job->status);
-+}
-+
-+/* Add @job results to statistics @stats */
-+static void account_stats(struct task *task, struct stats *stats,
-+ struct job *job)
-+{
-+ DBG("accounting job %s", job->inname);
-+
-+ if (job->type == JOB_INIT)
-+ return;
-+
-+ switch (job->status) {
-+ case JOB_DONE:
-+ stats->num_done++;
-+ if (job->type == JOB_CMD && task->opts->add_cmd_status)
-+ stats->num_done++;
-+ break;
-+ case JOB_PARTIAL:
-+ stats->num_done++;
-+ stats->num_partial++;
-+ if (job->type == JOB_CMD && task->opts->add_cmd_status)
-+ stats->num_done++;
-+ break;
-+ case JOB_FAILED:
-+ stats->num_failed++;
-+ break;
-+ case JOB_EXCLUDED:
-+ stats->num_excluded++;
-+ break;
-+ default:
-+ break;
-+ }
-+}
-+
-+/* Add statistics @from to @to */
-+static void add_stats(struct stats *to, struct stats *from)
-+{
-+ to->num_done += from->num_done;
-+ to->num_partial += from->num_partial;
-+ to->num_excluded += from->num_excluded;
-+ to->num_failed += from->num_failed;
-+}
-+
-+/* Release resources allocated to @thread */
-+static void cleanup_thread(struct per_thread *thread)
-+{
-+ if (thread->job)
-+ free_job(thread->task, thread->job);
-+ buffer_free(&thread->buffer, false);
-+}
-+
-+/* Register activate @job at @thread */
-+static void start_thread_job(struct per_thread *thread, struct job *job)
-+{
-+ struct task *task = thread->task;
-+
-+ thread->job = job;
-+ job->content = &thread->buffer;
-+ if (task->opts->file_timeout > 0 && job->type != JOB_INIT) {
-+ /* Set up per-job timeout */
-+ set_timespec(&job->deadline, task->opts->file_timeout, 0);
-+ job->timed = true;
-+
-+ /* Signal main thread to update deadline timeout */
-+ _main_wakeup(task);
-+ }
-+}
-+
-+/* Unregister active @job at @thread */
-+static void stop_thread_job(struct per_thread *thread, struct job *job)
-+{
-+ thread->job = NULL;
-+ job->content = NULL;
-+ buffer_reset(&thread->buffer);
-+}
-+
-+/* Wait until a job is available in the job queue. When a job becomes
-+ * available, dequeue and return it. Return %NULL if no more jobs are
-+ * available, or if processing was aborted. Must be called with task->mutex
-+ * locked. */
-+static struct job *_get_next_job(struct task *task)
-+{
-+ struct job *job = NULL;
-+
-+ do {
-+ DBG("checking for jobs");
-+ if (task->aborted)
-+ break;
-+ job = _dequeue_job(task);
-+ if (job)
-+ break;
-+ if (task->num_jobs_active == 0)
-+ break;
-+ DBG("found no jobs (%d active)", task->num_jobs_active);
-+ } while (_worker_wait(task) == 0);
-+
-+ return job;
-+}
-+
-+/* Unlock the mutex specified by @data */
-+static void cleanup_unlock(void *data)
-+{
-+ pthread_mutex_t *mutex = data;
-+
-+ pthread_mutex_unlock(mutex);
-+}
-+
-+/* Write entry for data in @job to output */
-+static void write_job_data(struct task *task, struct job *job)
-+{
-+ DBG("write_job_data");
-+ output_lock(task);
-+ pthread_cleanup_push(cleanup_unlock, &task->output_mutex);
-+ cancel_enable();
-+
-+ _write_job_data(task, job);
-+
-+ cancel_disable();
-+ pthread_cleanup_pop(0);
-+ output_unlock(task);
-+}
-+
-+/* Perform second part of job processing for @job at @thread by writing the
-+ * resulting tar file entry */
-+static void postprocess_job(struct per_thread *thread, struct job *job,
-+ bool cancelable)
-+{
-+ struct task *task = thread->task;
-+
-+ account_stats(task, &thread->stats, job);
-+ if (cancelable)
-+ write_job_data(task, job);
-+ else
-+ _write_job_data(task, job);
-+}
-+
-+/* Mark @job as complete by releasing all associated resources. If this was
-+ * the last active job inform main thread. Must be called with main_lock
-+ * mutex held. */
-+static void _complete_job(struct task *task, struct job *job)
-+{
-+ task->num_jobs_active--;
-+ if (task->num_jobs_active == 0)
-+ _main_wakeup(task);
-+ free_job(task, job);
-+}
-+
-+static void init_thread(struct per_thread *thread, struct task *task, long num)
-+{
-+ memset(thread, 0, sizeof(struct per_thread));
-+ thread->task = task;
-+ thread->num = num;
-+}
-+
-+/* Dequeue and process all jobs on the job queue */
-+static int process_queue(struct task *task)
-+{
-+ struct job *job;
-+ struct per_thread thread;
-+
-+ init_thread(&thread, task, 0);
-+
-+ while ((job = _dequeue_job(task)) && !is_aborted(task)) {
-+ start_thread_job(&thread, job);
-+ process_job(&thread, job);
-+ postprocess_job(&thread, job, false);
-+ stop_thread_job(&thread, job);
-+ _complete_job(task, job);
-+ }
-+
-+ task->stats = thread.stats;
-+ cleanup_thread(&thread);
-+
-+ return EXIT_OK;
-+}
-+
-+/* Return %true if @job is in a final state, %false otherwise */
-+static bool job_is_final(struct job *job)
-+{
-+ switch (job->status) {
-+ case JOB_DONE:
-+ case JOB_PARTIAL:
-+ case JOB_EXCLUDED:
-+ case JOB_FAILED:
-+ return true;
-+ default:
-+ break;
-+ }
-+
-+ return false;
-+}
-+
-+/* Main thread function: process jobs on the job queue until all jobs
-+ * are processed or processing was aborted. */
-+static void *worker_thread_main(void *d)
-+{
-+ struct per_thread *thread = d;
-+ struct task *task = thread->task;
-+ struct job *job;
-+
-+ /* Allow cancel only at specific code points */
-+ cancel_disable();
-+ set_threadname("%*sworker %d", (thread->num + 1) * 2, "", thread->num);
-+
-+ /* Handle jobs left over from canceled thread */
-+ job = thread->job;
-+ if (job) {
-+ DBG("handle aborted job %p", job);
-+
-+ postprocess_job(thread, job, true);
-+
-+ main_lock(task);
-+ if (thread->timed_out)
-+ goto out;
-+ stop_thread_job(thread, job);
-+ _complete_job(task, job);
-+ main_unlock(task);
-+ }
-+
-+ DBG("enter worker loop");
-+
-+ main_lock(task);
-+ while ((job = _get_next_job(task))) {
-+ start_thread_job(thread, job);
-+ main_unlock(task);
-+
-+ process_job(thread, job);
-+ postprocess_job(thread, job, true);
-+
-+ main_lock(task);
-+ if (thread->timed_out)
-+ goto out;
-+ stop_thread_job(thread, job);
-+ _complete_job(task, job);
-+ }
-+
-+out:
-+ thread->running = false;
-+ _main_wakeup(task);
-+ main_unlock(task);
-+
-+ cancel_enable();
-+ DBG("leave work loop");
-+
-+ return NULL;
-+}
-+
-+/* Start a worker thread associated with the specified @data. Return %EXIT_OK on
-+ * success. */
-+static int start_worker_thread(struct per_thread *data)
-+{
-+ int rc;
-+
-+ DBG("start thread");
-+ global_threaded = true;
-+ data->timed_out = false;
-+ rc = pthread_create(&data->thread, NULL, &worker_thread_main, data);
-+ if (rc) {
-+ mwarnx("Cannot start thread: %s", strerror(rc));
-+ return EXIT_RUNTIME;
-+ }
-+ data->running = true;
-+
-+ return EXIT_OK;
-+}
-+
-+/* Perform timeout handling for thread associated with @data by canceling and
-+ * restarting the corresponding thread. Must be called with task->mutex
-+ * held. */
-+static void _timeout_thread(struct per_thread *data)
-+{
-+ struct task *task = data->task;
-+ struct job *job = data->job;
-+ pthread_t thread = data->thread;
-+ const char *op, *action;
-+
-+ if (!job) {
-+ /* Timeout raced with job completion */
-+ return;
-+ }
-+ if (job_is_final(job)) {
-+ /* Job processing done, timeout does not apply */
-+ return;
-+ }
-+
-+ data->timed_out = true;
-+
-+ /* Allow thread to obtain main lock during cancel handling */
-+ main_unlock(task);
-+ DBG("cancel num=%d thread=%p", data->num, thread);
-+ pthread_cancel(thread);
-+ DBG("join num=%d thread=%p", data->num, thread);
-+
-+ pthread_join(thread, NULL);
-+ main_lock(task);
-+
-+ DBG("join done");
-+
-+ if (job->type == JOB_CMD)
-+ op = "Command";
-+ else
-+ op = "Read";
-+
-+ if (task->opts->ignore_failed_read)
-+ action = "skipping";
-+ else
-+ action = "aborting";
-+
-+ if (!job->inname || !*job->inname)
-+ job_print(job);
-+ mwarnx("%s: %s%s timed out after %d second%s - %s", job->inname,
-+ task->opts->ignore_failed_read ? "Warning: " : "", op,
-+ task->opts->file_timeout,
-+ task->opts->file_timeout > 1 ? "s" : "", action);
-+ if (!task->opts->ignore_failed_read)
-+ _SET_ABORTED(task);
-+
-+ /* Interrupted job will be handled by new thread - adjust status */
-+ if (job->status == JOB_IN_PROGRESS)
-+ job->status = JOB_PARTIAL;
-+ else if (!job_is_final(job))
-+ job->status = JOB_FAILED;
-+
-+ if (start_worker_thread(data))
-+ _SET_ABORTED(task);
-+}
-+
-+/* Return the number of currently running jobs */
-+static long num_jobs_running(struct task *task, struct per_thread *threads)
-+{
-+ long i, num = 0;
-+
-+ for (i = 0; i < task->opts->jobs; i++) {
-+ if (threads[i].running)
-+ num++;
-+ }
-+
-+ return num;
-+}
-+
-+/* Wait until all jobs are done or timeout occurs */
-+static int wait_for_completion(struct task *task, struct per_thread *threads)
-+{
-+ int rc = 0, earliest_timeout;
-+ long i;
-+ struct per_thread *earliest_thread;
-+ struct timespec tool_deadline_ts, deadline_ts, *earliest_ts;
-+ struct job *job;
-+
-+ /* Set tool deadline */
-+ tool_deadline_ts = task->start_ts;
-+ inc_timespec(&tool_deadline_ts, task->opts->timeout, 0);
-+
-+ main_lock(task);
-+ while (!task->aborted && task->num_jobs_active > 0) {
-+ /* Calculate nearest timeout */
-+ earliest_timeout = 0;
-+ earliest_ts = NULL;
-+ earliest_thread = NULL;
-+
-+ if (task->opts->timeout > 0) {
-+ earliest_timeout = task->opts->timeout;
-+ earliest_ts = &tool_deadline_ts;
-+ }
-+
-+ for (i = 0; i < task->opts->jobs; i++) {
-+ job = threads[i].job;
-+ if (!job || !job->timed)
-+ continue;
-+ if (task->opts->file_timeout == 0)
-+ continue;
-+ if (!earliest_ts ||
-+ ts_before(&job->deadline, earliest_ts)) {
-+ earliest_timeout = task->opts->file_timeout;
-+ earliest_ts = &job->deadline;
-+ earliest_thread = &threads[i];
-+ }
-+ }
-+
-+ /* Wait for status change or timeout */
-+ if (earliest_ts) {
-+ deadline_ts = *earliest_ts;
-+ rc = _main_wait_timed(task, &deadline_ts);
-+ } else {
-+ rc = _main_wait(task);
-+ }
-+
-+ if (rc == 0)
-+ continue;
-+ if (rc != ETIMEDOUT) {
-+ mwarnx("Cannot wait for status change: %s",
-+ strerror(rc));
-+ _SET_ABORTED(task);
-+ break;
-+ }
-+
-+ /* Timeout handling */
-+ if (earliest_thread) {
-+ /* Per-file timeout, restart */
-+ _timeout_thread(earliest_thread);
-+ rc = 0;
-+ } else {
-+ /* Global timeout, abort */
-+ mwarnx("Operation timed out after %d second%s - "
-+ "aborting", earliest_timeout,
-+ earliest_timeout > 1 ? "s" : "");
-+ _SET_ABORTED(task);
-+ break;
-+ }
-+ }
-+
-+ if (task->aborted)
-+ DBG("aborted");
-+ else
-+ DBG("all work done");
-+ _worker_wakeup_all(task);
-+
-+ /* Allow jobs to finish */
-+ set_timespec(&deadline_ts, 0, NSEC_PER_SEC / 4);
-+ while (!task->aborted && num_jobs_running(task, threads) > 0) {
-+ DBG("waiting for %lu processes",
-+ num_jobs_running(task, threads));
-+
-+ if (_main_wait_timed(task, &deadline_ts))
-+ break;
-+ }
-+
-+ main_unlock(task);
-+
-+ return rc;
-+}
-+
-+/* Finalize output stream */
-+static void close_output(struct task *task)
-+{
-+#ifdef HAVE_ZLIB
-+ if (task->opts->gzip) {
-+ gzclose(task->output_gzfd);
-+ return;
-+ }
-+#endif /* HAVE_ZLIB */
-+
-+ if (task->output_fd != STDOUT_FILENO)
-+ close(task->output_fd);
-+}
-+
-+/* Start multi-threaded processing of job queue */
-+static int process_queue_threaded(struct task *task)
-+{
-+ struct per_thread *threads, *thread;
-+ int rc;
-+ long i;
-+
-+ tverb("Using %ld threads\n", task->opts->jobs);
-+ threads = mcalloc(sizeof(struct per_thread), task->opts->jobs);
-+
-+ rc = 0;
-+ for (i = 0; i < task->opts->jobs; i++) {
-+ init_thread(&threads[i], task, i);
-+ rc = start_worker_thread(&threads[i]);
-+ if (rc)
-+ break;
-+ }
-+
-+ if (!rc)
-+ wait_for_completion(task, threads);
-+
-+ DBG("thread cleanup");
-+ for (i = 0; i < task->opts->jobs; i++) {
-+ thread = &threads[i];
-+ if (thread->running) {
-+ DBG("cancel %p", thread->thread);
-+ pthread_cancel(thread->thread);
-+ }
-+ DBG("join %p", thread->thread);
-+ pthread_join(thread->thread, NULL);
-+ add_stats(&task->stats, &thread->stats);
-+ cleanup_thread(thread);
-+ }
-+
-+ free(threads);
-+
-+ return rc;
-+}
-+
-+/* Abort any remaining queued jobs and account to @stats */
-+static void abort_queued_jobs(struct task *task)
-+{
-+ struct job *job;
-+
-+ while ((job = _dequeue_job(task))) {
-+ DBG("aborting job %s", job->inname);
-+ task->stats.num_failed++;
-+ job->status = JOB_FAILED;
-+ _complete_job(task, job);
-+ }
-+}
-+
-+/* Print a summary line */
-+static void print_summary(struct task *task)
-+{
-+ char msg[MSG_LEN];
-+ size_t off = 0;
-+ int rc;
-+ struct stats *stats = &task->stats;
-+ struct timespec end_ts;
-+ int num_special;
-+ unsigned long num_added;
-+
-+ if (task->opts->quiet)
-+ return;
-+ set_timespec(&end_ts, 0, 0);
-+
-+ num_special = 0;
-+ num_special += stats->num_partial > 0 ? 1 : 0;
-+ num_special += stats->num_excluded > 0 ? 1 : 0;
-+ num_special += stats->num_failed > 0 ? 1 : 0;
-+
-+ num_added = stats->num_done;
-+ if (task->opts->ignore_failed_read)
-+ num_added += stats->num_partial + stats->num_failed;
-+
-+ rc = snprintf(&msg[off], MSG_LEN - off, "Dumped %lu entries ",
-+ num_added);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+
-+ if (num_special > 0) {
-+ rc = snprintf(&msg[off], MSG_LEN - off, "(");
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ if (stats->num_partial > 0) {
-+ rc = snprintf(&msg[off], MSG_LEN - off, "%lu partial",
-+ stats->num_partial);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ if (--num_special > 0) {
-+ rc = snprintf(&msg[off], MSG_LEN - off, ", ");
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ }
-+ }
-+ if (stats->num_excluded > 0) {
-+ rc = snprintf(&msg[off], MSG_LEN - off, "%lu excluded",
-+ stats->num_excluded);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ if (--num_special > 0) {
-+ rc = snprintf(&msg[off], MSG_LEN - off, ", ");
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ }
-+ }
-+ if (stats->num_failed > 0) {
-+ rc = snprintf(&msg[off], MSG_LEN - off, "%lu failed",
-+ stats->num_failed);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ }
-+ rc = snprintf(&msg[off], MSG_LEN - off, ") ");
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ }
-+
-+ rc = snprintf(&msg[off], MSG_LEN - off, "in ");
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ snprintf_duration(&msg[off], MSG_LEN - off, &task->start_ts, &end_ts);
-+
-+out:
-+ info("%s\n", msg);
-+}
-+
-+static int init_task(struct task *task, struct dump_opts *opts)
-+{
-+ pthread_condattr_t attr;
-+
-+ memset(task, 0, sizeof(struct task));
-+ set_timespec(&task->start_ts, 0, 0);
-+ task->opts = opts;
-+ pthread_mutex_init(&task->mutex, NULL);
-+ pthread_mutex_init(&task->output_mutex, NULL);
-+ pthread_cond_init(&task->worker_cond, NULL);
-+
-+ pthread_condattr_init(&attr);
-+ if (pthread_condattr_setclock(&attr, CLOCK_MONOTONIC) ||
-+ pthread_cond_init(&task->cond, &attr)) {
-+ mwarn("Could not adjust pthread clock");
-+ return EXIT_RUNTIME;
-+ }
-+
-+ return EXIT_OK;
-+}
-+
-+struct dump_opts *dump_opts_new(void)
-+{
-+ struct dump_opts *opts = mmalloc(sizeof(struct dump_opts));
-+
-+ opts->recursive = true;
-+ opts->read_chunk_size = DEFAULT_READ_CHUNK_SIZE;
-+ opts->max_buffer_size = DEFAULT_MAX_BUFFER_SIZE;
-+
-+ return opts;
-+}
-+
-+void dump_opts_free(struct dump_opts *opts)
-+{
-+ unsigned int i;
-+
-+ if (!opts)
-+ return;
-+
-+ free_strarray(&opts->exclude);
-+ for (i = 0; i < opts->num_specs; i++) {
-+ free(opts->specs[i].inname);
-+ free(opts->specs[i].outname);
-+ }
-+ free(opts->specs);
-+ free(opts);
-+}
-+
-+void dump_opts_print(struct dump_opts *opts)
-+{
-+ unsigned int i;
-+
-+ printf("DEBUG: dump_opts at %p\n", opts);
-+ if (!opts)
-+ return;
-+ printf("DEBUG: add_cmd_status=%d\n", opts->add_cmd_status);
-+ printf("DEBUG: append=%d\n", opts->append);
-+ printf("DEBUG: dereference=%d\n", opts->dereference);
-+ for (i = 0; i < NUM_EXCLUDE_TYPES; i++)
-+ printf("DEBUG: exclude_type[%d]=%d\n", i,
-+ opts->exclude_type[i]);
-+ printf("DEBUG: gzip=%d\n", opts->gzip);
-+ printf("DEBUG: ignore_failed_read=%d\n", opts->ignore_failed_read);
-+ printf("DEBUG: no_eof=%d\n", opts->no_eof);
-+ printf("DEBUG: quiet=%d\n", opts->quiet);
-+ printf("DEBUG: recursive=%d\n", opts->recursive);
-+ printf("DEBUG: threaded=%d\n", opts->threaded);
-+ printf("DEBUG: verbose=%d\n", opts->verbose);
-+ printf("DEBUG: output_file=%s\n", opts->output_file);
-+ printf("DEBUG: file_timeout=%d\n", opts->file_timeout);
-+ printf("DEBUG: timeout=%d\n", opts->timeout);
-+ printf("DEBUG: jobs=%ld\n", opts->jobs);
-+ printf("DEBUG: jobs_per_cpu=%ld\n", opts->jobs_per_cpu);
-+ printf("DEBUG: file_max_size=%zu\n", opts->file_max_size);
-+ printf("DEBUG: max_buffer_size=%zu\n", opts->max_buffer_size);
-+ printf("DEBUG: max_size=%zu\n", opts->max_size);
-+ printf("DEBUG: read_chunk_size=%zu\n", opts->read_chunk_size);
-+ for (i = 0; i < opts->exclude.num; i++)
-+ printf("DEBUG: exclude[%d]=%s\n", i, opts->exclude.str[i]);
-+ for (i = 0; i < opts->num_specs; i++) {
-+ printf("DEBUG: specs[%d]:\n", i);
-+ printf("DEBUG: inname=%s\n", opts->specs[i].inname);
-+ printf("DEBUG: outname=%s\n", opts->specs[i].outname);
-+ printf("DEBUG: is_cmd=%d\n", opts->specs[i].is_cmd);
-+ }
-+}
-+
-+/* Mark file type associated with character @c as excluded */
-+int dump_opts_set_type_excluded(struct dump_opts *opts, char c)
-+{
-+ int i;
-+
-+ for (i = 0; i < NUM_EXCLUDE_TYPES; i++) {
-+ if (exclude_types[i].c == c) {
-+ opts->exclude_type[i] = true;
-+ return 0;
-+ }
-+ }
-+ return -1;
-+}
-+
-+/* Add entry specification defined by @iname, @outname and @op to @opts. */
-+void dump_opts_add_spec(struct dump_opts *opts, char *inname, char *outname,
-+ bool is_cmd)
-+{
-+ unsigned int i = opts->num_specs;
-+
-+ opts->specs = mrealloc(opts->specs, (i + 1) * sizeof(struct dump_spec));
-+ opts->specs[i].inname = mstrdup(inname);
-+ if (outname)
-+ opts->specs[i].outname = mstrdup(outname);
-+ else
-+ opts->specs[i].outname = NULL;
-+ opts->specs[i].is_cmd = is_cmd;
-+ opts->num_specs++;
-+}
-+
-+int dump_to_tar(struct dump_opts *opts)
-+{
-+ struct task task;
-+ int rc;
-+ long num_cpus;
-+
-+ if (opts->jobs_per_cpu > 0) {
-+ num_cpus = sysconf(_SC_NPROCESSORS_ONLN);
-+ if (num_cpus < 1) {
-+ mwarn("Cannot determine number of CPUs - assuming 1 "
-+ "CPU");
-+ num_cpus = 1;
-+ }
-+ opts->jobs = num_cpus;
-+ }
-+
-+ if (opts->jobs == 0 && (opts->timeout > 0 || opts->file_timeout > 0)) {
-+ /* Separate thread needed to implement timeout via cancel */
-+ opts->jobs = 1;
-+ }
-+
-+ rc = init_task(&task, opts);
-+ if (rc)
-+ return rc;
-+
-+ /* Queue initial job */
-+ init_queue(&task);
-+
-+ /* Process queue */
-+ if (opts->jobs > 0)
-+ rc = process_queue_threaded(&task);
-+ else
-+ rc = process_queue(&task);
-+ abort_queued_jobs(&task);
-+
-+ if (task.output_num_files > 0 && !opts->no_eof)
-+ write_eof(&task);
-+
-+ print_summary(&task);
-+
-+ close_output(&task);
-+
-+ if (rc == 0 && task.aborted)
-+ rc = EXIT_RUNTIME;
-+
-+ return rc;
-+}
---- /dev/null
-+++ b/dump2tar/src/dump2tar.c
-@@ -0,0 +1,474 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Command line interface
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+
-+#include "dump.h"
-+#include "global.h"
-+#include "idcache.h"
-+#include "misc.h"
-+#include "strarray.h"
-+#include "util_opt.h"
-+#include "util_prg.h"
-+
-+#define MIN_BUFFER_SIZE 4096
-+
-+#define OPT_NOSHORT_BASE 256
-+
-+#define OPT_DEREFERENCE (OPT_NOSHORT_BASE + 0)
-+#define OPT_NORECURSION (OPT_NOSHORT_BASE + 1)
-+#define OPT_EXCLUDETYPE (OPT_NOSHORT_BASE + 2)
-+
-+/* Program description */
-+static const struct util_prg dump2tar_prg = {
-+ .desc = "Use dump2tar to create a tar archive from the contents "
-+ "of arbitrary files.\nIt works even when the size of actual "
-+ "file content is not known beforehand,\nsuch as with FIFOs, "
-+ "character devices or certain Linux debugfs or sysfs files.\n"
-+ "\nYou can also add files under different names and add "
-+ "command output using the\nformat described in section SPECS "
-+ "below. When no additional options are\nspecified, the "
-+ "resulting archive is written to the standard output stream\n"
-+ "in uncompressed tar format.",
-+ .args = "SPECS",
-+ .copyright_vec = {
-+ { "IBM Corp.", 2016, 2016 },
-+ UTIL_PRG_COPYRIGHT_END
-+ },
-+};
-+
-+/* Definition of command line options */
-+static struct util_opt dump2tar_opts[] = {
-+ UTIL_OPT_SECTION("OUTPUT OPTIONS"),
-+ {
-+ .option = { "output-file", required_argument, NULL, 'o' },
-+ .argument = "FILE",
-+ .desc = "Write archive to FILE (default: standard output)",
-+ },
-+#ifdef HAVE_ZLIB
-+ {
-+ .option = { "gzip", no_argument, NULL, 'z' },
-+ .desc = "Write a gzip compressed archive",
-+ },
-+#endif /* HAVE_ZLIB */
-+ {
-+ .option = { "max-size", required_argument, NULL, 'm' },
-+ .argument = "N",
-+ .desc = "Stop adding files when archive size exceeds N bytes",
-+ },
-+ {
-+ .option = { "timeout", required_argument, NULL, 't' },
-+ .argument = "SEC",
-+ .desc = "Stop adding files after SEC seconds",
-+ },
-+ {
-+ .option = { "no-eof", no_argument, NULL, 131 },
-+ .desc = "Do not write an end-of-file marker",
-+ .flags = UTIL_OPT_FLAG_NOSHORT,
-+ },
-+ {
-+ .option = { "add-cmd-status", no_argument, NULL, 132 },
-+ .desc = "Add status of commands as separate file",
-+ .flags = UTIL_OPT_FLAG_NOSHORT,
-+ },
-+ {
-+ .option = { "append", no_argument, NULL, 133 },
-+ .desc = "Append output to end of file",
-+ .flags = UTIL_OPT_FLAG_NOSHORT,
-+ },
-+
-+ UTIL_OPT_SECTION("INPUT OPTIONS"),
-+ {
-+ .option = { "files-from", required_argument, NULL, 'F' },
-+ .argument = "FILE",
-+ .desc = "Read filenames from FILE (- for standard input)",
-+ },
-+ {
-+ .option = { "ignore-failed-read", no_argument, NULL, 'i' },
-+ .desc = "Continue after read errors",
-+ },
-+ {
-+ .option = { "buffer-size", required_argument, NULL, 'b' },
-+ .argument = "N",
-+ .desc = "Read data in chunks of N byte (default: 16384)",
-+ },
-+ {
-+ .option = { "file-timeout", required_argument, NULL, 'T' },
-+ .desc = "Stop reading file after SEC seconds",
-+ },
-+ {
-+ .option = { "file-max-size", required_argument, NULL, 'M' },
-+ .argument = "N",
-+ .desc = "Stop reading file after N bytes",
-+ },
-+ {
-+ .option = { "jobs", required_argument, NULL, 'j' },
-+ .argument = "N",
-+ .desc = "Read N files in parallel (default: 1)",
-+ },
-+ {
-+ .option = { "jobs-per-cpu", required_argument, NULL, 'J' },
-+ .argument = "N",
-+ .desc = "Read N files per CPU in parallel",
-+ },
-+ {
-+ .option = { "exclude", required_argument, NULL, 'x' },
-+ .argument = "PATTERN",
-+ .desc = "Don't add files matching PATTERN",
-+ },
-+ {
-+ .option = { "exclude-from", required_argument, NULL, 'X' },
-+ .argument = "FILE",
-+ .desc = "Don't add files matching patterns in FILE",
-+ },
-+ {
-+ .option = { "exclude-type", required_argument, NULL,
-+ OPT_EXCLUDETYPE },
-+ .argument = "TYPE",
-+ .desc = "Don't add files of specified TYPE (one of: fdcbpls)",
-+ .flags = UTIL_OPT_FLAG_NOSHORT,
-+ },
-+ {
-+ .option = { "dereference", no_argument, NULL, OPT_DEREFERENCE },
-+ .desc = "Add link targets instead of links",
-+ .flags = UTIL_OPT_FLAG_NOSHORT,
-+ },
-+ {
-+ .option = { "no-recursion", no_argument, NULL,
-+ OPT_NORECURSION },
-+ .desc = "Don't add files from sub-directories",
-+ .flags = UTIL_OPT_FLAG_NOSHORT,
-+ },
-+
-+ UTIL_OPT_SECTION("MISC OPTIONS"),
-+ UTIL_OPT_HELP,
-+ UTIL_OPT_VERSION,
-+ {
-+ .option = { "verbose", no_argument, NULL, 'V' },
-+ .desc = "Print additional informational output",
-+ },
-+ {
-+ .option = { "quiet", no_argument, NULL, 'q' },
-+ .desc = "Suppress printing of informational output",
-+ },
-+ UTIL_OPT_END,
-+};
-+
-+/* Split buffer size specification in @arg into two numbers to be stored in
-+ * @from_ptr and @to_ptr. Return %EXIT_OK on success. */
-+static int parse_buffer_size(char *arg, size_t *from_ptr, size_t *to_ptr)
-+{
-+ char *err;
-+ unsigned long from, to;
-+
-+ if (!*arg) {
-+ mwarnx("Empty buffer size specified");
-+ return EXIT_USAGE;
-+ }
-+
-+ from = strtoul(arg, &err, 10);
-+
-+ if (*err == '-')
-+ to = strtoul(err + 1, &err, 10);
-+ else
-+ to = *to_ptr;
-+
-+ if (*err) {
-+ mwarnx("Invalid buffer size: %s", arg);
-+ return EXIT_USAGE;
-+ }
-+
-+ if (from < MIN_BUFFER_SIZE || to < MIN_BUFFER_SIZE) {
-+ mwarnx("Buffer size too low (minimum %u)", MIN_BUFFER_SIZE);
-+ return EXIT_USAGE;
-+ }
-+
-+ if (to < from)
-+ to = from;
-+
-+ *from_ptr = from;
-+ *to_ptr = to;
-+
-+ return EXIT_OK;
-+}
-+
-+static void parse_and_add_spec(struct dump_opts *opts, const char *spec)
-+{
-+ char *op, *s, *inname, *outname = NULL;
-+ bool is_cmd = false;
-+
-+ s = mstrdup(spec);
-+ op = strstr(s, "|=");
-+ if (op)
-+ is_cmd = true;
-+ else
-+ op = strstr(s, ":=");
-+
-+ if (op) {
-+ *op = 0;
-+ inname = op + 2;
-+ outname = s;
-+ } else {
-+ inname = s;
-+ }
-+ dump_opts_add_spec(opts, inname, outname, is_cmd);
-+ free(s);
-+}
-+
-+static int add_specs_from_file(struct dump_opts *opts, const char *filename)
-+{
-+ FILE *fd;
-+ char *line = NULL;
-+ size_t line_size;
-+ int rc = EXIT_RUNTIME;
-+ bool need_close = false, parse_spec = true;
-+
-+ if (strcmp(filename, "-") == 0)
-+ fd = stdin;
-+ else {
-+ fd = fopen(filename, "r");
-+ if (!fd) {
-+ mwarn("%s: Cannot open file", filename);
-+ goto out;
-+ }
-+ need_close = true;
-+ }
-+
-+ while ((getline(&line, &line_size, fd) != -1)) {
-+ chomp(line, "\n");
-+ if (line[0] == 0)
-+ continue;
-+ if (parse_spec && strcmp(line, "--") == 0) {
-+ /* After a line containing --, no more := or |= specs
-+ * are expected */
-+ parse_spec = false;
-+ continue;
-+ }
-+ if (parse_spec)
-+ parse_and_add_spec(opts, line);
-+ else
-+ dump_opts_add_spec(opts, line, NULL, false);
-+ }
-+
-+ if (ferror(fd))
-+ mwarn("%s: Cannot read file", filename);
-+ else
-+ rc = EXIT_OK;
-+
-+out:
-+ if (need_close)
-+ fclose(fd);
-+ free(line);
-+
-+ return rc;
-+}
-+
-+static void print_help(void)
-+{
-+ static const struct {
-+ const char *name;
-+ const char *desc;
-+ } specs[] = {
-+ { "PATH", "Add file or directory at PATH" },
-+ { "NEWPATH:=PATH", "Add file or directory at PATH as NEWPATH" },
-+ { "NEWPATH|=CMDLINE", "Add output of command line CMDLINE as "
-+ "NEWPATH" },
-+ { NULL, NULL },
-+ };
-+ int i;
-+
-+ util_prg_print_help();
-+ printf("SPECS\n");
-+ for (i = 0; specs[i].name; i++)
-+ util_opt_print_indented(specs[i].name, specs[i].desc);
-+ printf("\n");
-+ util_opt_print_help();
-+}
-+
-+int main(int argc, char *argv[])
-+{
-+ int rc = EXIT_USAGE, opt;
-+ long i;
-+ struct dump_opts *opts;
-+
-+ if (getenv("DUMP2TAR_DEBUG"))
-+ global_debug = true;
-+
-+ util_prg_init(&dump2tar_prg);
-+ util_opt_init(dump2tar_opts, "-");
-+ misc_init();
-+
-+ opts = dump_opts_new();
-+ opterr = 0;
-+ while ((opt = util_opt_getopt_long(argc, argv)) != -1) {
-+ switch (opt) {
-+ case 'h': /* --help */
-+ print_help();
-+ rc = EXIT_OK;
-+ goto out;
-+ case 'v': /* --version */
-+ util_prg_print_version();
-+ rc = EXIT_OK;
-+ goto out;
-+ case 'V': /* --verbose */
-+ global_verbose = true;
-+ global_quiet = false;
-+ opts->verbose = true;
-+ opts->quiet = false;
-+ break;
-+ case 'q': /* --quiet */
-+ global_quiet = true;
-+ global_verbose = false;
-+ opts->quiet = true;
-+ opts->verbose = false;
-+ break;
-+ case 'i': /* --ignore-failed-read */
-+ opts->ignore_failed_read = true;
-+ break;
-+ case 'j': /* --jobs N */
-+ opts->jobs = atoi(optarg);
-+ if (opts->jobs < 1) {
-+ mwarnx("Invalid number of jobs: %s", optarg);
-+ goto out;
-+ }
-+ break;
-+ case 'J': /* --jobs-per-cpu N */
-+ opts->jobs_per_cpu = atoi(optarg);
-+ if (opts->jobs_per_cpu < 1) {
-+ mwarnx("Invalid number of jobs: %s", optarg);
-+ goto out;
-+ }
-+ break;
-+ case 'b': /* --buffer-size N */
-+ if (parse_buffer_size(optarg, &opts->read_chunk_size,
-+ &opts->max_buffer_size))
-+ goto out;
-+ break;
-+ case 'x': /* --exclude PATTERN */
-+ add_str_to_strarray(&opts->exclude, optarg);
-+ break;
-+ case 'X': /* --exclude-from FILE */
-+ if (add_file_to_strarray(&opts->exclude, optarg))
-+ goto out;
-+ break;
-+ case 'F': /* --files-from FILE */
-+ if (add_specs_from_file(opts, optarg))
-+ goto out;
-+ break;
-+ case 'o': /* --output-file FILE */
-+ if (opts->output_file) {
-+ mwarnx("Output file specified multiple times");
-+ goto out;
-+ }
-+ opts->output_file = optarg;
-+ break;
-+ case OPT_DEREFERENCE: /* --dereference */
-+ opts->dereference = true;
-+ break;
-+ case OPT_NORECURSION: /* --no-recursion */
-+ opts->recursive = false;
-+ break;
-+ case OPT_EXCLUDETYPE: /* --exclude-type TYPE */
-+ for (i = 0; optarg[i]; i++) {
-+ if (dump_opts_set_type_excluded(opts,
-+ optarg[i]))
-+ break;
-+
-+ }
-+ if (optarg[i]) {
-+ mwarnx("Unrecognized file type: %c", optarg[i]);
-+ goto out;
-+ }
-+ break;
-+ case 131: /* --no-eof */
-+ opts->no_eof = true;
-+ break;
-+ case 132: /* --add-cmd-status */
-+ opts->add_cmd_status = true;
-+ break;
-+ case 133: /* --append */
-+ opts->append = true;
-+ break;
-+ case 't': /* --timeout VALUE */
-+ opts->timeout = atoi(optarg);
-+ if (opts->timeout < 1) {
-+ mwarnx("Invalid timeout value: %s", optarg);
-+ goto out;
-+ }
-+ break;
-+ case 'T': /* --file-timeout VALUE */
-+ opts->file_timeout = atoi(optarg);
-+ if (opts->file_timeout < 1) {
-+ mwarnx("Invalid timeout value: %s", optarg);
-+ goto out;
-+ }
-+ break;
-+ case 'm': /* --max-size N */
-+ opts->max_size = atol(optarg);
-+ if (opts->max_size < 2) {
-+ mwarnx("Invalid maximum size: %s", optarg);
-+ goto out;
-+ }
-+ break;
-+ case 'M': /* --file-max-size N */
-+ opts->file_max_size = atol(optarg);
-+ if (opts->file_max_size < 2) {
-+ mwarnx("Invalid maximum size: %s", optarg);
-+ goto out;
-+ }
-+ break;
-+ case 'z': /* --gzip */
-+ opts->gzip = true;
-+ break;
-+ case 1: /* Filename specification or unrecognized option */
-+ if (optarg[0] == '-') {
-+ mwarnx("Invalid option '%s'", optarg);
-+ goto out;
-+ }
-+ parse_and_add_spec(opts, optarg);
-+ break;
-+ case '?': /* Unrecognized option */
-+ if (optopt)
-+ mwarnx("Invalid option '-%c'", optopt);
-+ else
-+ mwarnx("Invalid option '%s'", argv[optind - 1]);
-+ goto out;
-+ case ':': /* Missing argument */
-+ mwarnx("Option '%s' requires an argument",
-+ argv[optind - 1]);
-+ goto out;
-+ default:
-+ break;
-+ }
-+ }
-+ if (optind >= argc && opts->num_specs == 0) {
-+ mwarnx("Please specify files to dump");
-+ goto out;
-+ }
-+
-+ for (i = optind; i < argc; i++)
-+ dump_opts_add_spec(opts, argv[i], NULL, false);
-+
-+ rc = dump_to_tar(opts);
-+
-+out:
-+ idcache_cleanup();
-+ misc_cleanup();
-+ dump_opts_free(opts);
-+
-+ if (rc == EXIT_USAGE)
-+ util_prg_print_parse_error();
-+
-+ return rc;
-+}
---- /dev/null
-+++ b/dump2tar/src/global.c
-@@ -0,0 +1,17 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Global variables
-+ *
-+ * Copyright IBM Corp. 2016
-+ *
-+ */
-+
-+#include "global.h"
-+
-+/* Global settings */
-+bool global_threaded;
-+bool global_debug;
-+bool global_verbose;
-+bool global_quiet;
-+bool global_timestamps;
---- /dev/null
-+++ b/dump2tar/src/idcache.c
-@@ -0,0 +1,153 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Caches for user and group ID lookups
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#include "idcache.h"
-+
-+#include
-+#include
-+#include
-+#include
-+#include
-+
-+#include "global.h"
-+#include "misc.h"
-+
-+/* Maximum user and group name lengths as defined in tar header */
-+#define ID_NAME_MAXLEN 32
-+
-+/* Types for user and group ID caches */
-+typedef uid_t generic_id_t; /* Assumes that uid_t == gid_t */
-+
-+struct id_cache_entry {
-+ generic_id_t id;
-+ char name[ID_NAME_MAXLEN];
-+};
-+
-+struct id_cache {
-+ unsigned int num;
-+ struct id_cache_entry entries[];
-+};
-+
-+/* cache_mutex serializes access to cached uid and gid data */
-+static pthread_mutex_t id_cache_mutex = PTHREAD_MUTEX_INITIALIZER;
-+static struct id_cache *id_cache_uid;
-+static struct id_cache *id_cache_gid;
-+
-+/* Lock cache mutex */
-+static void cache_lock(void)
-+{
-+ if (!global_threaded)
-+ return;
-+ pthread_mutex_lock(&id_cache_mutex);
-+}
-+
-+/* Unlock cache mutex */
-+static void cache_unlock(void)
-+{
-+ if (!global_threaded)
-+ return;
-+ pthread_mutex_unlock(&id_cache_mutex);
-+}
-+
-+/* Copy the name associated with @id in @cache to at most @len bytes at @dest.
-+ * Return %true if name was found in cache, %false otherwise. */
-+static bool strncpy_id_cache_entry(char *dest, struct id_cache *cache,
-+ generic_id_t id, size_t len)
-+{
-+ unsigned int i;
-+ bool hit = false;
-+
-+ cache_lock();
-+ if (cache) {
-+ for (i = 0; i < cache->num; i++) {
-+ if (cache->entries[i].id == id) {
-+ strncpy(dest, cache->entries[i].name, len);
-+ hit = true;
-+ break;
-+ }
-+ }
-+ }
-+ cache_unlock();
-+
-+ return hit;
-+}
-+
-+/* Add a new entry consisting of @id and @name to ID cache in @*cache_ptr.
-+ * Update @cache_ptr if necessary. */
-+static void add_id_cache_entry(struct id_cache **cache_ptr, generic_id_t id,
-+ char *name)
-+{
-+ struct id_cache *cache;
-+ unsigned int cache_num;
-+ size_t new_size;
-+ struct id_cache *new_cache;
-+
-+ cache_lock();
-+
-+ cache = *cache_ptr;
-+ cache_num = cache ? cache->num : 0;
-+ new_size = sizeof(struct id_cache) +
-+ sizeof(struct id_cache_entry) * (cache_num + 1);
-+ new_cache = mrealloc(cache, new_size);
-+ if (cache_num == 0)
-+ new_cache->num = 0;
-+ new_cache->entries[cache_num].id = id;
-+ strncpy(new_cache->entries[cache_num].name, name, ID_NAME_MAXLEN);
-+ new_cache->num++;
-+ *cache_ptr = new_cache;
-+
-+ cache_unlock();
-+}
-+
-+/* Copy the user name corresponding to user ID @uid to at most @len bytes
-+ * at @name */
-+void uid_to_name(uid_t uid, char *name, size_t len)
-+{
-+ struct passwd pwd, *pwd_ptr;
-+ char buffer[PWD_BUFFER_SIZE], *result;
-+
-+ if (strncpy_id_cache_entry(name, id_cache_uid, uid, len))
-+ return;
-+
-+ /* getpwuid() can be slow so cache results */
-+ getpwuid_r(uid, &pwd, buffer, PWD_BUFFER_SIZE, &pwd_ptr);
-+ if (!pwd_ptr || !pwd_ptr->pw_name)
-+ return;
-+ result = pwd_ptr->pw_name;
-+
-+ add_id_cache_entry(&id_cache_uid, uid, result);
-+
-+ strncpy(name, result, len);
-+}
-+
-+/* Copy the group name corresponding to group ID @gid to at most @len bytes
-+ * at @name */
-+void gid_to_name(gid_t gid, char *name, size_t len)
-+{
-+ struct group grp, *grp_ptr;
-+ char buffer[GRP_BUFFER_SIZE], *result;
-+
-+ if (strncpy_id_cache_entry(name, id_cache_gid, gid, len))
-+ return;
-+
-+ /* getgrgid() can be slow so cache results */
-+ getgrgid_r(gid, &grp, buffer, GRP_BUFFER_SIZE, &grp_ptr);
-+ if (!grp_ptr || !grp_ptr->gr_name)
-+ return;
-+ result = grp_ptr->gr_name;
-+
-+ add_id_cache_entry(&id_cache_gid, gid, result);
-+
-+ strncpy(name, result, len);
-+}
-+
-+void idcache_cleanup(void)
-+{
-+ free(id_cache_uid);
-+ free(id_cache_gid);
-+}
---- /dev/null
-+++ b/dump2tar/src/misc.c
-@@ -0,0 +1,492 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Helper functions
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#define _GNU_SOURCE /* for program_invocation_short_name */
-+
-+#include "misc.h"
-+
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+#include
-+
-+#include "dref.h"
-+#include "global.h"
-+
-+struct timespec main_start_ts;
-+
-+static pthread_key_t thread_name_key;
-+static bool stdout_data;
-+
-+/* Write @len bytes at @addr to @fd. Return %EXIT_OK on success, %EXIT_RUNTIME
-+ * otherwise. */
-+int misc_write_data(int fd, char *addr, size_t len)
-+{
-+ ssize_t w;
-+
-+ while (len > 0) {
-+ w = write(fd, addr, len);
-+ if (w < 0)
-+ return EXIT_RUNTIME;
-+ len -= w;
-+ addr += w;
-+ }
-+
-+ return EXIT_OK;
-+}
-+
-+/* Read at most @len bytes from @fd to @addr. Return the number of bytes read
-+ * or %-1 on error. */
-+ssize_t misc_read_data(int fd, char *addr, size_t len)
-+{
-+ size_t done = 0;
-+ ssize_t r;
-+
-+ while (len > 0) {
-+ r = read(fd, addr, len);
-+ if (r < 0)
-+ return -1;
-+ if (r == 0)
-+ break;
-+ len -= r;
-+ addr += r;
-+ done += r;
-+ }
-+
-+ return done;
-+}
-+
-+/* Advance timespec @ts by @sec seconds and @nsec nanoseconds */
-+void inc_timespec(struct timespec *ts, time_t sec, long nsec)
-+{
-+ ts->tv_nsec += nsec;
-+ ts->tv_sec += sec;
-+ if (ts->tv_nsec > NSEC_PER_SEC) {
-+ ts->tv_nsec -= NSEC_PER_SEC;
-+ ts->tv_sec++;
-+ }
-+}
-+
-+/* Set timespec @ts to point to @sec seconds and @nsec nanoseconds in the
-+ * future */
-+void set_timespec(struct timespec *ts, time_t sec, long nsec)
-+{
-+ clock_gettime(CLOCK_MONOTONIC, ts);
-+ inc_timespec(ts, sec, nsec);
-+}
-+
-+/* Return true if timespec @a refers to a point in time before @b */
-+bool ts_before(struct timespec *a, struct timespec *b)
-+{
-+ if (a->tv_sec < b->tv_sec ||
-+ (a->tv_sec == b->tv_sec && a->tv_nsec < b->tv_nsec))
-+ return true;
-+
-+ return false;
-+}
-+
-+/* Store a string representing the time duration between @start and @end in
-+ * at most @len bytes of @buff. */
-+int snprintf_duration(char *buff, size_t len, struct timespec *start,
-+ struct timespec *end)
-+{
-+ time_t sec;
-+ long nsec, msec, s, m, h;
-+
-+ sec = end->tv_sec - start->tv_sec;
-+ nsec = end->tv_nsec - start->tv_nsec;
-+
-+ if (nsec < 0) {
-+ nsec += NSEC_PER_SEC;
-+ sec--;
-+ }
-+
-+ msec = nsec / NSEC_PER_MSEC;
-+ s = sec % 60;
-+ sec /= 60;
-+ m = sec % 60;
-+ sec /= 60;
-+ h = sec;
-+
-+ if (h > 0)
-+ return snprintf(buff, len, "%luh%lum%lu.%03lus", h, m, s, msec);
-+ else if (m > 0)
-+ return snprintf(buff, len, "%lum%lu.%03lus", m, s, msec);
-+ else
-+ return snprintf(buff, len, "%lu.%03lus", s, msec);
-+}
-+
-+/* Return the name of the current thread */
-+char *get_threadname(void)
-+{
-+ return pthread_getspecific(thread_name_key);
-+}
-+
-+static int snprintf_timestamp(char *str, size_t size)
-+{
-+ struct timespec now_ts;
-+
-+ set_timespec(&now_ts, 0, 0);
-+ now_ts.tv_sec -= main_start_ts.tv_sec;
-+ now_ts.tv_nsec -= main_start_ts.tv_nsec;
-+ if (now_ts.tv_nsec < 0) {
-+ now_ts.tv_nsec += NSEC_PER_SEC;
-+ now_ts.tv_sec--;
-+ }
-+
-+ return snprintf(str, size, "[%3lu.%06lu] ", now_ts.tv_sec,
-+ now_ts.tv_nsec / NSEC_PER_USEC);
-+}
-+
-+/* When DUMP2TAR_DEBUG is set to non-zero, print debugging information */
-+void debug(const char *file, unsigned long line, const char *format, ...)
-+{
-+ char msg[MSG_LEN];
-+ size_t off = 0;
-+ int rc;
-+ va_list args;
-+
-+ /* Debug marker */
-+ rc = snprintf(&msg[off], MSG_LEN - off, "DEBUG: ");
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+
-+ /* Timestamp */
-+ rc = snprintf_timestamp(&msg[off], MSG_LEN - off);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+
-+ /* Thread name */
-+ rc = snprintf(&msg[off], MSG_LEN - off, "%s: ", get_threadname());
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+
-+ /* Message */
-+ va_start(args, format);
-+ rc = vsnprintf(&msg[off], MSG_LEN - off, format, args);
-+ va_end(args);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+
-+ /* Call site */
-+ rc = snprintf(&msg[off], MSG_LEN - off, " (%s:%lu)", file, line);
-+
-+out:
-+ fprintf(stderr, "%s\n", msg);
-+}
-+
-+/* Print a warning message consisting of @format and variable arguments.
-+ * If @print_errno is true, also print the text corresponding to errno.
-+ * We're not using err.h's warn since we want timestamps and synchronized
-+ * output. */
-+void _mwarn(bool print_errno, const char *format, ...)
-+{
-+ char msg[MSG_LEN];
-+ size_t off = 0;
-+ int rc;
-+ va_list args;
-+
-+ if (global_timestamps) {
-+ rc = snprintf_timestamp(&msg[off], MSG_LEN - off);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ }
-+
-+ rc = snprintf(&msg[off], MSG_LEN - off, "%s: ",
-+ program_invocation_short_name);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+
-+ va_start(args, format);
-+ rc = vsnprintf(&msg[off], MSG_LEN - off, format, args);
-+ va_end(args);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+
-+ if (print_errno)
-+ snprintf(&msg[off], MSG_LEN - off, ": %s", strerror(errno));
-+
-+out:
-+ fprintf(stderr, "%s\n", msg);
-+}
-+
-+/* Provide informational output if --verbose was specified */
-+void verb(const char *format, ...)
-+{
-+ char msg[MSG_LEN];
-+ size_t off = 0;
-+ int rc;
-+ va_list args;
-+ FILE *fd;
-+
-+ if (!global_verbose)
-+ return;
-+ if (stdout_data)
-+ fd = stderr;
-+ else
-+ fd = stdout;
-+ if (global_timestamps) {
-+ rc = snprintf_timestamp(&msg[off], MSG_LEN - off);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ }
-+
-+ va_start(args, format);
-+ rc = vsnprintf(&msg[off], MSG_LEN - off, format, args);
-+ va_end(args);
-+
-+out:
-+ fprintf(fd, "%s", msg);
-+}
-+
-+/* Provide informational output. */
-+void info(const char *format, ...)
-+{
-+ char msg[MSG_LEN];
-+ size_t off = 0;
-+ int rc;
-+ va_list args;
-+ FILE *fd;
-+
-+ if (global_quiet)
-+ return;
-+ if (stdout_data)
-+ fd = stderr;
-+ else
-+ fd = stdout;
-+
-+ if (global_timestamps) {
-+ rc = snprintf_timestamp(&msg[off], MSG_LEN - off);
-+ HANDLE_RC(rc, MSG_LEN, off, out);
-+ }
-+
-+ va_start(args, format);
-+ rc = vsnprintf(&msg[off], MSG_LEN - off, format, args);
-+ va_end(args);
-+
-+out:
-+ fprintf(fd, "%s", msg);
-+}
-+
-+/* Return a newly allocated buffer containing the result of the specified
-+ * string format arguments */
-+char *__masprintf(const char *func, const char *file, int line, const char *fmt,
-+ ...)
-+{
-+ char *str;
-+ va_list args;
-+
-+ va_start(args, fmt);
-+ __util_vasprintf(func, file, line, &str, fmt, args);
-+ va_end(args);
-+
-+ return str;
-+}
-+
-+/* Set the internal name of the calling thread */
-+void __set_threadname(const char *func, const char *file, int line,
-+ const char *fmt, ...)
-+{
-+ char *str;
-+ va_list args;
-+
-+ va_start(args, fmt);
-+ __util_vasprintf(func, file, line, &str, fmt, args);
-+ va_end(args);
-+
-+ pthread_setspecific(thread_name_key, str);
-+}
-+
-+/* Clear any previously set thread name */
-+void clear_threadname(void)
-+{
-+ void *addr = pthread_getspecific(thread_name_key);
-+
-+ if (addr) {
-+ pthread_setspecific(thread_name_key, NULL);
-+ free(addr);
-+ }
-+}
-+
-+/* Remove any number of trailing characters @c in @str */
-+void chomp(char *str, char *c)
-+{
-+ ssize_t i;
-+
-+ for (i = strlen(str) - 1; i >= 0 && strchr(c, str[i]); i--)
-+ str[i] = 0;
-+}
-+
-+/* Remove any number of leading characters @c in @str */
-+void lchomp(char *str, char *c)
-+{
-+ char *from;
-+
-+ for (from = str; *from && strchr(c, *from); from++)
-+ ;
-+ if (str != from)
-+ memmove(str, from, strlen(from) + 1);
-+}
-+
-+/* Perform a stat on file referenced by either @abs or @rel and @dref. Store
-+ * results in @stat and return stat()'s return code. */
-+int stat_file(bool dereference, const char *abs, const char *rel,
-+ struct dref *dref, struct stat *st)
-+{
-+ int rc;
-+
-+ if (dref) {
-+ if (dereference)
-+ rc = fstatat(dref->dirfd, rel, st, 0);
-+ else
-+ rc = fstatat(dref->dirfd, rel, st, AT_SYMLINK_NOFOLLOW);
-+ } else {
-+ if (dereference)
-+ rc = stat(abs, st);
-+ else
-+ rc = lstat(abs, st);
-+ }
-+
-+ return rc;
-+}
-+
-+/* Fill stat buffer @st with dummy values. */
-+void set_dummy_stat(struct stat *st)
-+{
-+ /* Fake stat */
-+ memset(st, 0, sizeof(struct stat));
-+ st->st_mode = S_IRUSR | S_IWUSR | S_IFREG;
-+ st->st_uid = geteuid();
-+ st->st_gid = getegid();
-+ st->st_mtime = time(NULL);
-+}
-+
-+/* Redirect all output streams to @fd and execute command @CMD */
-+int cmd_child(int fd, char *cmd)
-+{
-+ char *argv[] = { "/bin/sh", "-c", NULL, NULL };
-+ char *env[] = { NULL };
-+
-+ argv[2] = cmd;
-+ if (dup2(fd, STDOUT_FILENO) == -1 || dup2(fd, STDERR_FILENO) == -1) {
-+ mwarn("Could not redirect command output");
-+ return EXIT_RUNTIME;
-+ }
-+
-+ execve("/bin/sh", argv, env);
-+
-+ return EXIT_RUNTIME;
-+}
-+
-+#define PIPE_READ 0
-+#define PIPE_WRITE 1
-+
-+/* Run command @cmd as a child process and store its PID in @pid_ptr. On
-+ * success, return a file descriptor that is an output pipe to the standard
-+ * output and standard error streams of the child process. Return %-1 on
-+ * error. */
-+int cmd_open(char *cmd, pid_t *pid_ptr)
-+{
-+ int pfd[2];
-+ pid_t pid;
-+
-+ if (pipe(pfd) < 0)
-+ return -1;
-+
-+ pid = fork();
-+ if (pid < 0) {
-+ /* Fork error */
-+ close(pfd[PIPE_READ]);
-+ close(pfd[PIPE_WRITE]);
-+ return -1;
-+ } else if (pid == 0) {
-+ /* Child process */
-+ close(pfd[PIPE_READ]);
-+ exit(cmd_child(pfd[PIPE_WRITE], cmd));
-+ }
-+
-+ /* Parent process */
-+ close(pfd[PIPE_WRITE]);
-+
-+ *pid_ptr = pid;
-+
-+ return pfd[PIPE_READ];
-+}
-+
-+/* Close the file descriptor @fd and end the process with PID @pid. When
-+ * not %NULL, use @status_ptr to store the resulting process status. */
-+int cmd_close(int fd, pid_t pid, int *status_ptr)
-+{
-+ int status, rc = EXIT_OK;
-+
-+ close(fd);
-+ kill(pid, SIGQUIT);
-+ if (waitpid(pid, &status, 0) == -1) {
-+ status = -errno;
-+ rc = EXIT_RUNTIME;
-+ }
-+ if (status_ptr)
-+ *status_ptr = status;
-+
-+ return rc;
-+}
-+
-+void misc_init(void)
-+{
-+ set_timespec(&main_start_ts, 0, 0);
-+ pthread_key_create(&thread_name_key, free);
-+ set_threadname("main");
-+}
-+
-+void misc_cleanup(void)
-+{
-+ clear_threadname();
-+ pthread_key_delete(thread_name_key);
-+}
-+
-+void set_stdout_data(void)
-+{
-+ stdout_data = true;
-+}
-+
-+bool starts_with(const char *str, const char *prefix)
-+{
-+ size_t len;
-+
-+ len = strlen(prefix);
-+
-+ if (strncmp(str, prefix, len) == 0)
-+ return true;
-+ return false;
-+}
-+
-+bool ends_with(const char *str, const char *suffix)
-+{
-+ size_t str_len, s_len;
-+
-+ str_len = strlen(str);
-+ s_len = strlen(suffix);
-+
-+ if (str_len < s_len)
-+ return false;
-+ if (strcmp(str + str_len - s_len, suffix) != 0)
-+ return false;
-+
-+ return true;
-+}
-+
-+/* Remove subsequent slashes in @str */
-+void remove_double_slashes(char *str)
-+{
-+ size_t i, to;
-+ char last;
-+
-+ last = 0;
-+ for (i = 0, to = 0; str[i]; i++) {
-+ if (last != '/' || str[i] != '/')
-+ last = str[to++] = str[i];
-+ }
-+ str[to] = 0;
-+}
---- /dev/null
-+++ b/dump2tar/src/strarray.c
-@@ -0,0 +1,81 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * Dynamically growing string arrays
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#include "strarray.h"
-+
-+#include
-+#include
-+#include
-+
-+#include "misc.h"
-+
-+/* Release resources associated with string array @array */
-+void free_strarray(struct strarray *array)
-+{
-+ unsigned int i;
-+
-+ for (i = 0; i < array->num; i++)
-+ free(array->str[i]);
-+ free(array->str);
-+ array->str = NULL;
-+ array->num = 0;
-+}
-+
-+/* Add string @str to string array @array */
-+void add_str_to_strarray(struct strarray *array, const char *str)
-+{
-+ array->str = mrealloc(array->str, sizeof(char *) * (array->num + 2));
-+ array->str[array->num + 1] = NULL;
-+ array->str[array->num] = mstrdup(str);
-+ array->num++;
-+}
-+
-+/* Add string resulting from @fmt and additional arguments to @array */
-+void add_vstr_to_strarray(struct strarray *array, const char *fmt, ...)
-+{
-+ va_list args;
-+ char *str;
-+
-+ va_start(args, fmt);
-+ util_vasprintf(&str, fmt, args);
-+ va_end(args);
-+
-+ array->str = mrealloc(array->str, sizeof(char *) * (array->num + 2));
-+ array->str[array->num + 1] = NULL;
-+ array->str[array->num] = str;
-+ array->num++;
-+}
-+
-+/* Add all lines in file at @filename to @array */
-+int add_file_to_strarray(struct strarray *array, const char *filename)
-+{
-+ FILE *fd;
-+ char *line = NULL;
-+ size_t line_size;
-+ int rc = EXIT_OK;
-+
-+ fd = fopen(filename, "r");
-+ if (!fd) {
-+ mwarn("%s: Cannot open file", filename);
-+ return EXIT_RUNTIME;
-+ }
-+
-+ while (!feof(fd) && !ferror(fd)) {
-+ if (getline(&line, &line_size, fd) == -1)
-+ continue;
-+ chomp(line, "\n");
-+ add_str_to_strarray(array, line);
-+ }
-+ if (ferror(fd))
-+ rc = EXIT_RUNTIME;
-+ free(line);
-+
-+ fclose(fd);
-+
-+ return rc;
-+}
---- /dev/null
-+++ b/dump2tar/src/tar.c
-@@ -0,0 +1,270 @@
-+/*
-+ * dump2tar - tool to dump files and command output into a tar archive
-+ *
-+ * TAR file generation
-+ *
-+ * Copyright IBM Corp. 2016
-+ */
-+
-+#include "tar.h"
-+
-+#include
-+#include
-+
-+#include "buffer.h"
-+#include "idcache.h"
-+#include "misc.h"
-+
-+#define LONGLINK "././@LongLink"
-+#define TYPE_LONGLINK 'K'
-+#define TYPE_LONGNAME 'L'
-+
-+#define BLOCKSIZE 512
-+
-+/* Basic TAR header */
-+struct tar_header {
-+ char name[100];
-+ char mode[8];
-+ char uid[8];
-+ char gid[8];
-+ char size[12];
-+ char mtime[12];
-+ char chksum[8];
-+ char typeflag;
-+ char linkname[100];
-+ char magic[6];
-+ char version[2];
-+ char uname[32];
-+ char gname[32];
-+ char devmajor[8];
-+ char devminor[8];
-+ char prefix[155];
-+};
-+
-+/* Store the octal value of @value to at most @len bytes at @dest */
-+static void set_octal(char *dest, size_t len, unsigned long value)
-+{
-+ int i;
-+
-+ dest[len - 1] = 0;
-+ for (i = len - 2; i >= 0; i--) {
-+ dest[i] = '0' + (value & 7);
-+ value >>= 3;
-+ }
-+}
-+
-+/* Store time @value to at most @len bytes at @dest */
-+static void set_time(char *dest, size_t len, time_t value)
-+{
-+ time_t max = (1ULL << (3 * (len - 1))) - 1;
-+
-+ if (value >= 0 && value <= max) {
-+ set_octal(dest, len, value);
-+ return;
-+ }
-+
-+ for (; len > 0; len--) {
-+ dest[len - 1] = value & 0xff;
-+ value >>= 8;
-+ }
-+
-+ dest[0] |= 0x80;
-+}
-+
-+#define SET_FIELD(obj, name, value) \
-+ set_octal((obj)->name, sizeof((obj)->name), (unsigned long) (value))
-+#define SET_TIME_FIELD(obj, name, value) \
-+ set_time((obj)->name, sizeof((obj)->name), (time_t) (value))
-+#define SET_STR_FIELD(obj, name, value) \
-+ strncpy((obj)->name, (value), sizeof((obj)->name))
-+
-+/* Initialize the tar file @header with the provided data */
-+static void init_header(struct tar_header *header, const char *filename,
-+ const char *link, size_t len, struct stat *stat,
-+ char type)
-+{
-+ unsigned int i, checksum;
-+ unsigned char *c;
-+
-+ memset(header, 0, sizeof(*header));
-+
-+ /* Fill in header fields */
-+ SET_STR_FIELD(header, name, filename);
-+ if (link)
-+ SET_STR_FIELD(header, linkname, link);
-+ SET_FIELD(header, size, len);
-+ if (stat) {
-+ SET_FIELD(header, mode, stat->st_mode & 07777);
-+ SET_FIELD(header, uid, stat->st_uid);
-+ SET_FIELD(header, gid, stat->st_gid);
-+ SET_TIME_FIELD(header, mtime, stat->st_mtime);
-+ uid_to_name(stat->st_uid, header->uname, sizeof(header->uname));
-+ gid_to_name(stat->st_gid, header->gname, sizeof(header->gname));
-+ } else {
-+ SET_FIELD(header, mode, 0644);
-+ SET_FIELD(header, uid, 0);
-+ SET_FIELD(header, gid, 0);
-+ SET_TIME_FIELD(header, mtime, 0);
-+ uid_to_name(0, header->uname, sizeof(header->uname));
-+ gid_to_name(0, header->gname, sizeof(header->gname));
-+ }
-+ header->typeflag = type;
-+ memcpy(header->magic, "ustar ", sizeof(header->magic));
-+ memcpy(header->version, " ", sizeof(header->version));
-+
-+ /* Calculate checksum */
-+ memset(header->chksum, ' ', sizeof(header->chksum));
-+ checksum = 0;
-+ c = (unsigned char *) header;
-+ for (i = 0; i < sizeof(*header); i++)
-+ checksum += c[i];
-+ snprintf(header->chksum, 7, "%06o", checksum);
-+}
-+
-+/* Emit zero bytes via @emit_cb to pad @len to a multiple of BLOCKSIZE */
-+static int emit_padding(emit_cb_t emit_cb, void *data, size_t len)
-+{
-+ size_t pad = BLOCKSIZE - len % BLOCKSIZE;
-+ char zeroes[BLOCKSIZE];
-+
-+ if (len % BLOCKSIZE > 0) {
-+ memset(zeroes, 0, BLOCKSIZE);
-+ return emit_cb(data, zeroes, pad);
-+ }
-+
-+ return 0;
-+}
-+
-+/* Emit @len bytes at @addr via @emit_cb and pad data to BLOCKSIZE with zero
-+ * bytes */
-+static int emit_data(emit_cb_t emit_cb, void *data, void *addr, size_t len)
-+{
-+ int rc;
-+
-+ if (len == 0)
-+ return 0;
-+
-+ rc = emit_cb(data, addr, len);
-+ if (rc)
-+ return rc;
-+ return emit_padding(emit_cb, data, len);
-+}
-+
-+/* Emit a tar header via @emit_cb */
-+static int emit_header(emit_cb_t emit_cb, void *data, char *filename,
-+ char *link, size_t len, struct stat *stat, char type)
-+{
-+ struct tar_header header;
-+ size_t namelen = strlen(filename);
-+ size_t linklen;
-+ int rc;
-+
-+ /* /proc can contain unreadable links which causes tar to complain
-+ * during extract - use a dummy value to handle this more gracefully */
-+ if (link && !*link)
-+ link = " ";
-+
-+ linklen = link ? strlen(link) : 0;
-+ if (linklen > sizeof(header.linkname)) {
-+ rc = emit_header(emit_cb, data, LONGLINK, NULL, linklen + 1,
-+ NULL, TYPE_LONGLINK);
-+ if (rc)
-+ return rc;
-+ rc = emit_data(emit_cb, data, link, linklen + 1);
-+ if (rc)
-+ return rc;
-+ }
-+ if (namelen > sizeof(header.name)) {
-+ rc = emit_header(emit_cb, data, LONGLINK, NULL, namelen + 1,
-+ NULL, TYPE_LONGNAME);
-+ if (rc)
-+ return rc;
-+ rc = emit_data(emit_cb, data, filename, namelen + 1);
-+ if (rc)
-+ return rc;
-+ }
-+
-+ init_header(&header, filename, link, len, stat, type);
-+ return emit_data(emit_cb, data, &header, sizeof(header));
-+}
-+
-+struct emit_content_cb_data {
-+ emit_cb_t emit_cb;
-+ void *data;
-+ size_t len;
-+ int rc;
-+};
-+
-+/* Callback for emitting a single chunk of data of a buffer */
-+static int emit_content_cb(void *data, void *addr, size_t len)
-+{
-+ struct emit_content_cb_data *cb_data = data;
-+
-+ if (len > cb_data->len)
-+ len = cb_data->len;
-+ cb_data->len -= len;
-+
-+ cb_data->rc = cb_data->emit_cb(cb_data->data, addr, len);
-+
-+ if (cb_data->rc || cb_data->len == 0)
-+ return 1;
-+
-+ return 0;
-+}
-+
-+/* Emit at most @len bytes of contents of @buffer via @emit_cb and pad output
-+ * to BLOCKSIZE with zero bytes */
-+static int emit_content(emit_cb_t emit_cb, void *data, struct buffer *buffer,
-+ size_t len)
-+{
-+ struct emit_content_cb_data cb_data;
-+
-+ cb_data.emit_cb = emit_cb;
-+ cb_data.data = data;
-+ cb_data.len = len;
-+ cb_data.rc = 0;
-+ buffer_iterate(buffer, emit_content_cb, &cb_data);
-+ if (cb_data.rc)
-+ return cb_data.rc;
-+
-+ return emit_padding(emit_cb, data, buffer->total);
-+}
-+
-+/* Convert file meta data and content specified as @content into a
-+ * stream of bytes that is reported via the @emit_cb callback. @data is
-+ * passed through to the callback for arbitrary use. */
-+int tar_emit_file_from_buffer(char *filename, char *link, size_t len,
-+ struct stat *stat, char type,
-+ struct buffer *content, emit_cb_t emit_cb,
-+ void *data)
-+{
-+ int rc;
-+
-+ DBG("emit tar file=%s type=%d len=%zu", filename, type, len);
-+ rc = emit_header(emit_cb, data, filename, link, len, stat, type);
-+ if (rc)
-+ return rc;
-+ if (content)
-+ rc = emit_content(emit_cb, data, content, len);
-+
-+ return rc;
-+}
-+
-+/* Convert file meta data and content specified as @addr and @len into a
-+ * stream of bytes that is reported via the @emit_cb callback. @data is
-+ * passed through to the callback for arbitrary use. */
-+int tar_emit_file_from_data(char *filename, char *link, size_t len,
-+ struct stat *stat, char type, void *addr,
-+ emit_cb_t emit_cb, void *data)
-+{
-+ int rc;
-+
-+ DBG("emit tar file=%s type=%d len=%zu", filename, type, len);
-+ rc = emit_header(emit_cb, data, filename, link, len, stat, type);
-+ if (rc)
-+ return rc;
-+ if (addr)
-+ rc = emit_data(emit_cb, data, addr, len);
-+
-+ return rc;
-+}
diff --git a/s390-tools-sles12sp3-dbginfo-03-dbginfo.sh-Make-use-of-sysinfo-collection-helper.patch b/s390-tools-sles12sp3-dbginfo-03-dbginfo.sh-Make-use-of-sysinfo-collection-helper.patch
deleted file mode 100644
index e37fd0a..0000000
--- a/s390-tools-sles12sp3-dbginfo-03-dbginfo.sh-Make-use-of-sysinfo-collection-helper.patch
+++ /dev/null
@@ -1,72 +0,0 @@
-Subject: [PATCH] [FEAT RTL1601] dbginfo.sh: Make use of sysfs collection helper
-From: Peter Oberparleiter
-
-Summary: dbginfo.sh: Make use of sysfs collection helper
-Description: If available, use dump2tar in dbginfo.sh for sysfs
- file collection to significantly speed up dbginfo.sh
- processing. Example speedup on an LPAR with ~190k
- sysfs files:
-
- Unpatched:
- # time ./dbginfo.sh
- real 1m19.624s
- user 0m44.985s
- sys 0m45.173s
-
- Patched:
- # time ./dbginfo.sh
- real 0m5.822s
- user 0m4.523s
- sys 0m2.636s
-Upstream-ID: -
-Problem-ID: RTL1601
-
-Signed-off-by: Peter Oberparleiter
----
- scripts/dbginfo.sh | 33 +++++++++++++++++++++------------
- 1 file changed, 21 insertions(+), 12 deletions(-)
-
---- a/scripts/dbginfo.sh
-+++ b/scripts/dbginfo.sh
-@@ -620,20 +620,29 @@ collect_sysfs() {
- fi
- fi
-
-- call_run_command "find /sys -print0 | sort -z | xargs -0 -n 10 ls -ld" "${OUTPUT_FILE_SYSFS}"
-+ # Collect sysfs files using multiple threads (-J 1) while excluding
-+ # files known to block on read (-x). Stop reading a file that takes
-+ # more than 5 seconds (-T 5) such as an active ftrace buffer.
-+ dump2tar /sys -z -o "${WORKPATH}/sysfs.tgz" -x '*/tracing/trace_pipe*' \
-+ -x '*/tracing/per_cpu/*' --ignore-failed-read -J 1 -T 5
-
-- find /sys -noleaf -type d 2>/dev/null | while IFS= read -r dir_name; do
-- mkdir -p "${WORKPATH}${dir_name}"
-- done
-+ if [ $? -ne 0 ] ; then
-+ echo "${SCRIPTNAME}: Warning: dump2tar failed or is unavailable - falling back to slow path"
-+ call_run_command "find /sys -print0 | sort -z | xargs -0 -n 10 ls -ld" "${OUTPUT_FILE_SYSFS}"
-
-- find /sys -noleaf -type f -perm /444\
-- -a -not -name "*trace_pipe*"\
-- 2>/dev/null | while IFS= read -r file_name; do
-- echo " ${file_name}"
-- if ! dd if="${file_name}" status=noxfer iflag=nonblock of="${WORKPATH}${file_name}" >/dev/null 2>&1; then
-- echo "${SCRIPTNAME}: Warning: failed to copy \"${file_name}\""
-- fi
-- done
-+ find /sys -noleaf -type d 2>/dev/null | while IFS= read -r dir_name; do
-+ mkdir -p "${WORKPATH}${dir_name}"
-+ done
-+
-+ find /sys -noleaf -type f -perm /444\
-+ -a -not -name "*trace_pipe*"\
-+ 2>/dev/null | while IFS= read -r file_name; do
-+ echo " ${file_name}"
-+ if ! dd if="${file_name}" status=noxfer iflag=nonblock of="${WORKPATH}${file_name}" >/dev/null 2>&1; then
-+ echo "${SCRIPTNAME}: Warning: failed to copy \"${file_name}\""
-+ fi
-+ done
-+ fi
-
- if test ${debugfs_mounted} -eq 1; then
- umount "${MOUNT_POINT_DEBUGFS}"
diff --git a/s390-tools-sles12sp3-dbginfo-Collect-docker-debug-data.patch b/s390-tools-sles12sp3-dbginfo-Collect-docker-debug-data.patch
deleted file mode 100644
index d62ad51..0000000
--- a/s390-tools-sles12sp3-dbginfo-Collect-docker-debug-data.patch
+++ /dev/null
@@ -1,142 +0,0 @@
-Subject: [PATCH] [BZ 151594] dbginfo: Collect docker debug output.
-From: Karsten Graul
-
-Description: dbginfo: Collect docker debug output.
-Symptom: dbginfo.sh output does not contain docker debug output.
-Problem: dbginfo.sh does not collect docker debug output.
-Solution: Add invocation of docker debug commands to dbginfo.sh.
-Reproduction: -
-Upstream-ID: -
-Problem-ID: 151594
-
-Signed-off-by: Karsten Graul
----
- scripts/dbginfo.sh | 58 +++++++++++++++++++++++++++++++++++++++++++++++++----
- 1 file changed, 54 insertions(+), 4 deletions(-)
-
---- a/scripts/dbginfo.sh
-+++ b/scripts/dbginfo.sh
-@@ -2,7 +2,7 @@
- #
- # dbginfo.sh - Tool to collect runtime, configuration, and trace information
- #
--# Copyright IBM Corp. 2002, 2016
-+# Copyright IBM Corp. 2002, 2017
- #
-
- # Switching to neutral locale
-@@ -18,7 +18,7 @@ readonly SCRIPTNAME="${0##*/}"
- print_version() {
- cat </dev/null | cut -d'.' -f1)
-@@ -301,6 +304,8 @@ LOGFILES="\
- /var/log/yum.log\
- /var/log/openvswitch/ovs-vswitchd.log\
- /var/log/openvswitch/ovsdb-server.log\
-+ /var/run/docker/libcontainerd/containerd/events.log\
-+ /run/containerd/events.log\
- "
-
- ########################################
-@@ -351,6 +356,9 @@ CONFIGFILES="\
- /etc/xinet.d\
- /etc/*release\
- $(find /lib/modules -name modules.dep 2>/dev/null)\
-+ /etc/docker\
-+ /usr/lib/systemd\
-+ /etc/apparmor.d\
- "
-
- ########################################
-@@ -436,6 +444,16 @@ CMDS="uname -a\
- :systemctl --all --no-pager show\
- :systemctl --all --no-pager list-units\
- :systemctl --all --no-pager list-unit-files\
-+ :docker info\
-+ :docker images\
-+ :docker network ls\
-+ :docker ps\
-+ :docker ps -a\
-+ :docker stats --no-stream\
-+ :docker version\
-+ :systemctl status docker.service\
-+ :df /var/lib/docker\
-+ :df --inodes /var/lib/docker\
- "
-
- ########################################
-@@ -767,13 +785,43 @@ collect_domain_xml() {
- }
-
- ########################################
-+collect_docker() {
-+ local item_list
-+ local item
-+
-+ # call docker inspect for all containers
-+ item_list=$(docker ps -qa)
-+ if test -n "${item_list}"; then
-+ pr_syslog_stdout "10a of ${COLLECTION_COUNT}: Collecting docker container output"
-+ for item in ${item_list}; do
-+ call_run_command "docker inspect ${item}" "${OUTPUT_FILE_DOCKER}"
-+ done
-+ else
-+ pr_syslog_stdout "10a of ${COLLECTION_COUNT}: Collecting docker container output skipped"
-+ fi
-+
-+ # call docker inspect for all networks
-+ item_list=$(docker network ls -q)
-+ if test -n "${item_list}"; then
-+ pr_syslog_stdout "10b of ${COLLECTION_COUNT}: Collecting docker network output"
-+ for item in ${item_list}; do
-+ call_run_command "docker network inspect ${item}" "${OUTPUT_FILE_DOCKER}"
-+ done
-+ else
-+ pr_syslog_stdout "10b of ${COLLECTION_COUNT}: Collecting docker network output skipped"
-+ fi
-+
-+ pr_log_stdout " "
-+}
-+
-+########################################
- post_processing() {
- local file_mtime
- local file_mtime_epoche
- local tmp_file
- local file_name
-
-- pr_syslog_stdout "10 of ${COLLECTION_COUNT}: Postprocessing"
-+ pr_syslog_stdout "11 of ${COLLECTION_COUNT}: Postprocessing"
-
- find "${WORKPATH}etc/libvirt/qemu/" -maxdepth 1 -name "*.xml" 2>/dev/null | while IFS= read -r file_name; do
- file_mtime_epoche=$(stat --format=%Y "${file_name}")
-@@ -1056,6 +1104,8 @@ collect_ovs
-
- collect_domain_xml
-
-+collect_docker
-+
- post_processing
-
- create_package
diff --git a/s390-tools-sles12sp3-lsdasd-tunedasd-Add-channel-path-aware-erp.patch b/s390-tools-sles12sp3-lsdasd-tunedasd-Add-channel-path-aware-erp.patch
deleted file mode 100644
index b8a89d3..0000000
--- a/s390-tools-sles12sp3-lsdasd-tunedasd-Add-channel-path-aware-erp.patch
+++ /dev/null
@@ -1,558 +0,0 @@
-Subject: [PATCH] [FEAT LS1502] lsdasd/tunedasd: Add channel path aware erp
-From: Stefan Haberland
-
-Summary: lsdasd/tunedasd: Add channel path aware erp
-Description: Add tools support for channel path aware error recovery.
- lsdasd shows the newly added IFCC path mask and HPF availability.
- tunedasd gets an option to reset channel paths for a device.
-Upstream-ID: -
-Problem-ID: LS1502
-
-Signed-off-by: Stefan Haberland
----
- include/dasd_sys.h | 1
- libdasd/dasd_sys.c | 94 +++++++++++++++++++++++++++++++++++++
- tunedasd/include/disk.h | 3 -
- tunedasd/man/tunedasd.8 | 17 ++++++
- tunedasd/src/Makefile | 2
- tunedasd/src/disk.c | 35 ++++++++++++++
- tunedasd/src/tunedasd.c | 120 ++++++++++++++++++++++++++++--------------------
- zconf/lsdasd | 36 +++++++++++---
- 8 files changed, 251 insertions(+), 57 deletions(-)
-
---- a/include/dasd_sys.h
-+++ b/include/dasd_sys.h
-@@ -13,5 +13,6 @@
- #include "u2s.h"
-
- int dasd_sys_raw_track_access(char *);
-+int dasd_reset_chpid(char *, char *);
-
- #endif /* DASD_SYS_H */
---- a/libdasd/dasd_sys.c
-+++ b/libdasd/dasd_sys.c
-@@ -6,6 +6,9 @@
- * Copyright IBM Corp. 2016
- */
-
-+#include
-+#include
-+
- #include "dasd_sys.h"
-
- /**
-@@ -44,3 +47,94 @@ int dasd_sys_raw_track_access(char *devn
-
- return (rc == 1) ? 1 : 0;
- }
-+
-+
-+int dasd_get_pm_from_chpid(char *busid, unsigned int chpid, int *mask)
-+{
-+ unsigned int val;
-+ char path[40];
-+ int count, i;
-+ FILE *fp;
-+
-+ sprintf(path, "/sys/bus/ccw/devices/%s/../chpids", busid);
-+ *mask = 0;
-+ fp = fopen(path, "r");
-+ if (!fp)
-+ return ENODEV;
-+
-+ for (i = 0; i < 8; i++) {
-+ count = fscanf(fp, " %x", &val);
-+ if (count != 1) {
-+ fclose(fp);
-+ return EIO;
-+ }
-+ if (val == chpid)
-+ *mask = 0x80 >> i;
-+ }
-+ fclose(fp);
-+
-+ return 0;
-+}
-+
-+/**
-+ * reset chpid
-+ *
-+ * The "devnode" parameter can be any valid relative or absolute path to
-+ * a DASD device node, for example:
-+ *
-+ * - /dev/dasda
-+ * - /dev/disk/by-path/ccw-0.0.bf20
-+ *
-+ * @param[in] devnode Device node of interest
-+ * @param[in] chpid The chpid to reset
-+ * If NULL all chpids will be reset
-+ *
-+ * @return 0 on success, otherwise one of the following error codes:
-+ * - EINVAL No valid chpid specified.
-+ * - ENODEV Could not open device.
-+ * - ENOENT Specified chpid not found.
-+ * - EIO Other I/O error
-+ *
-+ */
-+int dasd_reset_chpid(char *devnode, char *chpid_char)
-+{
-+ unsigned int chpid;
-+ char path[41];
-+ char busid[9];
-+ int mask, rc;
-+ char *endptr;
-+ FILE *fp;
-+
-+ if (u2s_getbusid(devnode, busid))
-+ return ENODEV;
-+
-+ if (!chpid_char) {
-+ sprintf(path, "/sys/bus/ccw/devices/%s/path_reset", busid);
-+ fp = fopen(path, "w");
-+ if (!fp)
-+ return ENODEV;
-+ fprintf(fp, "%s", "all\n");
-+ fclose(fp);
-+ return 0;
-+ }
-+
-+ errno = 0;
-+ chpid = strtoul(chpid_char, &endptr, 16);
-+ if (errno || (endptr && (*endptr != '\0')))
-+ return EINVAL;
-+
-+ rc = dasd_get_pm_from_chpid(busid, chpid, &mask);
-+ if (rc)
-+ return rc;
-+ if (!mask)
-+ return ENOENT;
-+
-+ sprintf(path, "/sys/bus/ccw/devices/%s/path_reset", busid);
-+ fp = fopen(path, "w");
-+ if (!fp)
-+ return ENODEV;
-+ fprintf(fp, "%02x", mask);
-+ fclose(fp);
-+
-+ return 0;
-+}
---- a/tunedasd/include/disk.h
-+++ b/tunedasd/include/disk.h
-@@ -27,7 +27,8 @@ int disk_release (char* device);
- int disk_slock (char* device);
- int disk_query_reserve_status(char* device);
- int disk_profile (char* device, char* prof_item);
--int disk_reset_prof (char* device);
-+int disk_reset_prof(char *device);
-+int disk_reset_chpid(char *device, char *chpid);
-
- #endif /* not DISK_H */
-
---- a/tunedasd/man/tunedasd.8
-+++ b/tunedasd/man/tunedasd.8
-@@ -149,6 +149,18 @@ Following rows are supported:
- .TP
- .BR "\-R" " or " "\-\-reset_prof"
- Reset profile info of device.
-+.TP
-+.BR "\-p" " or " "\-\-path_reset "
-+Reset a channel path of a selected device. A channel path
-+might be suspended due to high IFCC error rates or a High Performance
-+FICON failure. Use this option to resume considering the channel path
-+for I/O.
-+.TP
-+.BR "\-\-path_reset_all"
-+Reset all channel paths of the selected device. The channel paths
-+might be suspended due to high IFCC error rates or a High Performance
-+FICON failure. Use this option to resume considering all defined
-+channel paths for I/O.
- .\"
- .\".TP
- .\".BR "\-o" " or " "\-\-online"
-@@ -166,8 +178,13 @@ Reset profile info of device.
- .br
-
- tunedasd -c prestage -n 1 /dev/dasdc
-+
-+.br
-+3. Scenario: Reset failed channel path with CHPID 45
- .br
-
-+ tunedasd -p 45 /dev/dasdc
-+.br
- .SH "SEE ALSO"
- .BR dasdview (8),
- .BR dasdfmt (8),
---- a/tunedasd/src/Makefile
-+++ b/tunedasd/src/Makefile
-@@ -5,7 +5,7 @@ includes = $(wildcard ../include/*.h)
-
- all: tunedasd
-
--objects = tunedasd.o disk.o
-+objects = tunedasd.o disk.o ../../libdasd/dasd_sys.o ../../libu2s/u2s.o
- $(objects): $(includes)
-
- tunedasd: $(objects)
---- a/tunedasd/src/disk.c
-+++ b/tunedasd/src/disk.c
-@@ -9,6 +9,7 @@
-
- #include "disk.h"
- #include "tunedasd.h"
-+#include "dasd_sys.h"
-
- #include
- #include
-@@ -682,3 +683,37 @@ disk_reset_prof (char* device)
- return 0;
- }
-
-+int disk_reset_chpid(char *device, char *chpid)
-+{
-+ int rc;
-+
-+ if (chpid)
-+ printf("Resetting chpid %s for device <%s>...\n", chpid,
-+ device);
-+ else
-+ printf("Resetting all chpids for device <%s>...\n", device);
-+
-+ rc = dasd_reset_chpid(device, chpid);
-+ switch (rc) {
-+ case 0:
-+ printf("Done.\n");
-+ return 0;
-+ case ENODEV:
-+ error_print("%s: %s", device, strerror(errno));
-+ break;
-+ case EINVAL:
-+ error_print("%s: Could not reset chpid %s: Invalid CHPID",
-+ device, chpid);
-+ break;
-+ case ENOENT:
-+ error_print("%s: Could not reset chpid %s: CHPID not defined for device",
-+ device, chpid);
-+ break;
-+ default:
-+ error_print("%s: Could not reset chpid %s",
-+ device, chpid);
-+ break;
-+ }
-+
-+ return -1;
-+}
---- a/tunedasd/src/tunedasd.c
-+++ b/tunedasd/src/tunedasd.c
-@@ -34,28 +34,30 @@ static const char* usage_text[] = {
- "(e.g. '/dev/dasda') or a list of devices separated by a space "
- "character.",
- "",
-- "-h, --help Print this help, then exit",
-- "-v, --version Print version information, then exit",
-- "-g, --get_cache Get current storage server caching behaviour",
-- "-c, --cache Define caching behavior on storage server",
-- " (normal/bypass/inhibit/sequential/prestage/"
-- "record)",
-- "-n, --no_cyl Number of cylinders to be cached ",
-- " (only valid together with --cache)",
-- "-S, --reserve Reserve device",
-- "-L, --release Release device",
-- "-O, --slock Unconditional reserve device",
-- " Note: Use with care, this breaks an existing "
-- "lock",
-- "-Q, --query_reserve Print reserve status of device ",
-- "-P, --profile Print profile info of device",
-- "-I, --prof_item Print single profile item",
-- " (reqs/sects/sizes/total/totsect/start/irq/",
-- " irqsect/end/queue)",
-- "-R, --reset_prof Reset profile info of device"
-+ "-h, --help Print this help, then exit",
-+ "-v, --version Print version information, then exit",
-+ "-g, --get_cache Get current storage server caching behaviour",
-+ "-c, --cache Define caching behavior on storage server",
-+ " (normal/bypass/inhibit/sequential/prestage/"
-+ "record)",
-+ "-n, --no_cyl Number of cylinders to be cached ",
-+ " (only valid together with --cache)",
-+ "-S, --reserve Reserve device",
-+ "-L, --release Release device",
-+ "-O, --slock Unconditional reserve device",
-+ " Note: Use with care, this breaks an existing "
-+ "lock",
-+ "-Q, --query_reserve Print reserve status of device ",
-+ "-P, --profile Print profile info of device",
-+ "-I, --prof_item Print single profile item",
-+ " (reqs/sects/sizes/total/totsect/start/irq/",
-+ " irqsect/end/queue)",
-+ "-R, --reset_prof Reset profile info of device",
-+ "-p, --path_reset Reset channel path of a device",
-+ " --path_reset_all Reset all channel paths of a device"
- };
-
--#define CMD_KEYWORD_NUM 12
-+#define CMD_KEYWORD_NUM 14
- #define DEVICES_NUM 256
-
- enum cmd_keyword_id {
-@@ -71,6 +73,8 @@ enum cmd_keyword_id {
- cmd_keyword_prof_item = 9,
- cmd_keyword_reset_prof = 10,
- cmd_keyword_query_reserve = 11,
-+ cmd_keyword_path = 12,
-+ cmd_keyword_path_all = 13,
- };
-
-
-@@ -79,18 +83,20 @@ static const struct {
- char* keyword;
- enum cmd_keyword_id id;
- } keyword_list[] = {
-- { "help", cmd_keyword_help },
-- { "version", cmd_keyword_version },
-- { "get_cache", cmd_keyword_get_cache },
-- { "cache", cmd_keyword_cache },
-- { "no_cyl", cmd_keyword_no_cyl },
-- { "reserve", cmd_keyword_reserve },
-- { "release", cmd_keyword_release },
-- { "slock", cmd_keyword_slock },
-- { "profile", cmd_keyword_profile },
-- { "prof_item", cmd_keyword_prof_item },
-- { "reset_prof", cmd_keyword_reset_prof },
-- { "query_reserve", cmd_keyword_query_reserve }
-+ { "help", cmd_keyword_help },
-+ { "version", cmd_keyword_version },
-+ { "get_cache", cmd_keyword_get_cache },
-+ { "cache", cmd_keyword_cache },
-+ { "no_cyl", cmd_keyword_no_cyl },
-+ { "reserve", cmd_keyword_reserve },
-+ { "release", cmd_keyword_release },
-+ { "slock", cmd_keyword_slock },
-+ { "profile", cmd_keyword_profile },
-+ { "prof_item", cmd_keyword_prof_item },
-+ { "reset_prof", cmd_keyword_reset_prof },
-+ { "query_reserve", cmd_keyword_query_reserve },
-+ { "path_reset", cmd_keyword_path },
-+ { "path_reset_all", cmd_keyword_path_all }
- };
-
-
-@@ -103,22 +109,24 @@ enum cmd_key_state {
-
- /* Determines which combination of keywords are valid */
- enum cmd_key_state cmd_key_table[CMD_KEYWORD_NUM][CMD_KEYWORD_NUM] = {
-- /* help vers get_ cach no_c rese rele sloc prof prof rese quer
-- * ion cach e yl rve ase k ile _ite t_pr y_re
-+ /* help vers get_ cach no_c rese rele sloc prof prof rese quer path path
-+ * ion cach e yl rve ase k ile _ite t_pr y_re _all
- * e m of serv
- */
-- /* help */ { req, opt, opt, opt, opt, opt, opt, opt, opt, opt, opt, inv },
-- /* version */ { inv, req, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv },
-- /* get_cache */ { opt, opt, req, inv, inv, inv, inv, inv, inv, inv, inv, inv },
-- /* cache */ { opt, opt, inv, req, opt, inv, inv, inv, inv, inv, inv, inv },
-- /* no_cyl */ { opt, opt, inv, req, req, inv, inv, inv, inv, inv, inv, inv },
-- /* reserve */ { opt, opt, inv, inv, inv, req, inv, inv, inv, inv, inv, inv },
-- /* release */ { opt, opt, inv, inv, inv, inv, req, inv, inv, inv, inv, inv },
-- /* slock */ { opt, opt, inv, inv, inv, inv, inv, req, inv, inv, inv, inv },
-- /* profile */ { opt, opt, inv, inv, inv, inv, inv, inv, req, opt, inv, inv },
-- /* prof_item */ { opt, opt, inv, inv, inv, inv, inv, inv, req, req, inv, inv },
-- /* reset_prof */ { opt, opt, inv, inv, inv, inv, inv, inv, inv, inv, req, inv },
-- /* query_reserve */ { inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, req },
-+ /* help */ { req, opt, opt, opt, opt, opt, opt, opt, opt, opt, opt, inv, inv, inv },
-+ /* version */ { inv, req, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv },
-+ /* get_cache */ { opt, opt, req, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv },
-+ /* cache */ { opt, opt, inv, req, opt, inv, inv, inv, inv, inv, inv, inv, inv, inv },
-+ /* no_cyl */ { opt, opt, inv, req, req, inv, inv, inv, inv, inv, inv, inv, inv, inv },
-+ /* reserve */ { opt, opt, inv, inv, inv, req, inv, inv, inv, inv, inv, inv, inv, inv },
-+ /* release */ { opt, opt, inv, inv, inv, inv, req, inv, inv, inv, inv, inv, inv, inv },
-+ /* slock */ { opt, opt, inv, inv, inv, inv, inv, req, inv, inv, inv, inv, inv, inv },
-+ /* profile */ { opt, opt, inv, inv, inv, inv, inv, inv, req, opt, inv, inv, inv, inv },
-+ /* prof_item */ { opt, opt, inv, inv, inv, inv, inv, inv, req, req, inv, inv, inv, inv },
-+ /* reset_prof */ { opt, opt, inv, inv, inv, inv, inv, inv, inv, inv, req, inv, inv, inv },
-+ /* query_reserve */ { inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, req, inv, inv },
-+ /* path */ { inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, req, inv },
-+ /* path_all */ { inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, inv, req },
- };
-
- struct parameter {
-@@ -146,11 +154,13 @@ static struct option options[] = {
- { "prof_item", required_argument, NULL, 'I'},
- { "reset_prof", no_argument, NULL, 'R'},
- { "query_reserve", no_argument, NULL, 'Q'},
-+ { "path_reset", required_argument, NULL, 'p'},
-+ { "path_reset_all", no_argument, NULL, 'A'},
- { NULL, 0, NULL, 0 }
- };
-
- /* Command line option abbreviations */
--static const char option_string[] = "hvgc:n:SLOPI:RQ";
-+static const char option_string[] = "hvgc:n:SLOPI:RQp:";
-
-
- /* Error message string */
-@@ -348,6 +358,14 @@ get_command_line (int argc, char* argv[]
- optarg);
- }
- break;
-+ case 'p':
-+ rc = store_option(&cmdline, cmd_keyword_path,
-+ optarg);
-+ break;
-+ case 'A':
-+ rc = store_option(&cmdline, cmd_keyword_path_all,
-+ optarg);
-+ break;
- case 'S':
- rc = store_option (&cmdline, cmd_keyword_reserve,
- optarg);
-@@ -400,7 +418,6 @@ get_command_line (int argc, char* argv[]
- return 0;
- }
-
--
- /*
- * Execute the command.
- */
-@@ -444,6 +461,13 @@ do_command (char* device, struct command
- case cmd_keyword_query_reserve:
- rc = disk_query_reserve_status(device);
- break;
-+ case cmd_keyword_path:
-+ rc = disk_reset_chpid(device,
-+ cmdline.parm[cmd_keyword_path].data);
-+ break;
-+ case cmd_keyword_path_all:
-+ rc = disk_reset_chpid(device, NULL);
-+ break;
- default:
- error_print ("Unknown command '%s' specified",
- get_keyword_name (i));
---- a/zconf/lsdasd
-+++ b/zconf/lsdasd
-@@ -368,13 +368,15 @@ function extended()
- CABLEPM=0
- CUIRPM=0
- HPFPM=0
-+ IFCCPM=0
-
- # additional information
- read DIAG 2> /dev/null < $DEVPATH/use_diag || continue
- read EER 2> /dev/null < $DEVPATH/eer_enabled || continue
- read ERP 2> /dev/null < $DEVPATH/erplog || continue
-+ read HPF 2> /dev/null < $DEVPATH/hpf
- # in case the path_masks do not exist simply ignore it
-- read OPM NPPM CABLEPM CUIRPM HPFPM 2> /dev/null < $DEVPATH/path_masks
-+ read OPM NPPM CABLEPM CUIRPM HPFPM IFCCPM 2> /dev/null < $DEVPATH/path_masks
- read -a C 2> /dev/null < $DEVPATH/../chpids || continue
- read PIM PAM POM 2> /dev/null < $DEVPATH/../pimpampom || continue
-
-@@ -385,6 +387,7 @@ function extended()
- CABLEPM=0x$CABLEPM
- CUIRPM=0x$CUIRPM
- HPFPM=0x$HPFPM
-+ IFCCPM=0x$IFCCPM
-
- #-----------------------------------------------------------#
- # aggregate chpids and path mask to useful information #
-@@ -397,6 +400,7 @@ function extended()
- CUIR_PATHS=(" " " " " " " " " " " " " " " ")
- CABLE_PATHS=(" " " " " " " " " " " " " " " ")
- HPF_PATHS=(" " " " " " " " " " " " " " " ")
-+ IFCC_PATHS=(" " " " " " " " " " " " " " " ")
-
- # installed paths
- j=0
-@@ -470,13 +474,25 @@ function extended()
- (( mask>>=1 ))
- done
-
-+ # IFCC unusable paths
-+ j=0
-+ mask=0x80
-+ for (( i=0; i<8; i++ )) ;do
-+ PM=$(($IFCCPM&$mask))
-+ if [ $PM -gt 0 ] ;then
-+ IFCC_PATHS[j]=${C[$i]} ;
-+ ((j++)) ;
-+ fi
-+ (( mask>>=1 ))
-+ done
-+
- #-------------------------------------------#
- # format data for output #
- #-------------------------------------------#
-
- if [[ "$ONLINE" == 0 ]]; then
- ACTIVE="offline"
-- printf "%s:%s:%s# status:\t\t\t\t%s# use_diag:\t\t\t\t%s# readonly:\t\t\t\t%s# eer_enabled:\t\t\t\t%s# erplog:\t\t\t\t%s# uid: \t\t\t\t%s# paths_installed: \t\t\t%s %s %s %s %s %s %s %s# paths_in_use: \t\t\t%s %s %s %s %s %s %s %s# paths_non_preferred: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_cabling: \t\t%s %s %s %s %s %s %s %s# paths_cuir_quiesced: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_hpf_characteristics: \t%s %s %s %s %s %s %s %s#\n" \
-+ printf "%s:%s:%s# status:\t\t\t\t%s# use_diag:\t\t\t\t%s# readonly:\t\t\t\t%s# eer_enabled:\t\t\t\t%s# erplog:\t\t\t\t%s# hpf:\t\t\t\t\t%s# uid: \t\t\t\t%s# paths_installed: \t\t\t%s %s %s %s %s %s %s %s# paths_in_use: \t\t\t%s %s %s %s %s %s %s %s# paths_non_preferred: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_cabling: \t\t%s %s %s %s %s %s %s %s# paths_cuir_quiesced: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_hpf_characteristics: \t%s %s %s %s %s %s %s %s# paths_error_threshold_exceeded: \t%s %s %s %s %s %s %s %s#\n" \
- "$SORTKEYLEN" "$SORTKEY" \
- "$BUSID" \
- "$ACTIVE" \
-@@ -484,18 +500,20 @@ function extended()
- "$READONLY" \
- "$EER" \
- "$ERP" \
-+ "$HPF" \
- "$DEV_UID" \
- "${INSTALLED_PATHS[@]}" \
- "${USED_PATHS[@]}" \
- "${NP_PATHS[@]}" \
- "${CABLE_PATHS[@]}" \
- "${CUIR_PATHS[@]}" \
-- "${HPF_PATHS[@]}" ;
-+ "${HPF_PATHS[@]}" \
-+ "${IFCC_PATHS[@]}" ;
- continue
- elif [[ "$ALIAS" == 1 ]]; then
- if [[ "$BASEONLY" == "false" ]]; then
- ACTIVE="alias"
-- printf "%s:%s:%s# status:\t\t\t\t%s# type: \t\t\t\t%s# use_diag:\t\t\t\t%s# readonly:\t\t\t\t%s# eer_enabled:\t\t\t\t%s# erplog:\t\t\t\t%s # uid: \t\t\t\t%s# paths_installed: \t\t\t%s %s %s %s %s %s %s %s# paths_in_use: \t\t\t%s %s %s %s %s %s %s %s# paths_non_preferred: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_cabling: \t\t%s %s %s %s %s %s %s %s# paths_cuir_quiesced: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_hpf_characteristics: \t%s %s %s %s %s %s %s %s#\n" \
-+ printf "%s:%s:%s# status:\t\t\t\t%s# type: \t\t\t\t%s# use_diag:\t\t\t\t%s# readonly:\t\t\t\t%s# eer_enabled:\t\t\t\t%s# erplog:\t\t\t\t%s# hpf:\t\t\t\t\t%s # uid: \t\t\t\t%s# paths_installed: \t\t\t%s %s %s %s %s %s %s %s# paths_in_use: \t\t\t%s %s %s %s %s %s %s %s# paths_non_preferred: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_cabling: \t\t%s %s %s %s %s %s %s %s# paths_cuir_quiesced: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_hpf_characteristics: \t%s %s %s %s %s %s %s %s# paths_error_threshold_exceeded: \t%s %s %s %s %s %s %s %s#\n" \
- "$SORTKEYLEN" "$SORTKEY" \
- "$BUSID" \
- "$ACTIVE" \
-@@ -504,13 +522,15 @@ function extended()
- "$READONLY" \
- "$EER" \
- "$ERP" \
-+ "$HPF" \
- "$DEV_UID" \
- "${INSTALLED_PATHS[@]}" \
- "${USED_PATHS[@]}" \
- "${NP_PATHS[@]}" \
- "${CABLE_PATHS[@]}" \
- "${CUIR_PATHS[@]}" \
-- "${HPF_PATHS[@]}" ;
-+ "${HPF_PATHS[@]}" \
-+ "${IFCC_PATHS[@]}" ;
- continue
- else
- continue
-@@ -533,7 +553,7 @@ function extended()
- COLON=":"
- fi
-
-- printf "%s:%s:%s/%s/%s%s%s# status:\t\t\t\t%s# type: \t\t\t\t%s# blksz:\t\t\t\t%s# size: \t\t\t\t%s# blocks:\t\t\t\t%s# use_diag:\t\t\t\t%s# readonly:\t\t\t\t%s# eer_enabled:\t\t\t\t%s# erplog:\t\t\t\t%s# uid: \t\t\t\t%s# paths_installed: \t\t\t%s %s %s %s %s %s %s %s# paths_in_use: \t\t\t%s %s %s %s %s %s %s %s# paths_non_preferred: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_cabling: \t\t%s %s %s %s %s %s %s %s# paths_cuir_quiesced: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_hpf_characteristics: \t%s %s %s %s %s %s %s %s#\n" \
-+ printf "%s:%s:%s/%s/%s%s%s# status:\t\t\t\t%s# type: \t\t\t\t%s# blksz:\t\t\t\t%s# size: \t\t\t\t%s# blocks:\t\t\t\t%s# use_diag:\t\t\t\t%s# readonly:\t\t\t\t%s# eer_enabled:\t\t\t\t%s# erplog:\t\t\t\t%s# hpf:\t\t\t\t\t%s# uid: \t\t\t\t%s# paths_installed: \t\t\t%s %s %s %s %s %s %s %s# paths_in_use: \t\t\t%s %s %s %s %s %s %s %s# paths_non_preferred: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_cabling: \t\t%s %s %s %s %s %s %s %s# paths_cuir_quiesced: \t\t\t%s %s %s %s %s %s %s %s# paths_invalid_hpf_characteristics: \t%s %s %s %s %s %s %s %s# paths_error_threshold_exceeded: \t%s %s %s %s %s %s %s %s#\n" \
- "$SORTKEYLEN" "$SORTKEY" \
- "$BUSID" \
- "$BLOCKNAME" \
-@@ -549,13 +569,15 @@ function extended()
- "$READONLY" \
- "$EER" \
- "$ERP" \
-+ "$HPF" \
- "$DEV_UID" \
- "${INSTALLED_PATHS[@]}" \
- "${USED_PATHS[@]}" \
- "${NP_PATHS[@]}" \
- "${CABLE_PATHS[@]}" \
- "${CUIR_PATHS[@]}" \
-- "${HPF_PATHS[@]}" ;
-+ "${HPF_PATHS[@]}" \
-+ "${IFCC_PATHS[@]}" ;
- }
-
- function host()
diff --git a/s390-tools-sles12sp3-mon_fsstatd-fix-double-free-in-error-path-and-skip-virtual-fs.patch b/s390-tools-sles12sp3-mon_fsstatd-fix-double-free-in-error-path-and-skip-virtual-fs.patch
deleted file mode 100644
index 0a5236a..0000000
--- a/s390-tools-sles12sp3-mon_fsstatd-fix-double-free-in-error-path-and-skip-virtual-fs.patch
+++ /dev/null
@@ -1,65 +0,0 @@
-Subject: [PATCH] [BZ 151416] mon_fsstatd: fix double free in error path and skip virtual fs
-From: Gerald Schaefer
-
-Description: mon_fsstatd: fix double free in error path and skip virtual fs
-Symptom: The daemon is stopped and the following messages can be found
- in syslog: "statvfs error on /run/user/0: Permission denied"
- and "double free or corruption (!prev)".
-Problem: The error handling code for a failing statvfs() call erroneously
- does an extra fclose(), conflicting with the later fclose().
- Also, breaking out of the inner loop at this point doesn't make
- sense, because there may be other mounted filesystems that would
- be skipped.
- The reason for the failing statvfs() (on /run/user/0) is a wrong
- selinux context (when started via systemctl), and this error will
- spam the syslog if we only fix the double free. Since /run/user/0
- is a tmpfs, and mon_fsstatd is supposed to collect only physical
- filesystem data, it is wrong to even try to access /run/user/0.
-Solution: Remove the extra fclose(), replace the break with a continue,
- and add tmpfs (and others) to the list of ignored virtual
- filesystems.
-Reproduction: Start mon_fsstatd via service/systemctl.
-Upstream-ID: -
-Problem-ID: 151416
-
-Signed-off-by: Gerald Schaefer
----
- mon_tools/mon_fsstatd.c | 19 +++++++++++++++++--
- 1 file changed, 17 insertions(+), 2 deletions(-)
-
---- a/mon_tools/mon_fsstatd.c
-+++ b/mon_tools/mon_fsstatd.c
-@@ -323,16 +323,31 @@ static int fsstatd_do_work(void)
- strncmp(ent->mnt_type, "none", 4) == 0 ||
- strncmp(ent->mnt_type, "proc", 4) == 0 ||
- strncmp(ent->mnt_type, "subfs", 5) == 0 ||
-+ strncmp(ent->mnt_type, "nfsd", 4) == 0 ||
-+ strncmp(ent->mnt_type, "tmpfs", 5) == 0 ||
-+ strncmp(ent->mnt_type, "sysfs", 5) == 0 ||
-+ strncmp(ent->mnt_type, "pstore", 6) == 0 ||
-+ strncmp(ent->mnt_type, "cgroup", 6) == 0 ||
-+ strncmp(ent->mnt_type, "mqueue", 6) == 0 ||
-+ strncmp(ent->mnt_type, "devpts", 6) == 0 ||
-+ strncmp(ent->mnt_type, "debugfs", 7) == 0 ||
-+ strncmp(ent->mnt_type, "devtmpfs", 8) == 0 ||
-+ strncmp(ent->mnt_type, "configfs", 8) == 0 ||
-+ strncmp(ent->mnt_type, "selinuxfs", 9) == 0 ||
-+ strncmp(ent->mnt_type, "hugetlbfs", 9) == 0 ||
-+ strncmp(ent->mnt_type, "securityfs", 10) == 0 ||
-+ strncmp(ent->mnt_type, "rpc_pipefs", 10) == 0 ||
-+ strncmp(ent->mnt_type, "binfmt_misc", 11) == 0 ||
- strncmp(ent->mnt_type, "ignore", 6) == 0)) {
- ent = getmntent(mnttab);
- continue;
- }
- result = statvfs(ent->mnt_dir, &buf);
- if (result != 0) {
-- fclose(mnttab);
- syslog(LOG_ERR, "statvfs error on %s: %s\n",
- ent->mnt_dir, strerror(errno));
-- break;
-+ ent = getmntent(mnttab);
-+ continue;
- }
-
- if (buf.f_blocks > 0)
diff --git a/s390-tools-sles12sp3-util_proc-fix-memory-allocation-error-messages.patch b/s390-tools-sles12sp3-util_proc-fix-memory-allocation-error-messages.patch
deleted file mode 100644
index 8b19b8a..0000000
--- a/s390-tools-sles12sp3-util_proc-fix-memory-allocation-error-messages.patch
+++ /dev/null
@@ -1,30 +0,0 @@
-util_proc: Fix memory allocation error messages
-
-Signed-off-by: Michael Holzheu
----
- libutil/util_proc.c | 7 +++----
- 1 file changed, 3 insertions(+), 4 deletions(-)
-
---- a/libutil/util_proc.c
-+++ b/libutil/util_proc.c
-@@ -54,8 +54,7 @@ util_proc_read_special_file(const char *
- count = 0;
- data = (char *) malloc(current_size);
- if (data == NULL) {
-- printf("Could not allocate %lld bytes of memory",
-- (unsigned long long) size);
-+ printf("Could not allocate %zu bytes of memory", current_size);
- fclose(file);
- return -1;
- }
-@@ -69,8 +68,8 @@ util_proc_read_special_file(const char *
- if (count >= current_size) {
- new_data = (char *) malloc(current_size * 2);
- if (new_data == NULL) {
-- printf("Could not allocate %lld bytes of memory",
-- (unsigned long long) size);
-+ printf("Could not allocate %zu bytes of memory",
-+ current_size * 2);
- free(data);
- fclose(file);
- return -1;
diff --git a/s390-tools-sles12sp3-ziomon-no-blktrace-kill-which-can-corrupt-kernel-blk.patch b/s390-tools-sles12sp3-ziomon-no-blktrace-kill-which-can-corrupt-kernel-blk.patch
deleted file mode 100644
index 7782dd1..0000000
--- a/s390-tools-sles12sp3-ziomon-no-blktrace-kill-which-can-corrupt-kernel-blk.patch
+++ /dev/null
@@ -1,144 +0,0 @@
-Subject: [PATCH] [BZ 149058] ziomon: no blktrace kill which can corrupt kernel blktrace state
-From: Steffen Maier
-
-Description: ziomon: no blktrace kill which can corrupt kernel blktrace state
-Symptom: Ziomon terminates with the following error messages:
-
- $ ziomon -d -o ...
- ...
- ziomon: Failed to stop trace on /dev/sdX
- ...
- ziomon: Failed to stop trace on /dev/sdY
- ziomon: blktrace has leftovers, manual cleanup required!
-
- Subsequent ziomon runs fail likely until the next reboot:
-
- $ ziomon -d -o ...
- ...
- Collecting data...BLKTRACESETUP(2) /dev/sdX failed: \
- 2/No such file or directory
- BLKTRACESETUP(2) /dev/sdY failed: 2/No such file or directory
- ...
- Thread Z failed open /sys/kernel/debug/block/(null)/traceZ: \
- 2/No such file or directory
- FAILED to start thread on CPU 0: 1/Operation not permitted
- ...
- Error: blktrace has errors, aborting
- ...
- ziomon: Failed to stop trace on /dev/sdX
- ...
- ziomon: Failed to stop trace on /dev/sdY
- ziomon: blktrace has leftovers, manual cleanup required!
-Problem: While we call blktrace with stopwatch option '-w $WRP_DURATION'
- to have itself terminate automatically after the (mandatory)
- sampling duration, there might be a race
- with our own timed blktrace checking loop between "Collecting
- data..." and "done"
- or any of SIGHUP SIGTERM SIGINT SIGQUIT from the user might
- prematurely trigger emergency_shutdown(),
- so blktrace might still be running or not have completed its
- cleanup when we reach the blktrace leftover checks in shutdown().
-
- In that case, ziomon uses the meanwhile hidden kill option of
- blktrace which can corrupt kernel state if blktrace user space
- has not yet completed its regular cleanup.
-Solution: The correct way to have the blktrace process terminate and
- cleanup properly including any dependent kernel state, before
- blktrace's stopwatch makes it terminate and cleanup
- automatically, is to only send SIGINT to it.
-
- http://git.kernel.org/cgit/linux/kernel/git/axboe/blktrace.git
- 86596c7579c6 ("blktrace: remove -k from manpage synopsis")
- fb7f86674a51 ("blktrace: disable kill option - take 2")
- Additional explanation for blktrace -k:
- http://marc.info/?l=linux-btrace&m=131245973528087&w=2i
- linux-btrace ("Re: Recent changes")
- The following kernel commits do not change above termination
- statement:
- http://git.kernel.org/cgit/linux/kernel/git/torvalds/linux.git
- 39cbb602b543 ("Remove double removal of blktrace directory")
- fd51d251e4cd ("blktrace: remove debugfs entries on bad path")
- f48fc4d32e24
- ("block: get rid of the manual directory counting in blktrace")
-
- As of this writing, the (multithreaded) blktrace also wires
- SIGHUP SIGTERM SIGALRM with the same signal handler
- handle_sigint(), but to adhere to its documentation move from
- SIGTERM to SIGINT anyway.
-
- We must not use the meanwhile hidden kill option of blktrace
- because this corrupts kernel state if blktrace user space has not
- yet completed its regular cleanup.
-
- In order to avoid zombies we wait for all children. After all, we
- want everything to be completed before ziomon terminates and the
- user could safely run another instance of ziomon. Also, we get
- the possibility for users to detect and report issues with
- children that don't finish timely.
-
- Since we do not know how to manually cleanup, drop the check for
- blktrace leftovers in shutdown().
-Reproduction: Run ziomon with many devices on the command line.
- Possibly (other) load on the system increases probability.
-Upstream-ID: -
-Problem-ID: 149058
-
-Signed-off-by: Steffen Maier
----
- ziomon/ziomon | 17 ++++++-----------
- 1 file changed, 6 insertions(+), 11 deletions(-)
-
---- a/ziomon/ziomon
-+++ b/ziomon/ziomon
-@@ -5,7 +5,7 @@
- #
- # Wrapper script to start all processes
- #
--# Copyright IBM Corp. 2008
-+# Copyright IBM Corp. 2008, 2016
- #
- # Author(s): Stefan Raspl
- #
-@@ -84,7 +84,7 @@ function check_for_int() {
-
- function print_version() {
- echo "$WRP_TOOLNAME: I/O data collection utility, version %S390_TOOLS_VERSION%";
-- echo "Copyright IBM Corp. 2008";
-+ echo "Copyright IBM Corp. 2008, 2016";
- }
-
-
-@@ -356,8 +356,6 @@ function start_trace() {
-
-
- function shutdown() {
-- local failed=0;
--
- echo "Shutting down";
- # one more second to write final result
- sleep 2;
-@@ -365,7 +363,7 @@ function shutdown() {
- [ -d /proc/$WRP_ZIOMON_UTIL_PID ] && echo "Shutting down utilization process" && kill -s SIGTERM $WRP_ZIOMON_UTIL_PID;
- fi
- if [ "$WRP_BLKTRACE_PID" != "" ]; then
-- [ -d /proc/$WRP_BLKTRACE_PID ] && echo "Shutting down blktrace process" && kill -s SIGTERM $WRP_BLKTRACE_PID;
-+ [ -d /proc/$WRP_BLKTRACE_PID ] && echo "Shutting down blktrace process" && kill -s SIGINT $WRP_BLKTRACE_PID;
- fi
- if [ "$WRP_BLKIOMON_PID" != "" ]; then
- [ -d /proc/$WRP_BLKIOMON_PID ] && echo "Shutting down blkiomon process" && kill -s SIGTERM $WRP_BLKIOMON_PID;
-@@ -381,12 +379,9 @@ function shutdown() {
- kill -s SIGTERM $WRP_ZIOMON_MGR_PID;
- fi
- fi
-- # kill running traces on devices
-- for dev in ${WRP_DEVICES[@]}; do
-- [ -d /sys/kernel/debug/block/${dev##*/} ] && debug "Trace on $dev exists, attempt to kill..." && blktrace -k $dev >/dev/null 2>&1;
-- [ -d /sys/kernel/debug/block/${dev##*/} ] && echo "$WRP_TOOLNAME: Failed to stop trace on $dev" && (( failed++ ));
-- done
-- [ $failed -ne 0 ] && echo "$WRP_TOOLNAME: blktrace has leftovers, manual cleanup required!";
-+ # synchronize with all children to avoid zombies
-+ # and to prepare for a clean subsequent re-run of ziomon
-+ wait
-
- if [ -e $WRP_MSG_Q_PATH ]; then
- if [ $WRP_DEBUG -gt 1 ]; then
diff --git a/s390-tools.changes b/s390-tools.changes
index e6fabeb..d53da7c 100644
--- a/s390-tools.changes
+++ b/s390-tools.changes
@@ -1,3 +1,181 @@
+-------------------------------------------------------------------
+Thu Oct 5 20:47:52 UTC 2017 - mpost@suse.com
+
+- Modified mkdump perl script to work with the updated version
+ of the IBM tools dasdview, zipl and zgetdump. (bsc#1020336).
+
+-------------------------------------------------------------------
+Wed Oct 4 19:45:29 UTC 2017 - mpost@suse.com
+
+- Upgraded to version 2.1.0 (Fate#323291).
+ Changes from 2.0.0 to 2.1.0
+ * Added the following tools:
+ netboot: Scripts for building a PXE-style netboot image for KVM
+ 90-cpi.rules/cpictl: New udev rule to update CPI when KVM is used
+ * Modified lsqeth/zdev to add VNIC Characteristics support
+ Bug fixes
+ * chzcrypt: Corrected handling of insufficient permissions
+ * cpacfstats: Add size setting to perf event
+ * fdasd: Skip partition check with the force option
+ * ttyrun: Fix deprecated BindTo usage in ttyrun-getty@.service.in
+ * lszcrypt: Fix core dump caused by stack overwrite
+ * lszcrypt: Fix random domain printout when no config available
+ * zdev: Fix segfault with unknown qeth attribute
+ * zdev: Fix IPv6 NDP proxy description
+ * zdev: Fix zdev dracut module temp file location
+ * zkey: Correctly detect abbreviated commands
+ * zkey: Validate XTS key: ignore domain and card
+ * zkey: Use octal values instead of S_IRWX* constants
+ * zkey: Properly set umask to prohibit permissions to group and others
+ * zkey: Add -ldl to LDLIBS (not LDFLAGS)
+ * znetconf: Re-add missing line in lsznet.raw
+ * Fix several gcc 7 warnings
+- Modified s390-tools-sles12-fdasd-skip-partition-check-and-BLKRRPART-ioctl.patch
+ to apply cleanly to this version.
+- Removed the following obsolete patches:
+ * s390-tools-sles15-zgetdump-Fix-gcc-7-warning.patch
+ * s390-tools-sles15-lscss-Get-rid-of-gcc-7-buffer-truncation-warnings.patch
+ * s390-tools-sles15-Get-rid-of-gcc-7-fall-through-warnings.patch
+ * s390-tools-sles15-lszcrypt-Fix-core-dump-caused-by-stack-overwrite.patch
+ * s390-tools-sles15-dasdinfo-Fix-GCC-7-overflow-warning.patch
+
+-------------------------------------------------------------------
+Thu Sep 8 21:06:37 UTC 2017 - mpost@suse.com
+
+- Added the following patches (bsc#1056498)
+ - s390-tools-sles15-zgetdump-Fix-gcc-7-warning.patch
+ - s390-tools-sles15-lscss-Get-rid-of-gcc-7-buffer-truncation-warnings.patch
+ - s390-tools-sles15-Get-rid-of-gcc-7-fall-through-warnings.patch
+ - s390-tools-sles15-lszcrypt-Fix-core-dump-caused-by-stack-overwrite.patch
+ - s390-tools-sles15-dasdinfo-Fix-GCC-7-overflow-warning.patch
+
+-------------------------------------------------------------------
+Tue Sep 8 20:01:35 UTC 2017 - mpost@suse.com
+
+- Upgraded to version 2.0.0 (Fate#323291).
+ - Changes from 1.34.0 to 1.35.0
+ - cpuplugd: Make default config file location more distribution agnostic
+ Add systemd service unit to start/stop the cpuplug daemon
+ Add force-reload action to init script for LSB-compliance
+ - mon_statd: Add force-reload action to init script for LSB-compliance
+ - chiucvallow: corrected verification return code
+ - lczdev/chzdev: Add support for resolving iSCSI block devices
+ Bug fixes
+ - chzdev, lszdev: Fix incorrect zfcp module parameter name
+ - chreipl: Fix chreipl node for virtio devices
+ - libus2s: Fix busid parsing
+ - zipl/boot: Fix failed start subchannel in FBA loader
+ - Changes from 1.35.0 to 1.36.0
+ - chzdev/lszdev: Add device resolution for vlan and bonding devices
+ - dasdfmt: Add quick format support
+ - lsdasd/fdasd/dasdfmt/zdsfs: Add query host access to volume support
+ - zfcpdbf: Make timestamp the first field in record output
+ Exclude Payload hex field from log message record output
+ Add Area field to record output
+ Add missing header fields to area record output
+ Print the Record id field for Area area class records
+ Mark oldest and newest trace area records
+ Mark oldest and newest def_err pseudo area records
+ Mark oldest and newest foreign area records
+ Mark oldest and newest log message records
+ Add a warning about low resolution timestamps
+ Add Trace Area Statistics table to script output
+ Add time zone command line option
+ Add collection of domain xml files
+ Bug fixes
+ - zfcpdbf: Move and fix CPU id field output to print_header function
+ Include microseconds in warning threshold calculation
+ Fix parsing of precise syslog timestamps
+ Fix labels for REC ERP action status and ERP step fields
+ - Changes from 1.36.0 to 1.36.1
+ - zfcpdbf: Support basic HBA record type without warning
+ Restore tracing of handle for port and LUN with HBA records
+ Print full payload for all SAN traces (req, resp, iels)
+ Print payload length everywhere applicable
+ Disambiguate printing of payload record content
+ Bug fixes
+ - fdasd: Prevent buffer overflow
+ - znetconf: Fix detection of chpids as OSX instead of OSM
+ - Changes from 1.36.1 to 1.37.0
+ - Added dump2tar: sysfs collection helper for dbginfo.sh
+ - qethconf: Remove check for OSN-device
+ - znetconf: Remove OSN-support
+ - lsdasd/tunedasd: Add channel path aware erp
+ - zcrypt: Add multi domain support for zcrypt device driver
+ - scm: Reimplement lsscm in C
+ - chp: Reimplement chchp and lschp in C
+ - dbginfo.sh: Make use of sysinfo collection helper
+ Bug fixes
+ - zfcpdbf: Only cap SAN short payload if pl_len exists
+ - chzdev: Fix bug when configuring QETH devices
+ - libu2s: Prevent buffer overflow
+ - ziomon: No blktrace kill which can corrupt kernel blktrace state
+ - Changes from 1.37.0 to 1.37.1
+ - dbginfo.sh: Add Docker debug data
+ Bug fixes
+ - lsdasd: Replace continue with return
+ - lscss: Allow to specify devices from subchannel set 3
+ - mon_fsstatd: Only use physical filesystems
+ Fix double free in error path
+ - mon_procd: Fix segmentation fault
+ - Changes from 1.37.1 to 1.38.0
+ - systemd: Add new units for cpi, dumpconf, mon_fsstatd, and mon_procd
+ - fdasd: Add new command action 'l' to list known partition types
+ - chmem: Try to online memory to zone movable
+ Starting with kernel 4.13 (commit f1dd2cd1), the default zone for
+ hotplug memory is changed from zone movable to zone normal. In order
+ to preserve the previous default behaviour, chmem will now always try
+ to set memory online to the zone movable, before trying to set it
+ online to the default zone.
+ - zfcpdbf: Print high part of "new" 64 bit SCSI LUN
+ Print payload with full FCP_RSP IU in SCSI trace records
+ - util_opt: Add command specific command line parsing
+ - util_rec: Implement util_rec_iterate() function
+ - util_rec: Record field values can be stored and processed in argz format
+ Bug fixes
+ - zipl/zgetdump: Add missing SCSI multipath dump information to man pages
+ - lsqeth: Fix attributes name: ipa -> ipa_takeover and parp -> rxip
+ Fix lsqeth output of ipa/vipa/rxip addresses of interfaces
+ - zfcpdbf: Fix payload length for zfcp_dbf_hba_bit_err
+ - chzdev: Add implicit qeth layer settings handling
+ - qethqoat: Fix OSA 6S detection
+ - Changes from 1.38.0 to 2.0.0 (also known as 1.39.0)
+ - IBM changed the license to MIT.
+ - Added zkey: Generate, re-encipher, and validate secure AES keys
+ - cpumf: Add hardware counters for z13 and z13s
+ - lscss: Reimplement lscss script in C
+ - lsqeth: Reimplement lsqeth script in C
+ Bug fixes
+ - lscpumf: Fix Perl warnings
+ - Removed the following obsolete patches:
+ s390-tools-sles12sp2-chiucvallow-verify.patch
+ s390-tools-sles12sp2-chreipl-virtio.patch
+ s390-tools-sles12sp2-chzdev-disable-root-update.patch
+ s390-tools-sles12sp2-feat-01-dasd-query-host.patch
+ s390-tools-sles12sp2-lscss-allow-to-specify-devices-from-ssid-3.patch
+ s390-tools-sles12sp2-zipl-fix-failed-start-subchannel.patch
+ s390-tools-sles12sp3-chmem-try-to-online-zone-movable.patch
+ s390-tools-sles12sp3-dasdfmt-01-Fix-behaviour-of-t-combined-with-y.patch
+ s390-tools-sles12sp3-dasdfmt-02-Fix-trailing-whitespace.patch
+ s390-tools-sles12sp3-dasdfmt-03-Apply-coding-convention.patch
+ s390-tools-sles12sp3-dasdfmt-04-Use-enhanced-DASD-information.patch
+ s390-tools-sles12sp3-dasdfmt-05-Refactor-do_format_dasd.patch
+ s390-tools-sles12sp3-dasdfmt-06-Make-the-IOCTL-BLKSSZGET-reusable.patch
+ s390-tools-sles12sp3-dasdfmt-07-Add-quick-format-support.patch
+ s390-tools-sles12sp3-dasdfmt-08-Make-progress-output-reusable-and-add-ETR.patch
+ s390-tools-sles12sp3-dasdfmt-09-Add-command-line-argument-check.patch
+ s390-tools-sles12sp3-dasdfmt-10-Add-expand-format-mode.patch
+ s390-tools-sles12sp3-dbginfo-01-libutil-Add-utility-functions.patch
+ s390-tools-sles12sp3-dbginfo-02-dump2tar-Add-sysfs-collection-helper-for-dbginfo.sh-v2.patch
+ s390-tools-sles12sp3-dbginfo-03-dbginfo.sh-Make-use-of-sysinfo-collection-helper.patch
+ s390-tools-sles12sp3-dbginfo-Collect-docker-debug-data.patch
+ s390-tools-sles12sp2-libu2s-Fix-busid-parsing.patch
+ s390-tools-sles12sp3-lsdasd-tunedasd-Add-channel-path-aware-erp.patch
+ s390-tools-sles12sp3-mon_fsstatd-fix-double-free-in-error-path-and-skip-virtual-fs.patch
+ s390-tools-sles12sp3-util_proc-fix-memory-allocation-error-messages.patch
+ s390-tools-sles12sp3-ziomon-no-blktrace-kill-which-can-corrupt-kernel-blk.patch
+- Added s390-tools-sles15-add--ldl-to-LDLIBS.patch to fix a problem with zkey/Makefile.
+
-------------------------------------------------------------------
Thu Sep 7 19:41:39 UTC 2017 - mpost@suse.com
diff --git a/s390-tools.spec b/s390-tools.spec
index 5d7cb49..bb08904 100644
--- a/s390-tools.spec
+++ b/s390-tools.spec
@@ -17,11 +17,11 @@
Name: s390-tools
-Version: 1.34.0
+Version: 2.1.0
Release: 0
-Url: https://www.ibm.com/developerworks/linux/linux390/
+Url: https://github.com/ibm-s390-tools/s390-tools
Summary: S/390 tools like zipl and dasdfmt
-License: GPL-2.0+ and BSD-2-Clause
+License: MIT
Group: System/Kernel
BuildRequires: dracut
BuildRequires: fuse-devel
@@ -46,7 +46,7 @@ Requires: util-linux
Provides: s390utils:/sbin/dasdfmt
# Don't build with pie to avoid problems with zipl
#!BuildIgnore: gcc-PIE
-Source: s390-tools-%{version}.tar.bz2
+Source: s390-tools-%{version}.tar.gz
Source1: s390-tools-rpmlintrc
Source2: zipl.conf
Source3: hsnc
@@ -62,8 +62,6 @@ Source12: mkdump.pl
Source13: sysconfig.osasnmpd
Source14: zfcp_san_disc
Source15: mkdump.8
-Source16: boot.cpi
-Source17: sysconfig.cpi
Source18: zpxe.rexx
Source19: rules.xpram
Source20: rules.hw_random
@@ -106,36 +104,10 @@ Patch1: s390-tools-sles12-zipl_boot_msg.patch
Patch2: s390-tools-sles12-sysconfig-compatible-dumpconf.patch
Patch3: s390-tools-sles12-create-filesystem-links.patch
Patch4: s390-tools-sles12-update-by_id-links-on-change-and-add-action.patch
-Patch5: s390-tools-sles12sp2-feat-01-dasd-query-host.patch
-Patch6: s390-tools-sles12-pardasdfmt.patch
-Patch7: dasdfmt-retry-BIODASDINFO-if-device-is-busy.patch
-Patch8: 59-dasd.rules-wait_for.patch
-Patch9: s390-tools-sles12-fdasd-skip-partition-check-and-BLKRRPART-ioctl.patch
-Patch10: s390-tools-sles12sp2-libu2s-Fix-busid-parsing.patch
-Patch11: s390-tools-sles12sp2-chiucvallow-verify.patch
-Patch12: s390-tools-sles12sp2-zipl-fix-failed-start-subchannel.patch
-Patch13: s390-tools-sles12sp2-chreipl-virtio.patch
-Patch14: s390-tools-sles12sp2-chzdev-disable-root-update.patch
-Patch15: s390-tools-sles12sp2-lscss-allow-to-specify-devices-from-ssid-3.patch
-Patch16: s390-tools-sles12sp3-dbginfo-01-libutil-Add-utility-functions.patch
-Patch17: s390-tools-sles12sp3-dbginfo-02-dump2tar-Add-sysfs-collection-helper-for-dbginfo.sh-v2.patch
-Patch18: s390-tools-sles12sp3-dbginfo-03-dbginfo.sh-Make-use-of-sysinfo-collection-helper.patch
-Patch19: s390-tools-sles12sp3-lsdasd-tunedasd-Add-channel-path-aware-erp.patch
-Patch20: s390-tools-sles12sp3-dasdfmt-01-Fix-behaviour-of-t-combined-with-y.patch
-Patch21: s390-tools-sles12sp3-dasdfmt-02-Fix-trailing-whitespace.patch
-Patch22: s390-tools-sles12sp3-dasdfmt-03-Apply-coding-convention.patch
-Patch23: s390-tools-sles12sp3-dasdfmt-04-Use-enhanced-DASD-information.patch
-Patch24: s390-tools-sles12sp3-dasdfmt-05-Refactor-do_format_dasd.patch
-Patch25: s390-tools-sles12sp3-dasdfmt-06-Make-the-IOCTL-BLKSSZGET-reusable.patch
-Patch26: s390-tools-sles12sp3-dasdfmt-07-Add-quick-format-support.patch
-Patch27: s390-tools-sles12sp3-dasdfmt-08-Make-progress-output-reusable-and-add-ETR.patch
-Patch28: s390-tools-sles12sp3-dasdfmt-09-Add-command-line-argument-check.patch
-Patch29: s390-tools-sles12sp3-dasdfmt-10-Add-expand-format-mode.patch
-Patch30: s390-tools-sles12sp3-util_proc-fix-memory-allocation-error-messages.patch
-Patch31: s390-tools-sles12sp3-mon_fsstatd-fix-double-free-in-error-path-and-skip-virtual-fs.patch
-Patch32: s390-tools-sles12sp3-dbginfo-Collect-docker-debug-data.patch
-Patch33: s390-tools-sles12sp3-ziomon-no-blktrace-kill-which-can-corrupt-kernel-blk.patch
-Patch34: s390-tools-sles12sp3-chmem-try-to-online-zone-movable.patch
+Patch5: s390-tools-sles12-pardasdfmt.patch
+Patch6: dasdfmt-retry-BIODASDINFO-if-device-is-busy.patch
+Patch7: 59-dasd.rules-wait_for.patch
+Patch8: s390-tools-sles12-fdasd-skip-partition-check-and-BLKRRPART-ioctl.patch
BuildRoot: %{_tmppath}/%{name}-%{version}-build
ExclusiveArch: s390 s390x
@@ -196,36 +168,10 @@ to list files and directories.
%patch2 -p1
%patch3 -p1
%patch4 -p1
-%patch5 -p1
+#patch5 -p1
%patch6 -p1
%patch7 -p1
%patch8 -p1
-%patch9 -p1
-%patch10 -p1
-%patch11 -p1
-%patch12 -p1
-%patch13 -p1
-%patch14 -p1
-%patch15 -p1
-%patch16 -p1
-%patch17 -p1
-%patch18 -p1
-%patch19 -p1
-%patch20 -p1
-%patch21 -p1
-%patch22 -p1
-%patch23 -p1
-%patch24 -p1
-%patch25 -p1
-%patch26 -p1
-%patch27 -p1
-%patch28 -p1
-%patch29 -p1
-%patch30 -p1
-%patch31 -p1
-%patch32 -p1
-%patch33 -p1
-%patch34 -p1
cp -vi %{S:22} CAUTION
@@ -253,15 +199,15 @@ install -m644 -t $RPM_BUILD_ROOT/%{_mandir}/man8 %{S:87}
export ROOT_BUILD_DIR="$RPM_BUILD_DIR/%{name}-%{version}/zfcpdump/kernel"
install -D -m600 /boot/image-*-zfcpdump $RPM_BUILD_ROOT/usr/lib/s390-tools/zfcpdump/zfcpdump_part.image
-install -D -m644 etc/sysconfig/dumpconf $RPM_BUILD_ROOT/var/adm/fillup-templates/sysconfig.dumpconf
-install -D -m755 etc/init.d/dumpconf $RPM_BUILD_ROOT/etc/init.d/dumpconf
install -D -m644 etc/udev/rules.d/40-z90crypt.rules $RPM_BUILD_ROOT/usr/lib/udev/rules.d/40-z90crypt.rules
-install -D -m644 etc/udev/rules.d/59-dasd.rules $RPM_BUILD_ROOT/usr/lib/udev/rules.d/59-dasd.rules
install -D -m644 etc/udev/rules.d/57-osasnmpd.rules $RPM_BUILD_ROOT/usr/lib/udev/rules.d/57-osasnmpd.rules
+install -D -m644 etc/udev/rules.d/59-dasd.rules $RPM_BUILD_ROOT/usr/lib/udev/rules.d/59-dasd.rules
+install -D -m644 etc/udev/rules.d/90-cpi.rules $RPM_BUILD_ROOT/usr/lib/udev/rules.d/90-cpi.rules
+install -D -m644 etc/sysconfig/cpi $RPM_BUILD_ROOT/var/adm/fillup-templates/sysconfig.cpi
+install -D -m644 etc/sysconfig/dumpconf $RPM_BUILD_ROOT/var/adm/fillup-templates/sysconfig.dumpconf
+install -D -m644 etc/sysconfig/mon_fsstatd $RPM_BUILD_ROOT/var/adm/fillup-templates/sysconfig.mon_fsstatd
+install -D -m644 etc/sysconfig/mon_procd $RPM_BUILD_ROOT/var/adm/fillup-templates/sysconfig.mon_procd
install -D -m644 etc/sysconfig/mon_statd $RPM_BUILD_ROOT/var/adm/fillup-templates/sysconfig.mon_statd
-install -D -m755 etc/init.d/mon_statd $RPM_BUILD_ROOT/etc/init.d/mon_statd
-install -D -m644 etc/sysconfig/cpuplugd $RPM_BUILD_ROOT/var/adm/fillup-templates/sysconfig.cpuplugd
-install -D -m755 etc/init.d/cpuplugd $RPM_BUILD_ROOT/etc/init.d/cpuplugd
mv iucvterm/doc/ts-shell/iucvconn_on_login $RPM_BUILD_ROOT/usr/bin/iucvconn_on_login
install -D -m644 %{S:26} $RPM_BUILD_ROOT/%{_unitdir}/cio_ignore.service
install -D -m755 %{S:27} $RPM_BUILD_ROOT/usr/lib/systemd/scripts/setup_cio_ignore.sh
@@ -287,8 +233,6 @@ install -D -m755 %{S:12} sbin/mkdump
install -D -m644 %{S:13} var/adm/fillup-templates/sysconfig.osasnmpd
install -D -m755 %{S:14} sbin/zfcp_san_disc
install -D -m644 %{S:15} $RPM_BUILD_ROOT/%{_mandir}/man8
-install -D -m755 %{S:16} etc/init.d/boot.cpi
-install -D -m644 %{S:17} var/adm/fillup-templates/sysconfig.cpi
install -D -m644 %{S:19} $RPM_BUILD_ROOT/usr/lib/udev/rules.d/52-xpram.rules
install -D -m644 %{S:20} $RPM_BUILD_ROOT/usr/lib/udev/rules.d/52-hw_random.rules
install -D -m644 %{S:21} $RPM_BUILD_ROOT/usr/lib/udev/rules.d/59-graf.rules
@@ -304,14 +248,16 @@ if [ ! -d %{_sbindir} ]; then
mkdir -p %{_sbindir}
fi
(cd usr/sbin; ln -s ../../etc/init.d/appldata rcappldata)
-(cd usr/sbin; ln -s ../../etc/init.d/cpuplugd rccpuplugd)
-(cd usr/sbin; ln -s ../../etc/init.d/dumpconf rcdumpconf)
(cd usr/sbin; ln -s ../../etc/init.d/hsnc rchsnc)
-(cd usr/sbin; ln -s ../../etc/init.d/mon_statd rcmon_statd)
(cd usr/sbin; ln -s ../../etc/init.d/vmlogrdr rcvmlogrdr)
(cd usr/sbin; ln -s ../../etc/init.d/xpram rcxpram)
(cd usr/sbin; ln -s service rccio_ignore)
(cd usr/sbin; ln -s service rccpacfstatsd)
+(cd usr/sbin; ln -s service rccpi)
+(cd usr/sbin; ln -s service rccpuplugd)
+(cd usr/sbin; ln -s service rcdumpconf)
+(cd usr/sbin; ln -s service rcmon_fsstatd)
+(cd usr/sbin; ln -s service rcmon_procd)
(cd usr/sbin; ln -s service rcvirtsetup)
if [ ! -d %{_bindir} ]; then
@@ -364,6 +310,13 @@ chmod 755 osasnmpd
getent group ts-shell >/dev/null 2>&1 || groupadd -r ts-shell
%service_add_pre cio_ignore.service
%service_add_pre cpacfstatsd.service
+%service_add_pre cpi.service
+%service_add_pre cpuplugd.service
+%service_add_pre dumpconf.service
+%service_add_pre iucvtty-login@.service
+%service_add_pre mon_fsstatd.service
+%service_add_pre mon_procd.service
+%service_add_pre ttyrun-getty@.service
%service_add_pre virtsetup.service
%post
@@ -378,17 +331,25 @@ fi
# Create symbolic links to the scripts from setup and boot directories
%service_add_post cio_ignore.service
%service_add_post cpacfstatsd.service
+%service_add_post cpi.service
+%service_add_post cpuplugd.service
+%service_add_post dumpconf.service
+%service_add_post iucvtty-login@.service
+%service_add_post mon_fsstatd.service
+%service_add_post mon_procd.service
+%service_add_post ttyrun-getty@.service
%service_add_post virtsetup.service
+
# -i: skip insserv parts - has to be configured before use:
%{fillup_and_insserv -i -n appldata}
-%{fillup_and_insserv -i -n cpuplugd}
+%{fillup_and_insserv -i -n cpi}
%{fillup_and_insserv -i -n dumpconf}
%{fillup_and_insserv -i -n hsnc}
+%{fillup_and_insserv -i -n mon_fsstatd}
+%{fillup_and_insserv -i -n mon_procd}
%{fillup_and_insserv -i -n mon_statd}
-%{fillup_and_insserv -i -n vmlogrdr}
%{fillup_and_insserv -i -n virtsetup}
%{fillup_and_insserv -i -n xpram}
-%{fillup_and_insserv -y -n cpi boot.cpi}
%triggerin -- kernel-default
grep -q '^/usr/bin/ts-shell$' /etc/shells \
@@ -399,28 +360,35 @@ grep -q '^/usr/bin/ts-shell$' /etc/shells \
%preun
%{stop_on_removal appldata}
-%{stop_on_removal cpuplugd}
-%{stop_on_removal dumpconf}
%{stop_on_removal hsnc}
-%{stop_on_removal mon_statd}
%{stop_on_removal vmlogrdr}
%{stop_on_removal xpram}
%service_del_preun cio_ignore.service
%service_del_preun cpacfstatsd.service
+%service_del_preun cpi.service
+%service_del_preun cpuplugd.service
+%service_del_preun dumpconf.service
+%service_del_preun iucvtty-login@.service
+%service_del_preun mon_fsstatd.service
+%service_del_preun mon_procd.service
%service_del_preun virtsetup.service
%postun
-%{restart_on_update boot.cpi}
%{restart_on_update appldata}
-%{restart_on_update cpuplugd}
-%{restart_on_update dumpconf}
%{restart_on_update hsnc}
-%{restart_on_update mon_statd}
%{restart_on_update vmlogrdr}
%{restart_on_update xpram}
%service_del_postun cio_ignore.service
%service_del_postun cpacfstatsd.service
+%service_del_postun cpi.service
+%service_del_postun cpuplugd.service
+%service_del_postun dumpconf.service
+%service_del_postun iucvtty-login@.service
+%service_del_postun mon_fsstatd.service
+%service_del_postun mon_procd.service
+%service_del_postun ttyrun-getty@.service
%service_del_postun virtsetup.service
+
read INITPGM < /proc/1/comm
if [ "${INITPGM}" == "systemd" ]; then
echo "Running systemctl daemon-reload."
@@ -447,7 +415,7 @@ rm -rf $RPM_BUILD_ROOT
%files -f %{_builddir}/%{name}.list
%defattr(-,root,root)
-%doc README
+%doc README.md
%doc README.SUSE
%doc iucvterm/doc/ts-shell
@@ -470,6 +438,7 @@ rm -rf $RPM_BUILD_ROOT
%dir %{_unitdir}
%dir /usr/share/s390-tools
%dir /usr/share/s390-tools/cpumf
+%dir /usr/share/s390-tools/netboot
%dir /boot/zipl
%dir /lib/s390-tools/
%exclude /usr/lib/udev/rules.d/57-osasnmpd.rules
diff --git a/sysconfig.cpi b/sysconfig.cpi
deleted file mode 100644
index 33651c8..0000000
--- a/sysconfig.cpi
+++ /dev/null
@@ -1,22 +0,0 @@
-## Path: Kernel/CPI
-## Description: set Control Program Indentification data
-## Type: list(yes,no)
-## Default: no
-#
-# Set Control Program Identification data
-#
-CPI_SET="yes"
-
-## Type: string
-## Default: ""
-#
-# CPI system name
-#
-CPI_SYSTEM_NAME=""
-
-## Type: string
-## Default: ""
-#
-# CPI sysplex name
-#
-CPI_SYSPLEX_NAME=""