diff options
author | Lorry Tar Creator <lorry-tar-importer@lorry> | 2015-12-28 21:37:38 +0000 |
---|---|---|
committer | Lorry Tar Creator <lorry-tar-importer@lorry> | 2015-12-28 21:37:38 +0000 |
commit | d7fdaf4d3a9795a294e93cad6f7d8238ba3754a6 (patch) | |
tree | 9bff05ea75330b7d8e2ade183cdcd1e4282f27d9 /lib/listfile.c | |
download | findutils-tarball-master.tar.gz |
findutils-4.6.0HEADfindutils-4.6.0master
Diffstat (limited to 'lib/listfile.c')
-rw-r--r-- | lib/listfile.c | 529 |
1 files changed, 529 insertions, 0 deletions
diff --git a/lib/listfile.c b/lib/listfile.c new file mode 100644 index 0000000..01a4ed7 --- /dev/null +++ b/lib/listfile.c @@ -0,0 +1,529 @@ +/* listfile.c -- display a long listing of a file + Copyright (C) 1991, 1993, 2000, 2004, 2005, 2007, 2008, 2010, 2011 + Free Software Foundation, Inc. + + 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, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will 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, see <http://www.gnu.org/licenses/>. +*/ +/* config.h must be included first. */ +#include <config.h> + +/* system headers. */ +#include <alloca.h> +#include <errno.h> +#include <fcntl.h> +#include <grp.h> +#include <locale.h> +#include <pwd.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <sys/stat.h> +#include <sys/types.h> +#include <time.h> +#include <unistd.h> /* for readlink() */ + +/* gnulib headers. */ +#include "areadlink.h" +#include "error.h" +#include "filemode.h" +#include "human.h" +#include "mbswidth.h" +#include "idcache.h" +#include "pathmax.h" +#include "stat-size.h" +#include "gettext.h" + +/* find headers. */ +#include "listfile.h" + +/* Since major is a function on SVR4, we can't use `ifndef major'. */ +#ifdef MAJOR_IN_MKDEV +#include <sys/mkdev.h> +#define HAVE_MAJOR +#endif +#ifdef MAJOR_IN_SYSMACROS +#include <sys/sysmacros.h> +#define HAVE_MAJOR +#endif + +#ifdef major /* Might be defined in sys/types.h. */ +#define HAVE_MAJOR +#endif +#ifndef HAVE_MAJOR +#define major(dev) (((dev) >> 8) & 0xff) +#define minor(dev) ((dev) & 0xff) +#endif +#undef HAVE_MAJOR + +#if ENABLE_NLS +# include <libintl.h> +# define _(Text) gettext (Text) +#else +# define _(Text) Text +#endif +#ifdef gettext_noop +# define N_(String) gettext_noop (String) +#else +/* See locate.c for explanation as to why not use (String) */ +# define N_(String) String +#endif + +static bool print_name (register const char *p, FILE *stream, int literal_control_chars); + +/* We have some minimum field sizes, though we try to widen these fields on systems + * where we discover examples where the field width we started with is not enough. */ +static int inode_number_width = 9; +static int block_size_width = 6; +static int nlink_width = 3; +static int owner_width = 8; +static int group_width = 8; +/* We don't print st_author even if the system has it. */ +static int major_device_number_width = 3; +static int minor_device_number_width = 3; +static int file_size_width = 8; + +static bool print_num(FILE *stream, unsigned long num, int *width) +{ + const int chars_out = fprintf (stream, "%*lu", *width, num); + if (chars_out >= 0) + { + if (*width < chars_out) + *width = chars_out; + return true; + } + return false; +} + + +/* NAME is the name to print. + RELNAME is the path to access it from the current directory. + STATP is the results of stat or lstat on it. + Use CURRENT_TIME to decide whether to print yyyy or hh:mm. + Use OUTPUT_BLOCK_SIZE to determine how to print file block counts + and sizes. + STREAM is the stdio stream to print on. */ + +void +list_file (const char *name, + int dir_fd, + char *relname, + const struct stat *statp, + time_t current_time, + int output_block_size, + int literal_control_chars, + FILE *stream) +{ + char modebuf[12]; + struct tm const *when_local; + char const *user_name; + char const *group_name; + char hbuf[LONGEST_HUMAN_READABLE + 1]; + bool output_good = true; + int chars_out; + int failed_at = 000; + int inode_field_width; + +#if HAVE_ST_DM_MODE + /* Cray DMF: look at the file's migrated, not real, status */ + strmode (statp->st_dm_mode, modebuf); +#else + strmode (statp->st_mode, modebuf); +#endif + + chars_out = fprintf (stream, "%*s", inode_number_width, + human_readable ((uintmax_t) statp->st_ino, hbuf, + human_ceiling, + 1u, 1u)); + if (chars_out < 0) + { + output_good = false; + failed_at = 100; + } + else if (chars_out > inode_number_width) + { + inode_number_width = chars_out; + } + if (output_good) + { + if (EOF == putc(' ', stream)) + { + output_good = false; + failed_at = 150; + } + chars_out = fprintf (stream, "%*s", + block_size_width, + human_readable ((uintmax_t) ST_NBLOCKS (*statp), hbuf, + human_ceiling, + ST_NBLOCKSIZE, output_block_size)); + if (chars_out < 0) + { + output_good = false; + failed_at = 200; + } + else + { + if (chars_out > block_size_width) + block_size_width = chars_out; + } + } + + if (output_good) + { + if (EOF == putc(' ', stream)) + { + output_good = false; + failed_at = 250; + } + /* modebuf includes the space between the mode and the number of links, + as the POSIX "optional alternate access method flag". */ + if (fprintf (stream, "%s%3lu ", modebuf, (unsigned long) statp->st_nlink) < 0) + { + output_good = false; + failed_at = 300; + } + } + + if (output_good) + { + if (EOF == putc(' ', stream)) + { + output_good = false; + failed_at = 250; + } + user_name = getuser (statp->st_uid); + if (user_name) + { + int len = mbswidth (user_name, 0); + if (len > owner_width) + owner_width = len; + output_good = (fprintf (stream, "%-*s ", owner_width, user_name) >= 0); + if (!output_good) + failed_at = 400; + } + else + { + chars_out = fprintf (stream, "%-8lu ", (unsigned long) statp->st_uid); + if (chars_out > owner_width) + owner_width = chars_out; + output_good = (chars_out > 0); + if (!output_good) + failed_at = 450; + } + } + + if (output_good) + { + group_name = getgroup (statp->st_gid); + if (group_name) + { + int len = mbswidth (group_name, 0); + if (len > group_width) + group_width = len; + output_good = (fprintf (stream, "%-*s ", group_width, group_name) >= 0); + if (!output_good) + failed_at = 500; + } + else + { + chars_out = fprintf (stream, "%-*lu", + group_width, (unsigned long) statp->st_gid); + if (chars_out > group_width) + group_width = chars_out; + output_good = (chars_out >= 0); + if (output_good) + { + if (EOF == putc(' ', stream)) + { + output_good = false; + failed_at = 525; + } + } + else + { + if (!output_good) + failed_at = 550; + } + } + } + + if (output_good) + { + if (S_ISCHR (statp->st_mode) || S_ISBLK (statp->st_mode)) + { +#ifdef HAVE_STRUCT_STAT_ST_RDEV + if (!print_num (stream, + (unsigned long) major (statp->st_rdev), + &major_device_number_width)) + { + output_good = false; + failed_at = 600; + } + if (output_good) + { + if (fprintf (stream, ", ") < 0) + { + output_good = false; + failed_at = 625; + } + } + if (output_good) + { + if (!print_num (stream, + (unsigned long) minor (statp->st_rdev), + &minor_device_number_width)) + { + output_good = false; + failed_at = 650; + } + } +#else + if (fprintf (stream, "%*s %*s", + major_device_number_width, + minor_device_number_width) < 0) + { + output_good = false; + failed_at = 700; + } +#endif + } + else + { + const int blocksize = output_block_size < 0 ? output_block_size : 1; + chars_out = fprintf (stream, "%*s", + file_size_width, + human_readable ((uintmax_t) statp->st_size, hbuf, + human_ceiling, + 1, blocksize)); + if (chars_out < 0) + { + output_good = false; + failed_at = 800; + } + else + { + if (chars_out > file_size_width) + { + file_size_width = chars_out; + } + } + } + } + + if (output_good) + { + if (EOF == putc(' ', stream)) + { + output_good = false; + failed_at = 850; + } + } + + if (output_good) + { + if ((when_local = localtime (&statp->st_mtime))) + { + char init_bigbuf[256]; + char *buf = init_bigbuf; + size_t bufsize = sizeof init_bigbuf; + + /* Use strftime rather than ctime, because the former can produce + locale-dependent names for the month (%b). + + Output the year if the file is fairly old or in the future. + POSIX says the cutoff is 6 months old; + approximate this by 6*30 days. + Allow a 1 hour slop factor for what is considered "the future", + to allow for NFS server/client clock disagreement. */ + char const *fmt = + ((current_time - 6 * 30 * 24 * 60 * 60 <= statp->st_mtime + && statp->st_mtime <= current_time + 60 * 60) + ? "%b %e %H:%M" + : "%b %e %Y"); + + while (!strftime (buf, bufsize, fmt, when_local)) + buf = alloca (bufsize *= 2); + + if (fprintf (stream, "%s ", buf) < 0) + { + output_good = false; + failed_at = 900; + } + } + else + { + /* The time cannot be represented as a local time; + print it as a huge integer number of seconds. */ + int width = 12; + + if (statp->st_mtime < 0) + { + char const *num = human_readable (- (uintmax_t) statp->st_mtime, + hbuf, human_ceiling, 1, 1); + int sign_width = width - strlen (num); + if (fprintf (stream, "%*s%s ", + sign_width < 0 ? 0 : sign_width, "-", num) < 0) + { + output_good = false; + failed_at = 1000; + } + } + else + { + if (fprintf (stream, "%*s ", width, + human_readable ((uintmax_t) statp->st_mtime, hbuf, + human_ceiling, + 1, 1)) < 0) + { + output_good = false; + failed_at = 1100; + } + } + } + } + + if (output_good) + { + output_good = print_name (name, stream, literal_control_chars); + if (!output_good) + { + failed_at = 1200; + } + } + + if (output_good) + { + if (S_ISLNK (statp->st_mode)) + { + char *linkname = areadlinkat (dir_fd, relname); + if (linkname) + { + if (fputs (" -> ", stream) < 0) + { + output_good = false; + failed_at = 1300; + } + if (output_good) + { + output_good = print_name (linkname, stream, literal_control_chars); + if (!output_good) + { + failed_at = 1350; + } + } + } + else + { + /* POSIX requires in the case of find that if we issue a + * diagnostic we should have a nonzero status. However, + * this function doesn't have a way of telling the caller to + * do that. However, since this function is only used when + * processing "-ls", we're already using an extension. + */ + error (0, errno, "%s", name); + } + free (linkname); + } + if (output_good) + { + if (EOF == putc ('\n', stream)) + { + output_good = false; + if (!output_good) + { + failed_at = 1400; + } + } + } + } + if (!output_good) + { + error (EXIT_FAILURE, errno, _("Failed to write output (at stage %d)"), failed_at); + } +} + + +static bool +print_name_without_quoting (const char *p, FILE *stream) +{ + return (fprintf (stream, "%s", p) >= 0); +} + + +static bool +print_name_with_quoting (register const char *p, FILE *stream) +{ + register unsigned char c; + + while ((c = *p++) != '\0') + { + int fprintf_result = -1; + switch (c) + { + case '\\': + fprintf_result = fprintf (stream, "\\\\"); + break; + + case '\n': + fprintf_result = fprintf (stream, "\\n"); + break; + + case '\b': + fprintf_result = fprintf (stream, "\\b"); + break; + + case '\r': + fprintf_result = fprintf (stream, "\\r"); + break; + + case '\t': + fprintf_result = fprintf (stream, "\\t"); + break; + + case '\f': + fprintf_result = fprintf (stream, "\\f"); + break; + + case ' ': + fprintf_result = fprintf (stream, "\\ "); + break; + + case '"': + fprintf_result = fprintf (stream, "\\\""); + break; + + default: + if (c > 040 && c < 0177) + { + if (EOF == putc (c, stream)) + return false; + fprintf_result = 1; /* otherwise it's used uninitialized. */ + } + else + { + fprintf_result = fprintf (stream, "\\%03o", (unsigned int) c); + } + } + if (fprintf_result < 0) + return false; + } + return true; +} + +static bool print_name (register const char *p, FILE *stream, int literal_control_chars) +{ + if (literal_control_chars) + return print_name_without_quoting (p, stream); + else + return print_name_with_quoting (p, stream); +} |