105 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			105 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* vi: set sw=4 ts=4: */
 | 
						|
/*
 | 
						|
 * sleep implementation for busybox
 | 
						|
 *
 | 
						|
 * Copyright (C) 2003  Manuel Novoa III  <mjn3@codepoet.org>
 | 
						|
 *
 | 
						|
 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
 | 
						|
 */
 | 
						|
 | 
						|
/* BB_AUDIT SUSv3 compliant */
 | 
						|
/* BB_AUDIT GNU issues -- fancy version matches except args must be ints. */
 | 
						|
/* http://www.opengroup.org/onlinepubs/007904975/utilities/sleep.html */
 | 
						|
 | 
						|
/* Mar 16, 2003      Manuel Novoa III   (mjn3@codepoet.org)
 | 
						|
 *
 | 
						|
 * Rewritten to do proper arg and error checking.
 | 
						|
 * Also, added a 'fancy' configuration to accept multiple args with
 | 
						|
 * time suffixes for seconds, minutes, hours, and days.
 | 
						|
 */
 | 
						|
 | 
						|
#include "libbb.h"
 | 
						|
 | 
						|
/* This is a NOFORK applet. Be very careful! */
 | 
						|
 | 
						|
 | 
						|
#if ENABLE_FEATURE_FANCY_SLEEP || ENABLE_FEATURE_FLOAT_SLEEP
 | 
						|
static const struct suffix_mult sfx[] = {
 | 
						|
	{ "s", 1 },
 | 
						|
	{ "m", 60 },
 | 
						|
	{ "h", 60*60 },
 | 
						|
	{ "d", 24*60*60 },
 | 
						|
	{ }
 | 
						|
};
 | 
						|
#endif
 | 
						|
 | 
						|
int sleep_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
 | 
						|
int sleep_main(int argc UNUSED_PARAM, char **argv)
 | 
						|
{
 | 
						|
#if ENABLE_FEATURE_FLOAT_SLEEP
 | 
						|
	double duration;
 | 
						|
	struct timespec ts;
 | 
						|
#else
 | 
						|
	unsigned duration;
 | 
						|
#endif
 | 
						|
 | 
						|
	++argv;
 | 
						|
	if (!*argv)
 | 
						|
		bb_show_usage();
 | 
						|
 | 
						|
#if ENABLE_FEATURE_FLOAT_SLEEP
 | 
						|
 | 
						|
	duration = 0;
 | 
						|
	do {
 | 
						|
		char *arg = *argv;
 | 
						|
		if (strchr(arg, '.')) {
 | 
						|
			double d;
 | 
						|
			int len = strspn(arg, "0123456789.");
 | 
						|
			char sv = arg[len];
 | 
						|
			arg[len] = '\0';
 | 
						|
			d = bb_strtod(arg, NULL);
 | 
						|
			if (errno)
 | 
						|
				bb_show_usage();
 | 
						|
			arg[len] = sv;
 | 
						|
			len--;
 | 
						|
			sv = arg[len];
 | 
						|
			arg[len] = '1';
 | 
						|
			duration += d * xatoul_sfx(&arg[len], sfx);
 | 
						|
			arg[len] = sv;
 | 
						|
		} else
 | 
						|
			duration += xatoul_sfx(arg, sfx);
 | 
						|
	} while (*++argv);
 | 
						|
 | 
						|
	ts.tv_sec = MAXINT(typeof(ts.tv_sec));
 | 
						|
	ts.tv_nsec = 0;
 | 
						|
	if (duration >= 0 && duration < ts.tv_sec) {
 | 
						|
		ts.tv_sec = duration;
 | 
						|
		ts.tv_nsec = (duration - ts.tv_sec) * 1000000000;
 | 
						|
	}
 | 
						|
	do {
 | 
						|
		errno = 0;
 | 
						|
		nanosleep(&ts, &ts);
 | 
						|
	} while (errno == EINTR);
 | 
						|
 | 
						|
#elif ENABLE_FEATURE_FANCY_SLEEP
 | 
						|
 | 
						|
	duration = 0;
 | 
						|
	do {
 | 
						|
		duration += xatou_range_sfx(*argv, 0, UINT_MAX - duration, sfx);
 | 
						|
	} while (*++argv);
 | 
						|
	sleep(duration);
 | 
						|
 | 
						|
#else /* simple */
 | 
						|
 | 
						|
	duration = xatou(*argv);
 | 
						|
	sleep(duration);
 | 
						|
	// Off. If it's really needed, provide example why
 | 
						|
	//if (sleep(duration)) {
 | 
						|
	//	bb_perror_nomsg_and_die();
 | 
						|
	//}
 | 
						|
 | 
						|
#endif
 | 
						|
 | 
						|
	return EXIT_SUCCESS;
 | 
						|
}
 |