File s390-tools-sles15sp4-02-libutil-Move-some-zdev-file-reading-functions-into-u.patch of Package s390-tools.29120
Subject: [PATCH] [FEAT VS1822] libutil: Move some zdev file reading functions into util
From: Matthew Rosato <mjrosato@linux.ibm.com>
Summary: ap_tools: add ap-check and the ap device type to zdev
Description: This feature adds multiple components in support of persistent
configuration of vfio-ap mediated devices.
The ap-check utility is added as a callout function for the
mdevctl utility. This allows for meaningful error messages to be
presented to end-users when vfio-ap configuration errors are
detected while using mdevctl to create/modify vfio-ap mediated
devices.
Additionally, the 'ap' device type is added to zdev, providing a
command-line interface for managing the apmask and aqmask, which
determine what AP resources are available for vfio-ap usage.
'chzdev' is updated to allow for modifying the active masks as
well as to specify persistent mask values via a udev rule.
'lszdev' is updated to allow for querying of the active and
persistent mask values.
Upstream-ID: f64b4087dd1c9253c5eafae7736d96b5edaf18f0
Problem-ID: VS1822
Upstream-Description:
libutil: Move some zdev file reading functions into util
In preparation for sharing some zdev udev code with other libraries, move
some file operation code from zdev into util_file.
Reviewed-by: Jan Hoeppner <hoeppner@linux.ibm.com>
Reviewed-by: Tony Krowiak <akrowiak@linux.ibm.com>
Reviewed-by: Peter Oberparleiter <oberpar@linux.ibm.com>
Signed-off-by: Matthew Rosato <mjrosato@linux.ibm.com>
Signed-off-by: Jan Hoeppner <hoeppner@linux.ibm.com>
Signed-off-by: Matthew Rosato <mjrosato@linux.ibm.com>
---
include/lib/util_file.h | 8 ++
libutil/util_file.c | 132 ++++++++++++++++++++++++++++++++++++++++++++++++
zdev/include/misc.h | 1
zdev/src/firmware.c | 8 ++
zdev/src/misc.c | 75 ++-------------------------
5 files changed, 152 insertions(+), 72 deletions(-)
--- a/include/lib/util_file.h
+++ b/include/lib/util_file.h
@@ -11,6 +11,8 @@
#ifndef LIB_UTIL_FILE_H
#define LIB_UTIL_FILE_H
+#include "lib/util_exit_code.h"
+
int util_file_read_line(char *str, size_t size, const char *fmt, ...);
int util_file_read_i(int *val, int base, const char *fmt, ...);
int util_file_read_l(long *val, int base, const char *fmt, ...);
@@ -26,4 +28,10 @@ int util_file_write_ul(unsigned long val
int util_file_write_ull(unsigned long long val, int base, const char *fmt, ...);
int util_file_read_va(const char *path, const char *fmt, ...);
+
+util_exit_code_t util_file_read_fd_buf(FILE *fd, void **buffer_ptr,
+ size_t *size_ptr);
+char *util_file_read_fd(FILE *fd, int chomp);
+char *util_file_read_text_file(const char *path, int chomp);
+
#endif /** LIB_UTIL_FILE_H @} */
--- a/libutil/util_file.c
+++ b/libutil/util_file.c
@@ -9,6 +9,7 @@
* it under the terms of the MIT license. See LICENSE for details.
*/
+#include <ctype.h>
#include <errno.h>
#include <limits.h>
#include <linux/limits.h>
@@ -21,11 +22,16 @@
#include <unistd.h>
#include "lib/util_base.h"
+#include "lib/util_exit_code.h"
#include "lib/util_file.h"
#include "lib/util_libc.h"
#include "lib/util_panic.h"
#include "lib/util_prg.h"
+#define READ_CHUNK_SIZE 4096
+
+extern const char *toolname;
+
/*
* Read the first line of a file into given buffer
*/
@@ -536,3 +542,129 @@ int util_file_read_va(const char *path,
return -1;
return ret;
}
+
+/**
+ * Print an error message indicating an out-of-memory situation and exit.
+ */
+static void oom(void)
+{
+ fprintf(stderr, "Out of memory\n");
+
+ /* We can't rely on our clean-up routines to work reliably during an
+ * OOM situation, so just exit here.
+ */
+ exit(UTIL_EXIT_OUT_OF_MEMORY);
+}
+
+/**
+ * Read all data from @fd and return address of resulting buffer in
+ * @buffer_ptr. If @size_ptr is non-zero, use it to store the size of the
+ * resulting buffer. Return %UTIL_EXIT_OK on success.
+ *
+ * @param[in] fd File descriptor to read data from
+ * @param[in, out] buffer_ptr Buffer to read data into
+ * @param[in, out] size_ptr Buffer to save size of data read into buffer_ptr
+ *
+ * @retval 0 read was successful
+ * @retval UTIL_EXIT_RUNTIME_ERROR error while reading file
+ */
+util_exit_code_t util_file_read_fd_buf(FILE *fd, void **buffer_ptr,
+ size_t *size_ptr)
+{
+ char *buffer = NULL;
+ size_t done = 0;
+
+ while (!feof(fd)) {
+ buffer = realloc(buffer, done + READ_CHUNK_SIZE);
+ if (!buffer)
+ oom();
+ done += fread(&buffer[done], 1, READ_CHUNK_SIZE, fd);
+ if (ferror(fd)) {
+ free(buffer);
+ return UTIL_EXIT_RUNTIME_ERROR;
+ }
+ }
+
+ buffer = realloc(buffer, done);
+ if (!buffer && done > 0)
+ oom();
+
+ *buffer_ptr = buffer;
+ if (size_ptr)
+ *size_ptr = done;
+
+ return UTIL_EXIT_OK;
+}
+
+/**
+ * Read text from @fd and return resulting NULL-terminated text buffer.
+ * If @chomp is non-zero, remove trailing newline character. Return %NULL
+ * on error or when unprintable characters are read.
+ *
+ * @param[in] fd File descriptor to read data from
+ * @param[in] chomp Flag to indicate trailing newlines should be removed
+ *
+ * @retval NULL Error reading from fd
+ * @retval != 0 Data read successfully, buffer returned
+ */
+char *util_file_read_fd(FILE *fd, int chomp)
+{
+ char *buffer;
+ size_t done, i;
+
+ if (util_file_read_fd_buf(fd, (void **) &buffer, &done))
+ return NULL;
+
+ /* Check if this is a text file at all (required to filter out
+ * binary sysfs attributes).
+ */
+ for (i = 0; i < done; i++) {
+ if (!isgraph(buffer[i]) && !isspace(buffer[i])) {
+ free(buffer);
+ return NULL;
+ }
+ }
+
+ /* Remove trailing new-line character if requested. */
+ if (chomp && done > 0 && buffer[done - 1] == '\n')
+ done--;
+
+ /* NULL-terminate. */
+ buffer = realloc(buffer, done + 1);
+ if (!buffer)
+ oom();
+ buffer[done] = 0;
+
+ return buffer;
+}
+
+/**
+ * Read file as text and return NULL-terminated contents. Remove trailing
+ * newline if CHOMP is specified.
+ *
+ * @param[in] path Path to the file that will be read from
+ * @param[in] chomp Flag to indicate trailing newlines should be removed
+ *
+ * @retval NULL Error reading from file
+ * @retval != 0 File read successfully, contents returned in buffer
+ */
+char *util_file_read_text_file(const char *path, int chomp)
+{
+ char *buffer = NULL;
+ FILE *fd;
+
+ fd = fopen(path, "r");
+ if (!fd)
+ goto out;
+
+ buffer = util_file_read_fd(fd, chomp);
+ fclose(fd);
+
+out:
+ if (!buffer) {
+ fprintf(stderr, "Could not read file %s: %s\n", path,
+ strerror(errno));
+ }
+
+ return buffer;
+}
--- a/zdev/include/misc.h
+++ b/zdev/include/misc.h
@@ -173,7 +173,6 @@ bool misc_read_dir(const char *, struct
bool (*)(const char *, void *), void *);
bool file_is_devnode(const char *);
exit_code_t remove_file(const char *);
-exit_code_t misc_read_fd(FILE *fd, void **buffer, size_t *size_ptr);
char *misc_read_text_file(const char *, int, err_t);
char *misc_read_cmd_output(const char *, int, err_t);
char *config_read_cmd_output(const char *, int, err_t);
--- a/zdev/src/firmware.c
+++ b/zdev/src/firmware.c
@@ -19,6 +19,8 @@
#include <sys/types.h>
#include <unistd.h>
+#include "lib/util_file.h"
+
#include "attrib.h"
#include "ccw.h"
#include "ccwgroup.h"
@@ -187,7 +189,8 @@ static exit_code_t read_fw(struct fw_fil
for (retry = 0; retry < READ_RETRY; retry++) {
/* Read complete file once */
- rc = misc_read_fd(fd, (void **) &buffer, &size);
+ rc = (exit_code_t) util_file_read_fd_buf(fd, (void **) &buffer,
+ &size);
if (rc) {
warn("%s: Could not read file", filename);
return rc;
@@ -203,7 +206,8 @@ static exit_code_t read_fw(struct fw_fil
/* Could be a pipe, socket, or FIFO - accept v1. */
break;
}
- rc = misc_read_fd(fd, (void **) &buffer2, &size2);
+ rc = (exit_code_t) util_file_read_fd_buf(fd, (void **) &buffer2,
+ &size2);
if (rc || !buffer2) {
/* Could not get second version - accept v1. */
break;
--- a/zdev/src/misc.c
+++ b/zdev/src/misc.c
@@ -26,6 +26,8 @@
#include <time.h>
#include <unistd.h>
+#include "lib/util_file.h"
+
#include "dasd.h"
#include "device.h"
#include "devtype.h"
@@ -97,71 +99,6 @@ static void dryrun_end_data(void)
fprintf(fd, "%c", DRYRUN_DATA_END);
}
-#define READ_CHUNK_SIZE 4096
-
-/* Read all data from @fd and return address of resulting buffer in
- * @buffer_ptr. If @size_ptr is non-zero, use it to store the size of the
- * resulting buffer. Return %EXIT_OK on success. */
-exit_code_t misc_read_fd(FILE *fd, void **buffer_ptr, size_t *size_ptr)
-{
- char *buffer = NULL;
- size_t done = 0;
-
- while (!feof(fd)) {
- buffer = realloc(buffer, done + READ_CHUNK_SIZE);
- if (!buffer)
- oom();
- done += fread(&buffer[done], 1, READ_CHUNK_SIZE, fd);
- if (ferror(fd)) {
- free(buffer);
- return EXIT_RUNTIME_ERROR;
- }
- }
-
- buffer = realloc(buffer, done);
- if (!buffer && done > 0)
- oom();
-
- *buffer_ptr = buffer;
- if (size_ptr)
- *size_ptr = done;
-
- return EXIT_OK;
-}
-
-/* Read text from @fd and return resulting NULL-terminated text buffer.
- * If @chomp is non-zero, remove trailing newline character. Return %NULL
- * on error or when unprintable characters are read. */
-static char *read_fd(FILE *fd, int chomp)
-{
- char *buffer;
- size_t done, i;
-
- if (misc_read_fd(fd, (void **) &buffer, &done))
- return NULL;
-
- /* Check if this is a text file at all (required to filter out
- * binary sysfs attributes). */
- for (i = 0; i < done; i++) {
- if (!isgraph(buffer[i]) && !isspace(buffer[i])) {
- free(buffer);
- return NULL;
- }
- }
-
- /* Remove trailing new-line character if requested. */
- if (chomp && done > 0 && buffer[done - 1] == '\n')
- done--;
-
- /* NULL-terminate. */
- buffer = realloc(buffer, done + 1);
- if (!buffer)
- oom();
- buffer[done] = 0;
-
- return buffer;
-}
-
static int count_newline(const char *str)
{
int i, newline;
@@ -182,7 +119,7 @@ static void dryrun_print(void)
int cmd, newline;
fseek(dryrun_file, 0, SEEK_SET);
- txt = read_fd(dryrun_file, 0);
+ txt = util_file_read_fd(dryrun_file, 0);
if (!txt)
return;
@@ -979,7 +916,7 @@ char *misc_read_text_file(const char *pa
if (!fd)
goto out;
- buffer = read_fd(fd, chomp);
+ buffer = util_file_read_fd(fd, chomp);
misc_fclose(fd);
out:
@@ -1003,7 +940,7 @@ char *misc_read_cmd_output(const char *c
if (!fd)
goto out;
- buffer = read_fd(fd, chomp);
+ buffer = util_file_read_fd(fd, chomp);
pclose(fd);
out:
@@ -1027,7 +964,7 @@ char *config_read_cmd_output(const char
if (!fd)
goto out;
- buffer = read_fd(fd, chomp);
+ buffer = util_file_read_fd(fd, chomp);
misc_pclose(fd);
out: