 bc0a39238e
			
		
	
	
		bc0a39238e
		
	
	
	
	
		
			
			. also imports seq(1) to help it . add -C option to time(1) to print tsc difference . increase col width for ministat for tsc numbers
		
			
				
	
	
		
			188 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			188 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* time - time a command	Authors: Andy Tanenbaum & Michiel Huisjes */
 | |
| 
 | |
| #define NEW	1
 | |
| 
 | |
| #include <sys/types.h>
 | |
| #include <sys/times.h>
 | |
| #include <limits.h>
 | |
| #include <time.h>
 | |
| #include <signal.h>
 | |
| #include <stdlib.h>
 | |
| #include <unistd.h>
 | |
| #include <sys/wait.h>
 | |
| #include <minix/minlib.h>
 | |
| #include <minix/u64.h>
 | |
| #include <stdio.h>
 | |
| 
 | |
| /* -DNEW prints time to 0.01 sec. */
 | |
| #ifdef NEW		
 | |
| #define HUNDREDTHS 1
 | |
| #endif
 | |
| 
 | |
| char **args;
 | |
| char *name;
 | |
| 
 | |
| int digit_seen;
 | |
| char a[] = "        . \0";
 | |
| 
 | |
| _PROTOTYPE(int main, (int argc, char **argv));
 | |
| _PROTOTYPE(void print_time, (clock_t t));
 | |
| _PROTOTYPE(void twin, (int n, char *p));
 | |
| _PROTOTYPE(void execute, (void));
 | |
| 
 | |
| int main(argc, argv)
 | |
| int argc;
 | |
| char *argv[];
 | |
| {
 | |
|   int cycles = 0;
 | |
|   struct tms pre_buf, post_buf;
 | |
|   int status, pid;
 | |
| #if _VMD_EXT
 | |
|   struct timeval start_time, end_time;
 | |
| #else
 | |
|   struct tms dummy;
 | |
|   int start_time, end_time;
 | |
| #endif
 | |
|   u64_t start_tsc, end_tsc, spent_tsc;
 | |
|   clock_t real_time;
 | |
|   int c;
 | |
| 
 | |
|   if (argc == 1) exit(0);
 | |
| 
 | |
|   while((c=getopt(argc, argv, "C")) != EOF) {
 | |
|     switch(c) {
 | |
|   	case 'C':
 | |
| 		cycles = 1;
 | |
| 		break;
 | |
| 	default:
 | |
| 		fprintf(stderr, "usage: time [-C] <command>\n");
 | |
| 		exit(1);
 | |
|     }
 | |
|   }
 | |
|   
 | |
|   argv += optind;
 | |
|   argc -= optind;
 | |
| 
 | |
|   args = &argv[0];
 | |
|   name = argv[0];
 | |
| 
 | |
|   /* Get real time at start of run. */
 | |
| #if _VMD_EXT
 | |
|   (void) sysutime(UTIME_TIMEOFDAY, &start_time);
 | |
| #else
 | |
|   start_time = times(&dummy);
 | |
| #endif
 | |
|   read_tsc_64(&start_tsc);
 | |
| 
 | |
|   /* Fork off child. */
 | |
|   if ((pid = fork()) < 0) {
 | |
| 	std_err("Cannot fork\n");
 | |
| 	exit(1);
 | |
|   }
 | |
|   if (pid == 0) execute();
 | |
| 
 | |
|   /* Parent is the time program.  Disable interrupts and wait. */
 | |
|   signal(SIGINT, SIG_IGN);
 | |
|   signal(SIGQUIT, SIG_IGN);
 | |
| 
 | |
|   do {
 | |
| 	times(&pre_buf);
 | |
|   } while (wait(&status) != pid);
 | |
|   read_tsc_64(&end_tsc);
 | |
|   spent_tsc = sub64(end_tsc, start_tsc);
 | |
| #if _VMD_EXT
 | |
|   (void) sysutime(UTIME_TIMEOFDAY, &end_time);
 | |
|   real_time = (end_time.tv_sec - start_time.tv_sec) * CLOCKS_PER_SEC
 | |
| 	+ (end_time.tv_usec - start_time.tv_usec) * CLOCKS_PER_SEC / 1000000;
 | |
| #else
 | |
|   end_time = times(&dummy);
 | |
|   real_time = (end_time - start_time);
 | |
| #endif
 | |
| 
 | |
|   if ((status & 0377) != 0) std_err("Command terminated abnormally.\n");
 | |
|   times(&post_buf);
 | |
| 
 | |
|   if(cycles) {
 | |
|   	fprintf(stderr, "%qd tsc ", spent_tsc);
 | |
|   }
 | |
|   /* Print results. -DNEW enables time on one line to 0.01 sec */
 | |
| #ifndef NEW
 | |
|   std_err("real ");
 | |
|   print_time(real_time);
 | |
|   std_err("\nuser ");
 | |
|   print_time(post_buf.tms_cutime - pre_buf.tms_cutime);
 | |
|   std_err("\nsys  ");
 | |
|   print_time(post_buf.tms_cstime - pre_buf.tms_cstime);
 | |
|   std_err("\n");
 | |
| #else
 | |
|   print_time(real_time);
 | |
|   std_err(" real");
 | |
|   print_time(post_buf.tms_cutime - pre_buf.tms_cutime);
 | |
|   std_err(" user");
 | |
|   print_time(post_buf.tms_cstime - pre_buf.tms_cstime);
 | |
|   std_err(" sys\n");
 | |
| #endif
 | |
|   return((status & 0377) ? -1 : (status >> 8));
 | |
| }
 | |
