VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/checksum/RTSha1Digest.cpp@ 42072

Last change on this file since 42072 was 32569, checked in by vboxsync, 14 years ago

Runtime: fix burn

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 5.7 KB
Line 
1/* $Id: RTSha1Digest.cpp 32569 2010-09-16 15:42:51Z vboxsync $ */
2/** @file
3 * IPRT - SHA1 digest creation
4 */
5
6/*
7 * Copyright (C) 2009-2010 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 "internal/iprt.h"
32#include <iprt/sha.h>
33
34#include <iprt/alloca.h>
35#include <iprt/assert.h>
36#include <iprt/mem.h>
37#include <iprt/string.h>
38#include <iprt/file.h>
39
40#include <openssl/sha.h>
41
42
43RTR3DECL(int) RTSha1Digest(void* pvBuf, size_t cbBuf, char **ppszDigest, PFNRTPROGRESS pfnProgressCallback, void *pvUser)
44{
45 /* Validate input */
46 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
47 AssertPtrReturn(ppszDigest, VERR_INVALID_POINTER);
48 AssertPtrNullReturn(pfnProgressCallback, VERR_INVALID_PARAMETER);
49
50 int rc = VINF_SUCCESS;
51 *ppszDigest = NULL;
52
53 /* Initialize OpenSSL. */
54 SHA_CTX ctx;
55 if (!SHA1_Init(&ctx))
56 return VERR_INTERNAL_ERROR;
57
58 /* Buffer size for progress callback */
59 double rdMulti = 100.0 / cbBuf;
60
61 /* Working buffer */
62 char *pvTmp = (char*)pvBuf;
63
64 /* Process the memory in blocks */
65 size_t cbRead;
66 size_t cbReadTotal = 0;
67 for (;;)
68 {
69 cbRead = RT_MIN(cbBuf - cbReadTotal, _1M);
70 if(!SHA1_Update(&ctx, pvTmp, cbRead))
71 {
72 rc = VERR_INTERNAL_ERROR;
73 break;
74 }
75 cbReadTotal += cbRead;
76 pvTmp += cbRead;
77
78 /* Call the progress callback if one is defined */
79 if (pfnProgressCallback)
80 {
81 rc = pfnProgressCallback((unsigned)(cbReadTotal * rdMulti), pvUser);
82 if (RT_FAILURE(rc))
83 break; /* canceled */
84 }
85 /* Finished? */
86 if (cbReadTotal == cbBuf)
87 break;
88 }
89 if (RT_SUCCESS(rc))
90 {
91 /* Finally calculate & format the SHA1 sum */
92 unsigned char auchDig[RTSHA1_HASH_SIZE];
93 if (!SHA1_Final(auchDig, &ctx))
94 return VERR_INTERNAL_ERROR;
95
96 char *pszDigest;
97 rc = RTStrAllocEx(&pszDigest, RTSHA1_DIGEST_LEN + 1);
98 if (RT_SUCCESS(rc))
99 {
100 rc = RTSha1ToString(auchDig, pszDigest, RTSHA1_DIGEST_LEN + 1);
101 if (RT_SUCCESS(rc))
102 *ppszDigest = pszDigest;
103 else
104 RTStrFree(pszDigest);
105 }
106 }
107
108 return rc;
109}
110
111RTR3DECL(int) RTSha1DigestFromFile(const char *pszFile, char **ppszDigest, PFNRTPROGRESS pfnProgressCallback, void *pvUser)
112{
113 /* Validate input */
114 AssertPtrReturn(pszFile, VERR_INVALID_POINTER);
115 AssertPtrReturn(ppszDigest, VERR_INVALID_POINTER);
116 AssertPtrNullReturn(pfnProgressCallback, VERR_INVALID_PARAMETER);
117
118 *ppszDigest = NULL;
119
120 /* Initialize OpenSSL. */
121 SHA_CTX ctx;
122 if (!SHA1_Init(&ctx))
123 return VERR_INTERNAL_ERROR;
124
125 /* Open the file to calculate a SHA1 sum of */
126 RTFILE hFile;
127 int rc = RTFileOpen(&hFile, pszFile, RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE);
128 if (RT_FAILURE(rc))
129 return rc;
130
131 /* Fetch the file size. Only needed if there is a progress callback. */
132 double rdMulti = 0;
133 if (pfnProgressCallback)
134 {
135 uint64_t cbFile;
136 rc = RTFileGetSize(hFile, &cbFile);
137 if (RT_FAILURE(rc))
138 {
139 RTFileClose(hFile);
140 return rc;
141 }
142 rdMulti = 100.0 / cbFile;
143 }
144
145 /* Allocate a reasonably large buffer, fall back on a tiny one. */
146 void *pvBufFree;
147 size_t cbBuf = _1M;
148 void *pvBuf = pvBufFree = RTMemTmpAlloc(cbBuf);
149 if (!pvBuf)
150 {
151 cbBuf = 0x1000;
152 pvBuf = alloca(cbBuf);
153 }
154
155 /* Read that file in blocks */
156 size_t cbRead;
157 size_t cbReadTotal = 0;
158 for (;;)
159 {
160 rc = RTFileRead(hFile, pvBuf, cbBuf, &cbRead);
161 if (RT_FAILURE(rc) || !cbRead)
162 break;
163 if(!SHA1_Update(&ctx, pvBuf, cbRead))
164 {
165 rc = VERR_INTERNAL_ERROR;
166 break;
167 }
168 cbReadTotal += cbRead;
169
170 /* Call the progress callback if one is defined */
171 if (pfnProgressCallback)
172 {
173 rc = pfnProgressCallback((unsigned)(cbReadTotal * rdMulti), pvUser);
174 if (RT_FAILURE(rc))
175 break; /* canceled */
176 }
177 }
178 RTMemTmpFree(pvBufFree);
179 RTFileClose(hFile);
180
181 if (RT_FAILURE(rc))
182 return rc;
183
184 /* Finally calculate & format the SHA1 sum */
185 unsigned char auchDig[RTSHA1_HASH_SIZE];
186 if (!SHA1_Final(auchDig, &ctx))
187 return VERR_INTERNAL_ERROR;
188
189 char *pszDigest;
190 rc = RTStrAllocEx(&pszDigest, RTSHA1_DIGEST_LEN + 1);
191 if (RT_SUCCESS(rc))
192 {
193 rc = RTSha1ToString(auchDig, pszDigest, RTSHA1_DIGEST_LEN + 1);
194 if (RT_SUCCESS(rc))
195 *ppszDigest = pszDigest;
196 else
197 RTStrFree(pszDigest);
198 }
199
200 return rc;
201}
202
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