VirtualBox

source: vbox/trunk/src/VBox/Runtime/testcase/tstRTDigest.cpp@ 51844

Last change on this file since 51844 was 51820, checked in by vboxsync, 11 years ago

New message digest testcase + a fix to digest-builtin.cpp.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 16.4 KB
Line 
1/* $Id: tstRTDigest.cpp 51820 2014-07-02 19:32:59Z vboxsync $ */
2/** @file
3 * IPRT Testcase - RTSha*, RTMd5, RTCrc*.
4 */
5
6/*
7 * Copyright (C) 2009-2011 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#include <iprt/sha.h>
32#include <iprt/md2.h>
33#include <iprt/md5.h>
34#include <iprt/crc.h>
35
36#include <iprt/ctype.h>
37#include <iprt/err.h>
38#include <iprt/file.h>
39#include <iprt/getopt.h>
40#include <iprt/initterm.h>
41#include <iprt/param.h>
42#include <iprt/path.h>
43#include <iprt/process.h>
44#include <iprt/string.h>
45#include <iprt/stream.h>
46
47
48static int Error(const char *pszFormat, ...)
49{
50 char szName[RTPATH_MAX];
51 if (!RTProcGetExecutablePath(szName, sizeof(szName)))
52 strcpy(szName, "tstRTDigest");
53
54 RTStrmPrintf(g_pStdErr, "%s: error: ", RTPathFilename(szName));
55 va_list va;
56 va_start(va, pszFormat);
57 RTStrmPrintfV(g_pStdErr, pszFormat, va);
58 va_end(va);
59
60 return 1;
61}
62
63
64static int MyReadFile(RTFILE hFile, void *pvBuf, size_t cbToRead, size_t *pcbRead, uint64_t *pcbMaxLeft)
65{
66 int rc = VINF_SUCCESS;
67 if (*pcbMaxLeft > 0)
68 {
69 if (cbToRead > *pcbMaxLeft)
70 cbToRead = (size_t)*pcbMaxLeft;
71 rc = RTFileRead(hFile, pvBuf, cbToRead, pcbRead);
72 if (RT_SUCCESS(rc))
73 *pcbMaxLeft -= *pcbRead;
74 }
75 else
76 *pcbRead = 0;
77 return rc;
78}
79
80
81
82int main(int argc, char **argv)
83{
84 RTR3InitExe(argc, &argv, 0);
85
86 enum
87 {
88 kDigestType_NotSpecified,
89 kDigestType_CRC32,
90 kDigestType_CRC64,
91 kDigestType_MD2,
92 kDigestType_MD5,
93 kDigestType_SHA1,
94 kDigestType_SHA256,
95 kDigestType_SHA512
96 } enmDigestType = kDigestType_NotSpecified;
97 const char *pszDigestType = "NotSpecified";
98
99 enum
100 {
101 kMethod_Full,
102 kMethod_Block,
103 kMethod_File
104 } enmMethod = kMethod_Block;
105
106 uint64_t offStart = 0;
107 uint64_t cbMax = UINT64_MAX;
108 bool fTestcase = false;
109
110 static const RTGETOPTDEF s_aOptions[] =
111 {
112 { "--type", 't', RTGETOPT_REQ_STRING },
113 { "--method", 'm', RTGETOPT_REQ_STRING },
114 { "--help", 'h', RTGETOPT_REQ_NOTHING },
115 { "--length", 'l', RTGETOPT_REQ_UINT64 },
116 { "--offset", 'o', RTGETOPT_REQ_UINT64 },
117 { "--testcase", 'x', RTGETOPT_REQ_NOTHING },
118 };
119
120 int ch;
121 RTGETOPTUNION ValueUnion;
122 RTGETOPTSTATE GetState;
123 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
124 while ((ch = RTGetOpt(&GetState, &ValueUnion)))
125 {
126 switch (ch)
127 {
128 case 't':
129 if (!RTStrICmp(ValueUnion.psz, "crc32"))
130 {
131 pszDigestType = "CRC32";
132 enmDigestType = kDigestType_CRC32;
133 }
134 else if (!RTStrICmp(ValueUnion.psz, "crc64"))
135 {
136 pszDigestType = "CRC64";
137 enmDigestType = kDigestType_CRC64;
138 }
139 else if (!RTStrICmp(ValueUnion.psz, "md2"))
140 {
141 pszDigestType = "MD2";
142 enmDigestType = kDigestType_MD2;
143 }
144 else if (!RTStrICmp(ValueUnion.psz, "md5"))
145 {
146 pszDigestType = "MD5";
147 enmDigestType = kDigestType_MD5;
148 }
149 else if (!RTStrICmp(ValueUnion.psz, "sha1"))
150 {
151 pszDigestType = "SHA-1";
152 enmDigestType = kDigestType_SHA1;
153 }
154 else if (!RTStrICmp(ValueUnion.psz, "sha256"))
155 {
156 pszDigestType = "SHA-256";
157 enmDigestType = kDigestType_SHA256;
158 }
159 else if (!RTStrICmp(ValueUnion.psz, "sha512"))
160 {
161 pszDigestType = "SHA-512";
162 enmDigestType = kDigestType_SHA512;
163 }
164 else
165 {
166 Error("Invalid digest type: %s\n", ValueUnion.psz);
167 return 1;
168 }
169 break;
170
171 case 'm':
172 if (!RTStrICmp(ValueUnion.psz, "full"))
173 enmMethod = kMethod_Full;
174 else if (!RTStrICmp(ValueUnion.psz, "block"))
175 enmMethod = kMethod_Block;
176 else if (!RTStrICmp(ValueUnion.psz, "file"))
177 enmMethod = kMethod_File;
178 else
179 {
180 Error("Invalid digest method: %s\n", ValueUnion.psz);
181 return 1;
182 }
183 break;
184
185 case 'l':
186 cbMax = ValueUnion.u64;
187 break;
188
189 case 'o':
190 offStart = ValueUnion.u64;
191 break;
192
193 case 'x':
194 fTestcase = true;
195 break;
196
197 case 'h':
198 RTPrintf("usage: tstRTDigest -t <digest-type> [-o <offset>] [-l <length>] [-x] file [file2 [..]]\n");
199 return 1;
200
201 case VINF_GETOPT_NOT_OPTION:
202 {
203 if (enmDigestType == kDigestType_NotSpecified)
204 return Error("No digest type was specified\n");
205
206 switch (enmMethod)
207 {
208 case kMethod_Full:
209 return Error("Full file method is not implemented\n");
210
211 case kMethod_File:
212 if (offStart != 0 || cbMax != UINT64_MAX)
213 return Error("The -l and -o options do not work with the 'file' method.");
214 switch (enmDigestType)
215 {
216 case kDigestType_SHA1:
217 {
218 char *pszDigest;
219 int rc = RTSha1DigestFromFile(ValueUnion.psz, &pszDigest, NULL, NULL);
220 if (RT_FAILURE(rc))
221 return Error("RTSha1Digest(%s,) -> %Rrc\n", ValueUnion.psz, rc);
222 RTPrintf("%s %s\n", pszDigest, ValueUnion.psz);
223 RTStrFree(pszDigest);
224 break;
225 }
226
227 case kDigestType_SHA256:
228 {
229 char *pszDigest;
230 int rc = RTSha256DigestFromFile(ValueUnion.psz, &pszDigest, NULL, NULL);
231 if (RT_FAILURE(rc))
232 return Error("RTSha256Digest(%s,) -> %Rrc\n", ValueUnion.psz, rc);
233 RTPrintf("%s %s\n", pszDigest, ValueUnion.psz);
234 RTStrFree(pszDigest);
235 break;
236 }
237 default:
238 return Error("The file method isn't implemented for this digest\n");
239 }
240 break;
241
242 case kMethod_Block:
243 {
244 RTFILE hFile;
245 int rc = RTFileOpen(&hFile, ValueUnion.psz, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_WRITE);
246 if (RT_FAILURE(rc))
247 return Error("RTFileOpen(,%s,) -> %Rrc\n", ValueUnion.psz, rc);
248 if (offStart != 0)
249 {
250 rc = RTFileSeek(hFile, offStart, RTFILE_SEEK_BEGIN, NULL);
251 if (RT_FAILURE(rc))
252 return Error("RTFileSeek(%s,%ull) -> %Rrc\n", ValueUnion.psz, offStart, rc);
253 }
254
255 uint64_t cbMaxLeft = cbMax;
256 size_t cbRead;
257 uint8_t abBuf[_64K];
258 char *pszDigest = (char *)&abBuf[0];
259 switch (enmDigestType)
260 {
261 case kDigestType_CRC32:
262 {
263 uint32_t uCRC32 = RTCrc32Start();
264 for (;;)
265 {
266 rc = MyReadFile(hFile, abBuf, sizeof(abBuf), &cbRead, &cbMaxLeft);
267 if (RT_FAILURE(rc) || !cbRead)
268 break;
269 uCRC32 = RTCrc32Process(uCRC32, abBuf, cbRead);
270 }
271 uCRC32 = RTCrc32Finish(uCRC32);
272 RTStrPrintf(pszDigest, sizeof(abBuf), "%08RX32", uCRC32);
273 break;
274 }
275
276 case kDigestType_CRC64:
277 {
278 uint64_t uCRC64 = RTCrc64Start();
279 for (;;)
280 {
281 rc = MyReadFile(hFile, abBuf, sizeof(abBuf), &cbRead, &cbMaxLeft);
282 if (RT_FAILURE(rc) || !cbRead)
283 break;
284 uCRC64 = RTCrc64Process(uCRC64, abBuf, cbRead);
285 }
286 uCRC64 = RTCrc64Finish(uCRC64);
287 RTStrPrintf(pszDigest, sizeof(abBuf), "%016RX64", uCRC64);
288 break;
289 }
290
291 case kDigestType_MD2:
292 {
293 RTMD2CONTEXT Ctx;
294 RTMd2Init(&Ctx);
295 for (;;)
296 {
297 rc = MyReadFile(hFile, abBuf, sizeof(abBuf), &cbRead, &cbMaxLeft);
298 if (RT_FAILURE(rc) || !cbRead)
299 break;
300 RTMd2Update(&Ctx, abBuf, cbRead);
301 }
302 uint8_t abDigest[RTMD2_HASH_SIZE];
303 RTMd2Final(&Ctx, abDigest);
304 RTMd2ToString(abDigest, pszDigest, sizeof(abBuf));
305 break;
306 }
307
308 case kDigestType_MD5:
309 {
310 RTMD5CONTEXT Ctx;
311 RTMd5Init(&Ctx);
312 for (;;)
313 {
314 rc = MyReadFile(hFile, abBuf, sizeof(abBuf), &cbRead, &cbMaxLeft);
315 if (RT_FAILURE(rc) || !cbRead)
316 break;
317 RTMd5Update(&Ctx, abBuf, cbRead);
318 }
319 uint8_t abDigest[RTMD5HASHSIZE];
320 RTMd5Final(abDigest, &Ctx);
321 RTMd5ToString(abDigest, pszDigest, sizeof(abBuf));
322 break;
323 }
324
325 case kDigestType_SHA1:
326 {
327 RTSHA1CONTEXT Ctx;
328 RTSha1Init(&Ctx);
329 for (;;)
330 {
331 rc = MyReadFile(hFile, abBuf, sizeof(abBuf), &cbRead, &cbMaxLeft);
332 if (RT_FAILURE(rc) || !cbRead)
333 break;
334 RTSha1Update(&Ctx, abBuf, cbRead);
335 }
336 uint8_t abDigest[RTSHA1_HASH_SIZE];
337 RTSha1Final(&Ctx, abDigest);
338 RTSha1ToString(abDigest, pszDigest, sizeof(abBuf));
339 break;
340 }
341
342 case kDigestType_SHA256:
343 {
344 RTSHA256CONTEXT Ctx;
345 RTSha256Init(&Ctx);
346 for (;;)
347 {
348 rc = MyReadFile(hFile, abBuf, sizeof(abBuf), &cbRead, &cbMaxLeft);
349 if (RT_FAILURE(rc) || !cbRead)
350 break;
351 RTSha256Update(&Ctx, abBuf, cbRead);
352 }
353 uint8_t abDigest[RTSHA256_HASH_SIZE];
354 RTSha256Final(&Ctx, abDigest);
355 RTSha256ToString(abDigest, pszDigest, sizeof(abBuf));
356 break;
357 }
358
359 case kDigestType_SHA512:
360 {
361 RTSHA512CONTEXT Ctx;
362 RTSha512Init(&Ctx);
363 for (;;)
364 {
365 rc = MyReadFile(hFile, abBuf, sizeof(abBuf), &cbRead, &cbMaxLeft);
366 if (RT_FAILURE(rc) || !cbRead)
367 break;
368 RTSha512Update(&Ctx, abBuf, cbRead);
369 }
370 uint8_t abDigest[RTSHA512_HASH_SIZE];
371 RTSha512Final(&Ctx, abDigest);
372 RTSha512ToString(abDigest, pszDigest, sizeof(abBuf));
373 break;
374 }
375
376 default:
377 return Error("Internal error #1\n");
378 }
379 RTFileClose(hFile);
380 if (RT_FAILURE(rc) && rc != VERR_EOF)
381 {
382 RTPrintf("Partial: %s %s\n", pszDigest, ValueUnion.psz);
383 return Error("RTFileRead(%s) -> %Rrc\n", ValueUnion.psz, rc);
384 }
385
386 if (!fTestcase)
387 RTPrintf("%s %s\n", pszDigest, ValueUnion.psz);
388 else if (offStart)
389 RTPrintf(" { &g_abRandom72KB[%#4llx], %5llu, \"%s\", \"%s %llu bytes @%llu\" },\n",
390 offStart, cbMax - cbMaxLeft, pszDigest, pszDigestType, offStart, cbMax - cbMaxLeft);
391 else
392 RTPrintf(" { &g_abRandom72KB[0], %5llu, \"%s\", \"%s %llu bytes\" },\n",
393 cbMax - cbMaxLeft, pszDigest, pszDigestType, cbMax - cbMaxLeft);
394 break;
395 }
396
397 default:
398 return Error("Internal error #2\n");
399 }
400 break;
401 }
402
403 default:
404 return RTGetOptPrintError(ch, &ValueUnion);
405 }
406 }
407
408 return 0;
409}
410
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette