2014-08-25 10:18:34 +00:00
|
|
|
/*
|
2014-05-06 14:23:05 +00:00
|
|
|
* s3fs - FUSE-based file system backed by Amazon S3
|
|
|
|
*
|
|
|
|
* Copyright 2007-2008 Randy Rizun <rrizun@gmail.com>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <syslog.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <nss.h>
|
|
|
|
#include <pk11pub.h>
|
|
|
|
#include <hasht.h>
|
|
|
|
#include <prinit.h>
|
|
|
|
#include <string>
|
|
|
|
#include <map>
|
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "s3fs_auth.h"
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
// Utility Function for version
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
const char* s3fs_crypt_lib_name(void)
|
|
|
|
{
|
|
|
|
static const char version[] = "NSS";
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
// Utility Function for global init
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
bool s3fs_init_global_ssl(void)
|
|
|
|
{
|
2014-06-01 03:54:02 +00:00
|
|
|
NSS_Init(NULL);
|
2014-05-06 14:23:05 +00:00
|
|
|
NSS_NoDB_Init(NULL);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool s3fs_destroy_global_ssl(void)
|
|
|
|
{
|
|
|
|
NSS_Shutdown();
|
|
|
|
PL_ArenaFinish();
|
|
|
|
PR_Cleanup();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
// Utility Function for crypt lock
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
bool s3fs_init_crypt_mutex(void)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool s3fs_destroy_crypt_mutex(void)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
// Utility Function for HMAC
|
|
|
|
//-------------------------------------------------------------------
|
2015-02-02 16:36:08 +00:00
|
|
|
static bool s3fs_HMAC_RAW(const void* key, size_t keylen, const unsigned char* data, size_t datalen, unsigned char** digest, unsigned int* digestlen, bool is_sha256)
|
2014-05-06 14:23:05 +00:00
|
|
|
{
|
|
|
|
if(!key || 0 >= keylen || !data || 0 >= datalen || !digest || !digestlen){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
PK11SlotInfo* Slot;
|
|
|
|
PK11SymKey* pKey;
|
|
|
|
PK11Context* Context;
|
|
|
|
SECStatus SecStatus;
|
|
|
|
unsigned char tmpdigest[64];
|
2015-02-02 16:36:08 +00:00
|
|
|
SECItem KeySecItem = {siBuffer, reinterpret_cast<unsigned char*>(const_cast<void*>(key)), static_cast<unsigned int>(keylen)};
|
2014-05-06 14:23:05 +00:00
|
|
|
SECItem NullSecItem = {siBuffer, NULL, 0};
|
|
|
|
|
|
|
|
if(NULL == (Slot = PK11_GetInternalKeySlot())){
|
|
|
|
return false;
|
|
|
|
}
|
2015-02-02 16:36:08 +00:00
|
|
|
if(NULL == (pKey = PK11_ImportSymKey(Slot, (is_sha256 ? CKM_SHA256_HMAC : CKM_SHA_1_HMAC), PK11_OriginUnwrap, CKA_SIGN, &KeySecItem, NULL))){
|
2014-05-06 14:23:05 +00:00
|
|
|
PK11_FreeSlot(Slot);
|
|
|
|
return false;
|
|
|
|
}
|
2015-02-02 16:36:08 +00:00
|
|
|
if(NULL == (Context = PK11_CreateContextBySymKey((is_sha256 ? CKM_SHA256_HMAC : CKM_SHA_1_HMAC), CKA_SIGN, pKey, &NullSecItem))){
|
2014-05-06 14:23:05 +00:00
|
|
|
PK11_FreeSymKey(pKey);
|
|
|
|
PK11_FreeSlot(Slot);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*digestlen = 0;
|
|
|
|
if(SECSuccess != (SecStatus = PK11_DigestBegin(Context)) ||
|
|
|
|
SECSuccess != (SecStatus = PK11_DigestOp(Context, data, datalen)) ||
|
|
|
|
SECSuccess != (SecStatus = PK11_DigestFinal(Context, tmpdigest, digestlen, sizeof(tmpdigest))) )
|
|
|
|
{
|
|
|
|
PK11_DestroyContext(Context, PR_TRUE);
|
|
|
|
PK11_FreeSymKey(pKey);
|
|
|
|
PK11_FreeSlot(Slot);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
PK11_DestroyContext(Context, PR_TRUE);
|
|
|
|
PK11_FreeSymKey(pKey);
|
|
|
|
PK11_FreeSlot(Slot);
|
|
|
|
|
|
|
|
if(NULL == (*digest = (unsigned char*)malloc(*digestlen))){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
memcpy(*digest, tmpdigest, *digestlen);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-02-02 16:36:08 +00:00
|
|
|
bool s3fs_HMAC(const void* key, size_t keylen, const unsigned char* data, size_t datalen, unsigned char** digest, unsigned int* digestlen)
|
|
|
|
{
|
|
|
|
return s3fs_HMAC_RAW(key, keylen, data, datalen, digest, digestlen, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool s3fs_HMAC256(const void* key, size_t keylen, const unsigned char* data, size_t datalen, unsigned char** digest, unsigned int* digestlen)
|
|
|
|
{
|
|
|
|
return s3fs_HMAC_RAW(key, keylen, data, datalen, digest, digestlen, true);
|
|
|
|
}
|
|
|
|
|
2014-05-06 14:23:05 +00:00
|
|
|
//-------------------------------------------------------------------
|
|
|
|
// Utility Function for MD5
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
size_t get_md5_digest_length(void)
|
|
|
|
{
|
|
|
|
return MD5_LENGTH;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned char* s3fs_md5hexsum(int fd, off_t start, ssize_t size)
|
|
|
|
{
|
|
|
|
PK11Context* md5ctx;
|
|
|
|
unsigned char buf[512];
|
|
|
|
ssize_t bytes;
|
|
|
|
unsigned char* result;
|
|
|
|
unsigned int md5outlen;
|
|
|
|
|
|
|
|
if(-1 == size){
|
|
|
|
struct stat st;
|
|
|
|
if(-1 == fstat(fd, &st)){
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
size = static_cast<ssize_t>(st.st_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
// seek to top of file.
|
|
|
|
if(-1 == lseek(fd, start, SEEK_SET)){
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(buf, 0, 512);
|
|
|
|
md5ctx = PK11_CreateDigestContext(SEC_OID_MD5);
|
|
|
|
|
|
|
|
for(ssize_t total = 0; total < size; total += bytes){
|
|
|
|
bytes = 512 < (size - total) ? 512 : (size - total);
|
|
|
|
bytes = read(fd, buf, bytes);
|
|
|
|
if(0 == bytes){
|
|
|
|
// end of file
|
|
|
|
break;
|
|
|
|
}else if(-1 == bytes){
|
|
|
|
// error
|
|
|
|
DPRNNN("file read error(%d)", errno);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
PK11_DigestOp(md5ctx, buf, bytes);
|
|
|
|
memset(buf, 0, 512);
|
|
|
|
}
|
|
|
|
if(NULL == (result = (unsigned char*)malloc(get_md5_digest_length()))){
|
|
|
|
PK11_DestroyContext(md5ctx, PR_TRUE);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
PK11_DigestFinal(md5ctx, result, &md5outlen, get_md5_digest_length());
|
|
|
|
PK11_DestroyContext(md5ctx, PR_TRUE);
|
|
|
|
|
|
|
|
if(-1 == lseek(fd, start, SEEK_SET)){
|
|
|
|
free(result);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2015-02-02 16:36:08 +00:00
|
|
|
//-------------------------------------------------------------------
|
|
|
|
// Utility Function for SHA256
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
size_t get_sha256_digest_length(void)
|
|
|
|
{
|
|
|
|
return SHA256_LENGTH;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool s3fs_sha256(const unsigned char* data, unsigned int datalen, unsigned char** digest, unsigned int* digestlen)
|
|
|
|
{
|
|
|
|
(*digestlen) = static_cast<unsigned int>(get_sha256_digest_length());
|
|
|
|
if(NULL == ((*digest) = reinterpret_cast<unsigned char*>(malloc(*digestlen)))){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
PK11Context* sha256ctx;
|
|
|
|
unsigned int sha256outlen;
|
|
|
|
sha256ctx = PK11_CreateDigestContext(SEC_OID_SHA256);
|
|
|
|
|
|
|
|
PK11_DigestOp(sha256ctx, data, datalen);
|
|
|
|
PK11_DigestFinal(sha256ctx, *digest, &sha256outlen, *digestlen);
|
|
|
|
PK11_DestroyContext(sha256ctx, PR_TRUE);
|
|
|
|
*digestlen = sha256outlen;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned char* s3fs_sha256hexsum(int fd, off_t start, ssize_t size)
|
|
|
|
{
|
|
|
|
PK11Context* sha256ctx;
|
|
|
|
unsigned char buf[512];
|
|
|
|
ssize_t bytes;
|
|
|
|
unsigned char* result;
|
|
|
|
unsigned int sha256outlen;
|
|
|
|
|
|
|
|
if(-1 == size){
|
|
|
|
struct stat st;
|
|
|
|
if(-1 == fstat(fd, &st)){
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
size = static_cast<ssize_t>(st.st_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
// seek to top of file.
|
|
|
|
if(-1 == lseek(fd, start, SEEK_SET)){
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(buf, 0, 512);
|
|
|
|
sha256ctx = PK11_CreateDigestContext(SEC_OID_SHA256);
|
|
|
|
|
|
|
|
for(ssize_t total = 0; total < size; total += bytes){
|
|
|
|
bytes = 512 < (size - total) ? 512 : (size - total);
|
|
|
|
bytes = read(fd, buf, bytes);
|
|
|
|
if(0 == bytes){
|
|
|
|
// end of file
|
|
|
|
break;
|
|
|
|
}else if(-1 == bytes){
|
|
|
|
// error
|
|
|
|
DPRNNN("file read error(%d)", errno);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
PK11_DigestOp(sha256ctx, buf, bytes);
|
|
|
|
memset(buf, 0, 512);
|
|
|
|
}
|
|
|
|
if(NULL == (result = (unsigned char*)malloc(get_sha256_digest_length()))){
|
|
|
|
PK11_DestroyContext(sha256ctx, PR_TRUE);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
PK11_DigestFinal(sha256ctx, result, &sha256outlen, get_sha256_digest_length());
|
|
|
|
PK11_DestroyContext(sha256ctx, PR_TRUE);
|
|
|
|
|
|
|
|
if(-1 == lseek(fd, start, SEEK_SET)){
|
|
|
|
free(result);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-09-07 15:08:27 +00:00
|
|
|
/*
|
|
|
|
* Local variables:
|
|
|
|
* tab-width: 4
|
|
|
|
* c-basic-offset: 4
|
|
|
|
* End:
|
|
|
|
* vim600: noet sw=4 ts=4 fdm=marker
|
|
|
|
* vim<600: noet sw=4 ts=4
|
|
|
|
*/
|