From: Dave Reisner <d@falconindy.com>
To: Timofey Titovets <nefelim4ag@gmail.com>
Cc: util-linux@vger.kernel.org, Minchan Kim <minchan@kernel.org>,
Karel Zak <kzak@redhat.com>
Subject: Re: [RFC] [Patch] Created zramctl
Date: Wed, 23 Jul 2014 13:43:48 -0400 [thread overview]
Message-ID: <20140723174348.GA519@rampage> (raw)
In-Reply-To: <53CFF0CC.4000608@gmail.com>
On Wed, Jul 23, 2014 at 08:28:44PM +0300, Timofey Titovets wrote:
> Good time of day,
> I spend some time and rewrite zramctl for util-linux.
>
> Please review code and man pages, i think what i can do it better.
> If you have any suggestion, say it out.
>
> Can be pulled from:
> https://github.com/Nefelim4ag/util-linux
>
> For detail see man pages zramctl.8
>
I have a general fear of your repeated usage of strcpy-ish functions
into fixed size buffers and lack of error reporting...
I've left a few specific comments inline. I'm sure there's other things
that need pointing out.
> ----
> sys-utils/Makemodule.am | 5 +
> sys-utils/zramctl.8 | 92 ++++++++++++++
> sys-utils/zramctl.c | 319
> ++++++++++++++++++++++++++++++++++++++++++++++++
> 3 files changed, 416 insertions(+)
>
> diff --git a/sys-utils/Makemodule.am b/sys-utils/Makemodule.am
> index 68fd030..5d70c51 100644
> --- a/sys-utils/Makemodule.am
> +++ b/sys-utils/Makemodule.am
> @@ -172,6 +172,11 @@ eject_CFLAGS = $(AM_CFLAGS) -I$(ul_libmount_incdir)
> dist_man_MANS += sys-utils/eject.1
> endif
>
> +sbin_PROGRAMS += zramctl
> +dist_man_MANS += sys-utils/zramctl.8
> +zramctl_SOURCES = sys-utils/zramctl.c
> +zramctl_LDADD = $(LDADD) libcommon.la libsmartcols.la
> +zramctl_CFLAGS = $(AM_CFLAGS) -I$(ul_libsmartcols_incdir)
>
> if BUILD_LOSETUP
> sbin_PROGRAMS += losetup
> diff --git a/sys-utils/zramctl.8 b/sys-utils/zramctl.8
> new file mode 100644
> index 0000000..4da209d
> --- /dev/null
> +++ b/sys-utils/zramctl.8
> @@ -0,0 +1,92 @@
> +.TH ZRAMCTL 8 "July 2014" "util-linux" "System Administration"
> +.SH NAME
> +zramctl \- set up and control zram devices
> +.SH SYNOPSIS
> +.ad l
> +Get info:
> +.sp
> +.in +5
> +.B zramctl
> +.sp
> +.in -5
> +Reset zram:
> +.sp
> +.in +5
> +.B "zramctl \-r"
> +.IR zramdev
> +.sp
> +.in -5
> +Print name of first unused zram device:
> +.sp
> +.in +5
> +.B "zramctl \-f"
> +.sp
> +.in -5
> +Setup zram device:
> +.sp
> +.in +5
> +.B zramctl
> +.RB { \-f\ [ \-d\ \fIzramdev\fP] }
> +.RB [ \-s
> +.IR size ]
> +.RB \ [ \-t
> +.IR number ]
> +.in +8
> +.RB [ \-a
> +.IR algorithm ]
> +.sp
> +.in -13
> +.ad b
> +.SH DESCRIPTION
> +.B zramctl
> +is used to fast setup zram devices parametrs, to reset zram devices and to
> +query the status of a used zram devices.
> +If no option is given, all zram devices are shown.
> +
> +
> +.SH OPTIONS
> +
> +.IP "\fB\-s, \-\-size\fP \fIsize\fP
> +force zram driver to reread size of the file associated with the specified
> zram device
> ++The \fIsize\fR arguments may be followed by the multiplicative
> ++suffixes 128K 512M, 1G.
> +.IP "\fB\-r, \-\-reset\fP \fIzramdev\fP"
> +Reset options specified zram device(s). Zram device setting can be changed
> only
> +after reset.
> +.IP "\fB\-f, \-\-find\fP"
> +find the first unused zram device. If a
> +.R \-s \fIsize\fR
> +argument is present, use this device.
> +.IP "\fB\-h, \-\-help\fP"
> +print help
> +.IP "\fB\-t, \-\-threads \fInumber\fP"
> +Set number of maximum compress streams what used for device.
> +.IP "\fB\-a, \-\-alg \fI{lzo|lz4}\fP""
> +Set compress algorithm used for compress data in zram device.
> +
> +.SH RETURN VALUE
> +.B zramctl
> +returns 0 on success, nonzero on failure.
> +
> +.SH FILES
> +.TP
> +.I /dev/zram[0..N]
> +zram block devices
> +
> +.SH EXAMPLE
> +The following commands can be used as an example of using the zram device.
> +.nf
> +.IP
> +# zramctl --find --size 1024M
> +/dev/zram0
> +# mkswap /dev/zram0
> +# swapon /dev/zram0
> + ...
> +# swapoff /dev/zram0
> +# zramctl --reset /dev/zram0
> +.fi
> +.SH AUTHORS
> +Timofey Titovets <nefelim4ag@gmail.com>
> +.SH AVAILABILITY
> +The zramctl command is part of the util-linux package and is available from
> +ftp://ftp.kernel.org/pub/linux/utils/util-linux/.
> diff --git a/sys-utils/zramctl.c b/sys-utils/zramctl.c
> new file mode 100644
> index 0000000..6ddee9b
> --- /dev/null
> +++ b/sys-utils/zramctl.c
> @@ -0,0 +1,319 @@
> +/*
> + * zramctl - purpose of it
> + *
> + * Copyright (c) 20nn Example Commercial, Inc
> + * Written by Timofey Titovets <Nefelim4ag@gmail.com>
> + *
> + * This program is free software; you can redistribute it and/or
> + * modify it under the terms of the GNU General Public License as
> + * published by the Free Software Foundation.
> + *
> + * This program is distributed in the hope that it would be useful,
> + * but WITHOUT ANY WARRANTY; without even the implied warranty of
> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
> + * GNU General Public License for more details.
> + *
> + * You should have received a copy of the GNU General Public License along
> + * with this program; if not, write to the Free Software Foundation, Inc.,
> + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
> + */
> +
> +#include <getopt.h>
> +#include <stdio.h>
> +#include <stdlib.h>
> +#include <unistd.h>
> +#include <string.h>
> +#include <sys/types.h>
> +#include <sys/stat.h>
> +
> +#include "c.h"
> +#include "closestream.h"
> +#include "nls.h"
> +
> +static inline int zram_exist(char *name)
> +{
> + struct stat info;
> + char path[16] = "/dev/";
> + strncat(path, name, 8);
> + if(stat(path, &info) != 0 )
> + return 0;
> + if (info.st_mode&S_IFBLK)
> + return 1;
> + return 0;
> +}
> +
> +static inline int read_file(char *path, char *data)
> +{
> + FILE *file;
> + unsigned i = 0;
> + char ch = EOF;
> + char *p;
> + p = path; // hack for work fopen
I don't understand this comment, or code, at all. Your comment should
explain as much.
> + file = fopen(p, "r");
> + if (file == NULL) {
> + fprintf(stderr,_("can't open !%s!\n"), path);
> + return -1;
> + }
> +
> + while (1) {
> + ch = fgetc(file);
> + if (ch == EOF || ch == '\n')
> + break;
> + else
> + data[i]=ch;
> + i++;
> + }
> + fclose(file);
> +
> + return 0;
> +}
> +
> +static inline int used(char *name)
> +{
> + char path_to_disksize[32] = "/sys/block/";
> + char disksize[32]="";
> + strncat(path_to_disksize, name, 8);
> + strncat(path_to_disksize, "/disksize", 9);
> + if(read_file(path_to_disksize, disksize) < 0)
> + return -1;
This seems to be poorly reimplementing things in include/sysfs.h.
> +
> + if (disksize[0] == '0')
> + return 0;
> +
> + return 1;
> +}
> +
> +static inline int get_value(char *name, char *data, char *filename)
> +{
> + char path[48] = "/sys/block/";
> + strncat(path, name, 8);
> + strncat(path,"/", 2);
> + strncat(path, filename, 32);
> + return read_file(path, data);
Same as above...
> +}
> +
> +static inline int write_file(char *path, char *data)
> +{
> + FILE *file;
> + char *p = path;
> + file = fopen(p, "w");
> + if (file == NULL) {
> + fprintf(stderr,_("can't write to %s\n"), path);
> + return -1;
> + }
> + fwrite(data, strlen(data), 1, file);
> + if (fclose(file) != 0)
> + return -1;
> + return 1;
If this fails, it does so silently. There's no explanation of what your
synthesized return values mean.
> +}
> +
> +static inline int value2devparm(char *name, char *data, char *filename)
> +{
> + char path[48] = "/sys/block/";
> + strncat(path, name, 8);
> + strncat(path,"/", 2);
> + strncat(path, filename, 32);
> + return write_file(path, data);
> +}
Same comment about sysfs.h
> +
> +static inline int status(void)
> +{
> + char disksize[32]="";
> + char orig_data_size[32]="";
> + char compr_data_size[32]="";
> + char comp_algorithm[6]="";
> + char max_comp_streams[12]="";
> + unsigned i;
> + char *table = "%6s %12s %10s %10s %4s %4s \n";
> + printf(table, "NAME","DISKSIZE","ORIG","COMPRES","ALG","THR");
Not really extensible. You really ought to adopt the style used in utils
like lsblk or findmnt to make this dynamic.
> + for (i=0;;i++) {
> + char name[8] = "zram";
> + char num[4];
> + char *pos;
> + sprintf(num,"%i",i);
> + strncat(name, num, 8);
> + if(!zram_exist(name))
> + break;
> + if(!used(name))
> + continue;
> + get_value(name, disksize, "disksize");
> + get_value(name, orig_data_size, "orig_data_size");
> + get_value(name, compr_data_size, "compr_data_size");
> + get_value(name, comp_algorithm, "comp_algorithm");
> + pos = strstr(comp_algorithm, "[lzo]");
> + if (pos == NULL) {
> + pos = strstr(comp_algorithm, "[lz4]");
> + if (pos == NULL)
> + strncpy(comp_algorithm,"-", 2);
> + else
> + strncpy(comp_algorithm, "lz4", 4);
> + } else
> + strncpy(comp_algorithm, "lzo", 4);
> + get_value(name, max_comp_streams, "max_comp_streams");
> + printf(table,
> + name,
> + disksize,
> + orig_data_size,
> + compr_data_size,
> + comp_algorithm,
> + max_comp_streams
> + );
> + }
> +
> + return 0;
> + goto fail;
> +fail:
> + printf("Zram module not loaded");
> + return -1;
> +}
> +
> +static inline char *find(void)
> +{
> + char *ret = NULL;
> + for (unsigned i=0;;i++) {
> + char name[8] = "zram";
> + char num[4];
> +
> + sprintf(num,"%i",i);
> + strncat(name, num, 4);
> + if (!zram_exist(name))
> + break;
> + else //if enough one dir founded -> module loaded
> + ret = "USED";
> +
> + if (used(name) == 0) {
> + ret = name;
> + break;
> + }
> + }
> +
> + return ret;
> +}
> +
> +
> +
> +static inline void usage(FILE *out)
> +{
> + fputs(USAGE_HEADER, out);
> + fprintf(out, _(" %s [-d zramX|-f] -s 64M -a lz4 -t 16\n"), "zramctl");
> + fputs(USAGE_OPTIONS, out);
> + fputs(_(" <no args> return status of used devices\n"), out);
> + fputs(_(" -f, --find find free device\n"), out);
> + fputs(_(" -d, --device [name] specify device: zramX\n"), out);
> + fputs(_(" -r, --reset [name] reset specified device\n"), out);
> + fputs(_(" -s, --size [size] device size: 131072, 1024M...\n"), out);
> + fputs(_(" -a, --alg [lzo|lz4] compress algorithm\n"), out);
> + fputs(_(" -t, --threads [0<num] number of compress streams\n"), out);
> + fputs(USAGE_SEPARATOR, out);
> + fputs(USAGE_HELP, out);
> + fputs(USAGE_VERSION, out);
> + fprintf(out, USAGE_MAN_TAIL("zramctl(8)"));
> + exit(out == stderr ? EXIT_FAILURE : EXIT_SUCCESS);
> +}
> +
> +int main(int argc, char **argv)
> +{
> + int c = 0;
> + char *dev = NULL;
> + char *size = NULL;
> + char *alg = NULL;
These variable names are pretty nondescript. alg? what is that? a search
algorithm? red algae? size? size of what?
> + char threads[5] = "1";
> + unsigned f = 0;
> +
> + static const struct option longopts[] = {
> + {"find", no_argument, NULL, 'f'},
> + {"device", required_argument, NULL, 'd'},
> + {"size", required_argument, NULL, 's'},
> + {"alg", required_argument, NULL, 'a'},
> + {"threads", required_argument, NULL, 't'},
> + {"reset", required_argument, NULL, 'r'},
> + {"version", no_argument, NULL, 'V'},
> + {"help", no_argument, NULL, 'h'},
> + {NULL, 0, NULL, 0}
> + };
> +
> + setlocale(LC_ALL, "");
> + bindtextdomain(PACKAGE, LOCALEDIR);
> + textdomain(PACKAGE);
> + atexit(close_stdout);
> +
> + while ((c = getopt_long(argc, argv, "fd:s:a:t:r:Vh", longopts, NULL)) !=
> -1) {
> + switch (c) {
> + case 'f':
> + if (dev != NULL) {
> + fprintf(stderr,_("-f || -d <name>\n"));
> + return 1;
> + }
> + f = 1;
> + dev = find();
find *what*? The name of this function should be descriptive enough to
tell you that.
> + break;
> + case 'd':
> + if (dev != NULL) {
> + fprintf(stderr,_("-f || -d <name>\n"));
> + return 1;
> + }
> + dev = optarg;
> + break;
> + case 's':
> + size = optarg;
> + break;
> + case 'a':
> + if (!strcmp(optarg,"lzo") || !strcmp(optarg,"lz4"))
> + alg = optarg;
> + else {
> + fprintf(stderr,_("%s != lzo|lz4\n"),alg );
How would this string be localized? LZO is LZO is LZO, regardless of
what language you're speaking.
> + return 1;
> + }
> + break;
> + case 't':
> + strncpy(threads, optarg, 5);
Why do you need to copy this? More importantly, if someone passes
'100000', you'll be left with { '1', '0', '0', '0', '0' } in your array
(no zero byte termination!)
> + if (atoi(threads) < 1) {
Poor UX here -- you're erroring out and dumping a wall of text without
explaining why.
> + usage(stderr);
> + return 1;
> + }
> + break;
> + case 'r':
> + dev = optarg;
> + if (value2devparm(dev, "1", "reset") < 0)
> + return 1;
> + break;
> + case 'V':
> + printf(UTIL_LINUX_VERSION);
> + return EXIT_SUCCESS;
> + case 'h':
> + usage(stdout);
> + default:
> + usage(stderr);
> + }
> + }
> + if (argc == 1) {
> + status();
> + return EXIT_SUCCESS;
> + }
> +
> + if (!strcmp(dev, "USED")) {
> + fprintf(stderr,_("all device already used\n") );
> + return 1;
> + }
> +
> + if (dev != NULL && size != NULL) {
> + if (value2devparm(dev, "1", "reset") < 0)
> + return 1;
> + if (atoi(threads) > 1)
> +
> + if (value2devparm(dev, threads, "max_comp_streams") < 0)
> + return 1;
> + if (alg != NULL)
> + if (value2devparm(dev, alg, "comp_algorithm") < 0)
> + return 1;
> +
> + if (value2devparm(dev, size, "disksize") < 0)
> + return 1;
> + } else {
> + }
> +
> + if (dev != NULL && f > 0)
> + fprintf(stdout,_("%s\n"), dev);
> +
> + return EXIT_SUCCESS;
> +}
> \ No newline at end of file
> --
> To unsubscribe from this list: send the line "unsubscribe util-linux" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at http://vger.kernel.org/majordomo-info.html
next prev parent reply other threads:[~2014-07-23 17:43 UTC|newest]
Thread overview: 16+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-07-19 17:28 [RFC] Zram handle util Timofey Titovets
2014-07-19 19:12 ` Sami Kerola
[not found] ` <CAGqmi765JkidXDpi6bP2qUk5U7Xry5nF7y0WY4Y6M_Fq8Eiqeg@mail.gmail.com>
2014-07-19 22:44 ` Fwd: " Timofey Titovets
2014-07-20 9:38 ` Timofey Titovets
2014-07-20 20:36 ` Davidlohr Bueso
2014-07-21 7:57 ` Minchan Kim
2014-07-21 8:10 ` Karel Zak
2014-07-23 17:28 ` [RFC] [Patch] Created zramctl Timofey Titovets
2014-07-23 17:43 ` Dave Reisner [this message]
2014-07-23 17:51 ` Timofey Titovets
2014-07-24 7:29 ` Karel Zak
2014-07-24 7:26 ` Karel Zak
2014-07-24 10:23 ` Benno Schulenberg
2014-07-27 19:04 ` [RFC] [Patch v2] " Timofey Titovets
2014-08-01 10:37 ` Karel Zak
2014-08-02 14:22 ` Timofey Titovets
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20140723174348.GA519@rampage \
--to=d@falconindy.com \
--cc=kzak@redhat.com \
--cc=minchan@kernel.org \
--cc=nefelim4ag@gmail.com \
--cc=util-linux@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox