boinc/lib/md5_file.cpp

165 lines
3.9 KiB
C++
Raw Normal View History

// This file is part of BOINC.
// http://boinc.berkeley.edu
// Copyright (C) 2008 University of California
//
// BOINC is free software; you can redistribute it and/or modify it
// under the terms of the GNU Lesser General Public License
// as published by the Free Software Foundation,
// either version 3 of the License, or (at your option) any later version.
//
// BOINC is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
// See the GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
#if defined(_WIN32)
#include "boinc_win.h"
#else
#include "config.h"
#ifdef _USING_FCGI_
#include "boinc_fcgi.h"
#else
#include <cstdio>
#endif
#endif
#ifdef _WIN32
#include <wincrypt.h>
#endif
#ifdef ANDROID
#include <stdlib.h>
#endif
#include "error_numbers.h"
#include "md5.h"
#include "md5_file.h"
int md5_file(const char* path, char* output, double& nbytes, bool is_gzip) {
unsigned char buf[4096];
unsigned char binout[16];
md5_state_t state;
int i, n;
nbytes = 0;
#ifndef _USING_FCGI_
FILE *f = fopen(path, "rb");
#else
FILE *f = FCGI::fopen(path, "rb");
#endif
if (!f) {
fprintf(stderr, "md5_file: can't open %s\n", path);
- Added checks for net/*.h, arpa/*.h, netinet/*.h and code to figure out which of those files to include - Modified MAC address check to work on some non-Linux unixes. (mac_address.cpp) - Added suggested change to "already attached to project" checking. (ProjectInfoPage.cpp) - changed includes of standard c header files to their c++ equivalents (i.e. replaced <stdio.h> with <cstdio>) for namespace protection. - replaced "using namespace std;" with more explicit "using std::function" in several files. - Fixed bug in checking whether the os is OS/2 and added conditional OS_OS2 to the build environment. (boinc_platform.m4,configure.ac) - Changed build environment to not use -nostandardlibs unless we are using G++ and static linkage is specified. (configure.ac) - Added makefiles and package building files for solaris CSW package manager. - Fixed bug with attempting to find login name using logname. (configure.ac) - Added ifdef HAVE_* protection around some include files commonly found in sys. - Added support for unified binary for x86_64/i686-pc-solaris. (cs_platforms.cpp) - generate_host_cpid() now uses MAC address on non-linux unix. (hostinfo_network.cpp) - Macro BOINC_SET_COMPILE_FLAGS now doesn't check gcc only flags on non-gcc compilers. (boinc_set_compile_flags.m4) - Library compiles no longer depend upon the library extension or require the library to be prefixed with lib. - More fixes for fcgi builds. - Added declaration of "struct ether_addr" and ether_ntoa(). Have not yet implemented ether_ntoa() for machines that don't have it, or where it is buggy. (unix_util.h) - Added FCGI::perror() which calls FCGI_perror(). (boinc_fcgi.{h,cpp}) - Fixed library Makefiles so that all required headers get installed. svn path=/trunk/boinc/; revision=17388
2009-02-26 00:23:23 +00:00
#ifndef _USING_FCGI_
std::perror("md5_file");
#else
FCGI::perror("md5_file");
#endif
return ERR_FOPEN;
}
md5_init(&state);
// check and skip gzip header if needed
//
if (is_gzip) {
n = (int)fread(buf, 1, 10, f);
if (n != 10) {
2014-06-27 20:38:16 +00:00
fclose(f);
return ERR_BAD_FORMAT;
}
if (buf[0] != 0x1f || buf[1] != 0x8b || buf[2] != 0x08) {
2014-06-27 20:38:16 +00:00
fclose(f);
return ERR_BAD_FORMAT;
}
nbytes = 10;
}
while (1) {
n = (int)fread(buf, 1, 4096, f);
if (n<=0) break;
nbytes += n;
md5_append(&state, buf, n);
}
md5_finish(&state, binout);
for (i=0; i<16; i++) {
sprintf(output+2*i, "%02x", binout[i]);
}
output[32] = 0;
fclose(f);
return 0;
}
int md5_block(const unsigned char* data, int nbytes, char* output,
const unsigned char* data2, int nbytes2 // optional 2nd block
) {
unsigned char binout[16];
int i;
md5_state_t state;
md5_init(&state);
md5_append(&state, data, nbytes);
if (data2) {
md5_append(&state, data2, nbytes2);
}
md5_finish(&state, binout);
for (i=0; i<16; i++) {
sprintf(output+2*i, "%02x", binout[i]);
}
output[32] = 0;
return 0;
}
std::string md5_string(const unsigned char* data, int nbytes) {
char output[MD5_LEN];
md5_block(data, nbytes, output);
return std::string(output);
}
// make a secure (i.e. hard to guess)
// 32-char string using OS-supplied random bits
//
int make_secure_random_string_os(char* out) {
char buf[256];
#ifdef _WIN32
HCRYPTPROV hCryptProv;
if(! CryptAcquireContext(&hCryptProv, NULL, NULL, PROV_RSA_FULL, 0)) {
if (GetLastError() == NTE_BAD_KEYSET) {
if (!CryptAcquireContext(&hCryptProv, NULL, NULL, PROV_RSA_FULL, CRYPT_NEWKEYSET)) {
return -1;
}
} else {
return -2;
}
}
if(! CryptGenRandom(hCryptProv, (DWORD) 32, (BYTE *) buf)) {
CryptReleaseContext(hCryptProv, 0);
return -3;
}
CryptReleaseContext(hCryptProv, 0);
#elif defined ANDROID
return -1;
#else
#ifndef _USING_FCGI_
FILE* f = fopen("/dev/random", "r");
#else
FILE* f = FCGI::fopen("/dev/random", "r");
#endif
if (!f) {
return -1;
}
size_t n = fread(buf, 32, 1, f);
fclose(f);
if (n != 1) return -2;
#endif
md5_block((const unsigned char*)buf, 32, out);
return 0;
}