122 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* sum - checksum a file		Author: Martin C. Atkins */
 | |
| 
 | |
| /*
 | |
|  *	This program was written by:
 | |
|  *		Martin C. Atkins,
 | |
|  *		University of York,
 | |
|  *		Heslington,
 | |
|  *		York. Y01 5DD
 | |
|  *		England
 | |
|  *	and is released into the public domain, on the condition
 | |
|  *	that this comment is always included without alteration.
 | |
|  */
 | |
| 
 | |
| #include <sys/types.h>
 | |
| #include <fcntl.h>
 | |
| #include <stdlib.h>
 | |
| #include <unistd.h>
 | |
| #include <minix/minlib.h>
 | |
| #include <stdio.h>
 | |
| 
 | |
| #define BUFFER_SIZE (512)
 | |
| 
 | |
| int rc = 0;
 | |
| 
 | |
| char *defargv[] = {"-", 0};
 | |
| 
 | |
| _PROTOTYPE(int main, (int argc, char **argv));
 | |
| _PROTOTYPE(void error, (char *s, char *f));
 | |
| _PROTOTYPE(void sum, (int fd, char *fname));
 | |
| _PROTOTYPE(void putd, (int number, int fw, int zeros));
 | |
| 
 | |
| int main(argc, argv)
 | |
| int argc;
 | |
| char *argv[];
 | |
| {
 | |
|   register int fd;
 | |
| 
 | |
|   if (*++argv == 0) argv = defargv;
 | |
|   for (; *argv; argv++) {
 | |
| 	if (argv[0][0] == '-' && argv[0][1] == '\0')
 | |
| 		fd = 0;
 | |
| 	else
 | |
| 		fd = open(*argv, O_RDONLY);
 | |
| 
 | |
| 	if (fd == -1) {
 | |
| 		error("can't open ", *argv);
 | |
| 		rc = 1;
 | |
| 		continue;
 | |
| 	}
 | |
| 	sum(fd, (argc > 2) ? *argv : (char *) 0);
 | |
| 	if (fd != 0) close(fd);
 | |
|   }
 | |
|   return(rc);
 | |
| }
 | |
| 
 | |
| void error(s, f)
 | |
| char *s, *f;
 | |
| {
 | |
| 
 | |
|   std_err("sum: ");
 | |
|   std_err(s);
 | |
| 
 | |
|   if (f) std_err(f);
 | |
|   std_err("\n");
 | |
| }
 | |
| 
 | |
| void sum(fd, fname)
 | |
| int fd;
 | |
| char *fname;
 | |
| {
 | |
|   char buf[BUFFER_SIZE];
 | |
|   register int i, n;
 | |
|   long size = 0;
 | |
|   unsigned crc = 0;
 | |
|   unsigned tmp, blks;
 | |
| 
 | |
|   while ((n = read(fd, buf, BUFFER_SIZE)) > 0) {
 | |
| 	for (i = 0; i < n; i++) {
 | |
| 		crc = (crc >> 1) + ((crc & 1) ? 0x8000 : 0);
 | |
| 		tmp = buf[i] & 0377;
 | |
| 		crc += tmp;
 | |
| 		crc &= 0xffff;
 | |
| 		size++;
 | |
| 	}
 | |
|   }
 | |
| 
 | |
|   if (n < 0) {
 | |
| 	if (fname)
 | |
| 		error("read error on ", fname);
 | |
| 	else
 | |
| 		error("read error", (char *) 0);
 | |
| 	rc = 1;
 | |
| 	return;
 | |
|   }
 | |
|   putd(crc, 5, 1);
 | |
|   blks = (size + (long) BUFFER_SIZE - 1L) / (long) BUFFER_SIZE;
 | |
|   putd(blks, 6, 0);
 | |
|   if (fname) printf(" %s", fname);
 | |
|   printf("\n");
 | |
| }
 | |
| 
 | |
| void putd(number, fw, zeros)
 | |
| int number, fw, zeros;
 | |
| {
 | |
| /* Put a decimal number, in a field width, to stdout. */
 | |
| 
 | |
|   char buf[10];
 | |
|   int n;
 | |
|   unsigned num;
 | |
| 
 | |
|   num = (unsigned) number;
 | |
|   for (n = 0; n < fw; n++) {
 | |
| 	if (num || n == 0) {
 | |
| 		buf[fw - n - 1] = '0' + num % 10;
 | |
| 		num /= 10;
 | |
| 	} else
 | |
| 		buf[fw - n - 1] = zeros ? '0' : ' ';
 | |
|   }
 | |
|   buf[fw] = 0;
 | |
|   printf("%s", buf);
 | |
| }
 | 
