Re: [PATCH] xfs_db: fix metadump redirection (again)

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Tue, Jul 25, 2017 at 11:32:23AM -0500, Eric Sandeen wrote:
> On 07/25/2017 11:27 AM, Darrick J. Wong wrote:
> > In patch 4944defad4 ("xfs_db: redirect printfs when metadumping to
> > stdout"), we solved the problem of xfs_db printfs ending up in the
> > metadump stream by reassigning stdout for the duration of a stdout
> > metadump.  Unfortunately, musl doesn't allow stdout to be reassigned (in
> > their view "extern FILE *stdout" means "extern FILE * const stdout"), so
> > we abandon the old approach in favor of playing games with dup() to
> > switch the raw file descriptors.
> > 
> > While we're at it, fix a regression where an unconverted outf test
> > allows progress info to end up in the metadump stream.
> 
> "while we're at it" usually indicates the need for a separate patch ;)

Ok, though the "while we're at it" fix is only necessary if we keep the
stdout redirection stuff.

> So what if we just converted dbprintf to do fprintf, with the destination
> held in some global variable that metadump can change?   Seems like that
> might be less tricksy, and xfs_db is no stranger to globals.
> 
> there are about 5 bare printfs in xfs_db, but none should be encountered
> during a metadump (well ... maybe the one in set_cur).

Less tricksy, but more of a maintenance burden for us, because now we
can't ever allow printf() or fprintf(stdout) in any code path that gets
called from metadump.  The reviewers will have to remember (and document
for future reviewers).

OTOH, if we let printfs slip in to something that isn't called by
metadump and then a later patch moves it into the metadump call graph,
now we have a logic bomb that will manifest itself in silently corrupted
metadumps again.

So that's why I went with playing games making stdout (or now just fd 1)
dump to stderr -- the buffoonery required to make metadumping to stdout
work properly is contained in db/metadump.c.

--D

> 
> -eric
> 
> > Signed-off-by: Darrick J. Wong <darrick.wong@xxxxxxxxxx>
> > ---
> >  db/metadump.c |   47 ++++++++++++++++++++++++++++++++++++-----------
> >  1 file changed, 36 insertions(+), 11 deletions(-)
> > 
> > diff --git a/db/metadump.c b/db/metadump.c
> > index b58acef..8cdcb37 100644
> > --- a/db/metadump.c
> > +++ b/db/metadump.c
> > @@ -78,6 +78,7 @@ static int		obfuscate = 1;
> >  static int		zero_stale_data = 1;
> >  static int		show_warnings = 0;
> >  static int		progress_since_warning = 0;
> > +static bool		stdout_metadump;
> >  
> >  void
> >  metadump_init(void)
> > @@ -137,7 +138,7 @@ print_progress(const char *fmt, ...)
> >  	va_end(ap);
> >  	buf[sizeof(buf)-1] = '\0';
> >  
> > -	f = (outf == stdout) ? stderr : stdout;
> > +	f = stdout_metadump ? stderr : stdout;
> >  	fprintf(f, "\r%-59s", buf);
> >  	fflush(f);
> >  	progress_since_warning = 1;
> > @@ -2874,7 +2875,8 @@ metadump_f(
> >  	xfs_agnumber_t	agno;
> >  	int		c;
> >  	int		start_iocur_sp;
> > -	bool		stdout_metadump = false;
> > +	int		outfd = -1;
> > +	int		ret;
> >  	char		*p;
> >  
> >  	exitcode = 1;
> > @@ -2994,16 +2996,35 @@ metadump_f(
> >  		 * metadump operation so that dbprintf and other messages
> >  		 * are sent to the console instead of polluting the
> >  		 * metadump stream.
> > +		 *
> > +		 * We get to do this the hard way because musl doesn't
> > +		 * allow reassignment of stdout.
> >  		 */
> > -		outf = stdout;
> > -		stdout = stderr;
> > +		fflush(stdout);
> > +		outfd = dup(STDOUT_FILENO);
> > +		if (outfd < 0) {
> > +			perror("opening dump stream");
> > +			goto out;
> > +		}
> > +		ret = dup2(STDERR_FILENO, STDOUT_FILENO);
> > +		if (ret < 0) {
> > +			perror("redirecting stdout");
> > +			close(outfd);
> > +			goto out;
> > +		}
> > +		outf = fdopen(outfd, "a");
> > +		if (outf == NULL) {
> > +			fprintf(stderr, "cannot create dump stream\n");
> > +			dup2(outfd, 1);
> > +			close(outfd);
> > +			goto out;
> > +		}
> >  		stdout_metadump = true;
> >  	} else {
> >  		outf = fopen(argv[optind], "wb");
> >  		if (outf == NULL) {
> >  			print_warning("cannot create dump file");
> > -			free(metablock);
> > -			return 0;
> > +			goto out;
> >  		}
> >  	}
> >  
> > @@ -3031,15 +3052,19 @@ metadump_f(
> >  	if (progress_since_warning)
> >  		fputc('\n', stdout_metadump ? stderr : stdout);
> >  
> > -	if (stdout_metadump)
> > -		stdout = outf;
> > -	else
> > -		fclose(outf);
> > +	if (stdout_metadump) {
> > +		fflush(outf);
> > +		fflush(stdout);
> > +		ret = dup2(outfd, STDOUT_FILENO);
> > +		if (ret < 0)
> > +			perror("un-redirecting stdout");
> > +	}
> > +	fclose(outf);
> >  
> >  	/* cleanup iocur stack */
> >  	while (iocur_sp > start_iocur_sp)
> >  		pop_cur();
> > -
> > +out:
> >  	free(metablock);
> >  
> >  	return 0;
> > --
> > To unsubscribe from this list: send the line "unsubscribe linux-xfs" in
> > the body of a message to majordomo@xxxxxxxxxxxxxxx
> > More majordomo info at  http://vger.kernel.org/majordomo-info.html
> > 
> --
> To unsubscribe from this list: send the line "unsubscribe linux-xfs" in
> the body of a message to majordomo@xxxxxxxxxxxxxxx
> More majordomo info at  http://vger.kernel.org/majordomo-info.html
--
To unsubscribe from this list: send the line "unsubscribe linux-xfs" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [XFS Filesystem Development (older mail)]     [Linux Filesystem Development]     [Linux Audio Users]     [Yosemite Trails]     [Linux Kernel]     [Linux RAID]     [Linux SCSI]


  Powered by Linux