Sync usage with man page.
[netbsd-mini2440.git] / crypto / external / bsd / openssl / dist / engines / ccgost / gostsum.c
blobd57112eb54a70b39d3a28e6a869bb944ad3944e8
1 /**********************************************************************
2 * gostsum.c *
3 * Copyright (c) 2005-2006 Cryptocom LTD *
4 * This file is distributed under the same license as OpenSSL *
5 * *
6 * Almost drop-in replacement for md5sum and sha1sum *
7 * which computes GOST R 34.11-94 hashsum instead *
8 * *
9 **********************************************************************/
10 #include <stdio.h>
11 #include <stdlib.h>
12 #include <unistd.h>
13 #include <limits.h>
14 #include <fcntl.h>
15 #include <string.h>
16 #include "gosthash.h"
17 #define BUF_SIZE 262144
18 int hash_file(gost_hash_ctx *ctx,char *filename,char *sum,int mode);
19 int hash_stream(gost_hash_ctx *ctx,int fd, char *sum);
20 int get_line(FILE *f,char *hash,char *filename);
21 void help()
23 fprintf(stderr,"gostsum [-bvt] [-c [file]]| [files]\n"
24 "\t-c check message digests (default is generate)\n"
25 "\t-v verbose, print file names when checking\n"
26 "\t-b read files in binary mode\n"
27 "\t-t use test GOST paramset (default is CryptoPro paramset)\n"
28 "The input for -c should be the list of message digests and file names\n"
29 "that is printed on stdout by this program when it generates digests.\n");
30 exit(3);
33 #ifndef O_BINARY
34 #define O_BINARY 0
35 #endif
37 int main(int argc,char **argv)
39 int c,i;
40 int verbose=0;
41 int errors=0;
42 int open_mode = O_RDONLY;
43 gost_subst_block *b= &GostR3411_94_CryptoProParamSet;
44 FILE *check_file = NULL;
45 gost_hash_ctx ctx;
47 while( (c=getopt(argc,argv,"bc::tv"))!=-1)
49 switch (c)
51 case 'v': verbose=1; break;
52 case 't': b= &GostR3411_94_TestParamSet; break;
53 case 'b': open_mode |= O_BINARY; break;
54 case 'c':
55 if (optarg)
57 check_file = fopen(optarg,"r");
58 if (!check_file)
60 perror(optarg);
61 exit(2);
64 else
66 check_file= stdin;
68 break;
69 default:
70 fprintf(stderr,"invalid option %c",optopt);
71 help();
74 init_gost_hash_ctx(&ctx,b);
75 if (check_file)
77 char inhash[65],calcsum[65],filename[PATH_MAX];
78 int failcount=0,count=0;;
79 if (check_file==stdin && optind<argc)
81 check_file=fopen(argv[optind],"r");
82 if (!check_file)
84 perror(argv[optind]);
85 exit(2);
88 while (get_line(check_file,inhash,filename))
90 if (!hash_file(&ctx,filename,calcsum,open_mode))
92 exit (2);
94 count++;
95 if (!strncmp(calcsum,inhash,65))
97 if (verbose)
99 fprintf(stderr,"%s\tOK\n",filename);
102 else
104 if (verbose)
106 fprintf(stderr,"%s\tFAILED\n",filename);
108 else
110 fprintf(stderr,"%s: GOST hash sum check failed for '%s'\n",
111 argv[0],filename);
113 failcount++;
116 if (verbose && failcount)
118 fprintf(stderr,"%s: %d of %d file(f) failed GOST hash sum check\n",
119 argv[0],failcount,count);
121 exit (failcount?1:0);
123 if (optind==argc)
125 char sum[65];
126 if (!hash_stream(&ctx,fileno(stdin),sum))
128 perror("stdin");
129 exit(1);
131 printf("%s -\n",sum);
132 exit(0);
134 for (i=optind;i<argc;i++)
136 char sum[65];
137 if (!hash_file(&ctx,argv[i],sum,open_mode))
139 errors++;
141 else
143 printf("%s %s\n",sum,argv[i]);
146 exit(errors?1:0);
149 int hash_file(gost_hash_ctx *ctx,char *filename,char *sum,int mode)
151 int fd;
152 if ((fd=open(filename,mode))<0)
154 perror(filename);
155 return 0;
157 if (!hash_stream(ctx,fd,sum))
159 perror(filename);
160 return 0;
162 close(fd);
163 return 1;
166 int hash_stream(gost_hash_ctx *ctx,int fd, char *sum)
168 unsigned char buffer[BUF_SIZE];
169 ssize_t bytes;
170 int i;
171 start_hash(ctx);
172 while ((bytes=read(fd,buffer,BUF_SIZE))>0)
174 hash_block(ctx,buffer,bytes);
176 if (bytes<0)
178 return 0;
180 finish_hash(ctx,buffer);
181 for (i=0;i<32;i++)
183 sprintf(sum+2*i,"%02x",buffer[31-i]);
185 return 1;
188 int get_line(FILE *f,char *hash,char *filename)
190 int i;
191 if (fread(hash,1,64,f)<64) return 0;
192 hash[64]=0;
193 for (i=0;i<64;i++)
195 if (hash[i]<'0' || (hash[i]>'9' && hash[i]<'A') || (hash[i]>'F'
196 && hash[i]<'a')||hash[i]>'f')
198 fprintf(stderr,"Not a hash value '%s'\n",hash);
199 return 0;
202 if (fgetc(f)!=' ')
204 fprintf(stderr,"Malformed input line\n");
205 return 0;
207 i=strlen(fgets(filename,PATH_MAX,f));
208 while (filename[--i]=='\n'||filename[i]=='\r') filename[i]=0;
209 return 1;