| 
 | |
| void print_time(t)
 | |
| register clock_t t;
 | |
| {
 | |
| /* Print the time 't' in hours: minutes: seconds.  't' is in ticks. */
 | |
| 
 | |
|   int hours, minutes, seconds, hundredths, i;
 | |
|   u32_t system_hz;
 | |
| 
 | |
|   system_hz = (u32_t) sysconf(_SC_CLK_TCK);
 | |
| 
 | |
|   digit_seen = 0;
 | |
|   for (i = 0; i < 8; i++) a[i] = ' ';
 | |
|   hours = (int) (t / ((clock_t) 3600 * system_hz));
 | |
|   t -= (clock_t) hours * 3600 * system_hz;
 | |
|   minutes = (int) (t / ((clock_t) 60 * system_hz));
 | |
|   t -= (clock_t) minutes * 60 * system_hz;
 | |
|   seconds = (int) (t / system_hz);
 | |
|   t -= (clock_t) seconds * system_hz;
 | |
|   hundredths = (int) (t * 100 / system_hz);
 | |
| 
 | |
|   if (hours) {
 | |
| 	twin(hours, &a[0]);
 | |
| 	a[2] = ':';
 | |
|   }
 | |
|   if (minutes || digit_seen) {
 | |
| 	twin(minutes, &a[3]);
 | |
| 	a[5] = ':';
 | |
|   }
 | |
|   if (seconds || digit_seen)
 | |
| 	twin(seconds, &a[6]);
 | |
|   else
 | |
| 	a[7] = '0';
 | |
|   a[9] = hundredths / 10 + '0';
 | |
| #ifdef HUNDREDTHS		/* tenths used to be enough */
 | |
|   a[10] = hundredths % 10 + '0';
 | |
| #endif
 | |
|   std_err(a);
 | |
| }
 | |
| 
 | |
| void twin(n, p)
 | |
| int n;
 | |
| char *p;
 | |
| {
 | |
|   char c1, c2;
 | |
|   c1 = (n / 10) + '0';
 | |
|   c2 = (n % 10) + '0';
 | |
|   if (digit_seen == 0 && c1 == '0') c1 = ' ';
 | |
|   *p++ = c1;
 | |
|   *p++ = c2;
 | |
|   if (n > 0) digit_seen = 1;
 | |
| }
 | |
| 
 | |
| void execute()
 | |
| {
 | |
|   execvp(name, args);
 | |
|   std_err("Cannot execute ");
 | |
|   std_err(name);
 | |
|   std_err("\n");
 | |
|   exit(-1);
 | |
| }
 |