subtly
10 years ago
19 changed files with 1143 additions and 193 deletions
@ -0,0 +1,89 @@ |
|||||
|
/*
|
||||
|
This file is part of ethash. |
||||
|
|
||||
|
ethash 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 3 of the License, or |
||||
|
(at your option) any later version. |
||||
|
|
||||
|
ethash 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 ethash. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
*/ |
||||
|
/** @file io.c
|
||||
|
* @author Lefteris Karapetsas <lefteris@ethdev.com> |
||||
|
* @date 2015 |
||||
|
*/ |
||||
|
#include "io.h" |
||||
|
#include <string.h> |
||||
|
#include <stdio.h> |
||||
|
|
||||
|
// silly macro to save some typing
|
||||
|
#define PASS_ARR(c_) (c_), sizeof(c_) |
||||
|
|
||||
|
static bool ethash_io_write_file(char const *dirname, |
||||
|
char const* filename, |
||||
|
size_t filename_length, |
||||
|
void const* data, |
||||
|
size_t data_size) |
||||
|
{ |
||||
|
bool ret = false; |
||||
|
char *fullname = ethash_io_create_filename(dirname, filename, filename_length); |
||||
|
if (!fullname) { |
||||
|
return false; |
||||
|
} |
||||
|
FILE *f = fopen(fullname, "wb"); |
||||
|
if (!f) { |
||||
|
goto free_name; |
||||
|
} |
||||
|
if (data_size != fwrite(data, 1, data_size, f)) { |
||||
|
goto close; |
||||
|
} |
||||
|
|
||||
|
ret = true; |
||||
|
close: |
||||
|
fclose(f); |
||||
|
free_name: |
||||
|
free(fullname); |
||||
|
return ret; |
||||
|
} |
||||
|
|
||||
|
bool ethash_io_write(char const *dirname, |
||||
|
ethash_params const* params, |
||||
|
ethash_blockhash_t seedhash, |
||||
|
void const* cache, |
||||
|
uint8_t **data, |
||||
|
size_t *data_size) |
||||
|
{ |
||||
|
char info_buffer[DAG_MEMO_BYTESIZE]; |
||||
|
// allocate the bytes
|
||||
|
uint8_t *temp_data_ptr = malloc(params->full_size); |
||||
|
if (!temp_data_ptr) { |
||||
|
goto end; |
||||
|
} |
||||
|
ethash_compute_full_data(temp_data_ptr, params, cache); |
||||
|
|
||||
|
if (!ethash_io_write_file(dirname, PASS_ARR(DAG_FILE_NAME), temp_data_ptr, params->full_size)) { |
||||
|
goto fail_free; |
||||
|
} |
||||
|
|
||||
|
ethash_io_serialize_info(REVISION, seedhash, info_buffer); |
||||
|
if (!ethash_io_write_file(dirname, PASS_ARR(DAG_MEMO_NAME), info_buffer, DAG_MEMO_BYTESIZE)) { |
||||
|
goto fail_free; |
||||
|
} |
||||
|
|
||||
|
*data = temp_data_ptr; |
||||
|
*data_size = params->full_size; |
||||
|
return true; |
||||
|
|
||||
|
fail_free: |
||||
|
free(temp_data_ptr); |
||||
|
end: |
||||
|
return false; |
||||
|
} |
||||
|
|
||||
|
#undef PASS_ARR |
@ -0,0 +1,116 @@ |
|||||
|
/*
|
||||
|
This file is part of ethash. |
||||
|
|
||||
|
ethash 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 3 of the License, or |
||||
|
(at your option) any later version. |
||||
|
|
||||
|
ethash 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 ethash. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
*/ |
||||
|
/** @file io.h
|
||||
|
* @author Lefteris Karapetsas <lefteris@ethdev.com> |
||||
|
* @date 2015 |
||||
|
*/ |
||||
|
#pragma once |
||||
|
#include <stdlib.h> |
||||
|
#include <stdint.h> |
||||
|
#include <stdbool.h> |
||||
|
#include "ethash.h" |
||||
|
|
||||
|
#ifdef __cplusplus |
||||
|
extern "C" { |
||||
|
#endif |
||||
|
|
||||
|
typedef struct ethash_blockhash { uint8_t b[32]; } ethash_blockhash_t; |
||||
|
|
||||
|
static const char DAG_FILE_NAME[] = "full"; |
||||
|
static const char DAG_MEMO_NAME[] = "full.info"; |
||||
|
// MSVC thinks that "static const unsigned int" is not a compile time variable. Sorry for the #define :(
|
||||
|
#define DAG_MEMO_BYTESIZE 36 |
||||
|
|
||||
|
/// Possible return values of @see ethash_io_prepare
|
||||
|
enum ethash_io_rc { |
||||
|
ETHASH_IO_FAIL = 0, ///< There has been an IO failure
|
||||
|
ETHASH_IO_MEMO_MISMATCH, ///< Memo file either did not exist or there was content mismatch
|
||||
|
ETHASH_IO_MEMO_MATCH, ///< Memo file existed and contents matched. No need to do anything
|
||||
|
}; |
||||
|
|
||||
|
/**
|
||||
|
* Prepares io for ethash |
||||
|
* |
||||
|
* Create the DAG directory if it does not exist, and check if the memo file matches. |
||||
|
* If it does not match then it's deleted to pave the way for @ref ethash_io_write() |
||||
|
* |
||||
|
* @param dirname A null terminated c-string of the path of the ethash |
||||
|
* data directory. If it does not exist it's created. |
||||
|
* @param seedhash The seedhash of the current block number |
||||
|
* @return For possible return values @see enum ethash_io_rc |
||||
|
*/ |
||||
|
enum ethash_io_rc ethash_io_prepare(char const *dirname, ethash_blockhash_t seedhash); |
||||
|
/**
|
||||
|
* Fully computes data and writes it to the file on disk. |
||||
|
* |
||||
|
* This function should be called after @see ethash_io_prepare() and only if |
||||
|
* its return value is @c ETHASH_IO_MEMO_MISMATCH. Will write both the full data |
||||
|
* and the memo file. |
||||
|
* |
||||
|
* @param[in] dirname A null terminated c-string of the path of the ethash |
||||
|
* data directory. Has to exist. |
||||
|
* @param[in] params An ethash_params object containing the full size |
||||
|
* and the cache size |
||||
|
* @param[in] seedhash The seedhash of the current block number |
||||
|
* @param[in] cache The cache data. Would have usually been calulated by |
||||
|
* @see ethash_prep_light(). |
||||
|
* @param[out] data Pass a pointer to uint8_t by reference here. If the |
||||
|
* function is succesfull then this point to the allocated |
||||
|
* data calculated by @see ethash_prep_full(). Memory |
||||
|
* ownership is transfered to the callee. Remember that |
||||
|
* you eventually need to free this with a call to free(). |
||||
|
* @param[out] data_size Pass a size_t by value. If the function is succesfull |
||||
|
* then this will contain the number of bytes allocated |
||||
|
* for @a data. |
||||
|
* @return True for success and false in case of failure. |
||||
|
*/ |
||||
|
bool ethash_io_write(char const *dirname, |
||||
|
ethash_params const* params, |
||||
|
ethash_blockhash_t seedhash, |
||||
|
void const* cache, |
||||
|
uint8_t **data, |
||||
|
size_t *data_size); |
||||
|
|
||||
|
static inline void ethash_io_serialize_info(uint32_t revision, |
||||
|
ethash_blockhash_t seed_hash, |
||||
|
char *output) |
||||
|
{ |
||||
|
// if .info is only consumed locally we don't really care about endianess
|
||||
|
memcpy(output, &revision, 4); |
||||
|
memcpy(output + 4, &seed_hash, 32); |
||||
|
} |
||||
|
|
||||
|
static inline char *ethash_io_create_filename(char const *dirname, |
||||
|
char const* filename, |
||||
|
size_t filename_length) |
||||
|
{ |
||||
|
// in C the cast is not needed, but a C++ compiler will complain for invalid conversion
|
||||
|
char *name = (char*)malloc(strlen(dirname) + filename_length); |
||||
|
if (!name) { |
||||
|
return NULL; |
||||
|
} |
||||
|
|
||||
|
name[0] = '\0'; |
||||
|
strcat(name, dirname); |
||||
|
strcat(name, filename); |
||||
|
return name; |
||||
|
} |
||||
|
|
||||
|
|
||||
|
#ifdef __cplusplus |
||||
|
} |
||||
|
#endif |
@ -0,0 +1,76 @@ |
|||||
|
/*
|
||||
|
This file is part of ethash. |
||||
|
|
||||
|
ethash 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 3 of the License, or |
||||
|
(at your option) any later version. |
||||
|
|
||||
|
ethash 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 ethash. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
*/ |
||||
|
/** @file io_posix.c
|
||||
|
* @author Lefteris Karapetsas <lefteris@ethdev.com> |
||||
|
* @date 2015 |
||||
|
*/ |
||||
|
|
||||
|
#include "io.h" |
||||
|
#include <sys/types.h> |
||||
|
#include <sys/stat.h> |
||||
|
#include <errno.h> |
||||
|
#include <libgen.h> |
||||
|
#include <stdio.h> |
||||
|
#include <unistd.h> |
||||
|
|
||||
|
enum ethash_io_rc ethash_io_prepare(char const *dirname, ethash_blockhash_t seedhash) |
||||
|
{ |
||||
|
char read_buffer[DAG_MEMO_BYTESIZE]; |
||||
|
char expect_buffer[DAG_MEMO_BYTESIZE]; |
||||
|
enum ethash_io_rc ret = ETHASH_IO_FAIL; |
||||
|
|
||||
|
// assert directory exists, full owner permissions and read/search for others
|
||||
|
int rc = mkdir(dirname, S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH); |
||||
|
if (rc == -1 && errno != EEXIST) { |
||||
|
goto end; |
||||
|
} |
||||
|
|
||||
|
char *memofile = ethash_io_create_filename(dirname, DAG_MEMO_NAME, sizeof(DAG_MEMO_NAME)); |
||||
|
if (!memofile) { |
||||
|
goto end; |
||||
|
} |
||||
|
|
||||
|
// try to open memo file
|
||||
|
FILE *f = fopen(memofile, "rb"); |
||||
|
if (!f) { |
||||
|
// file does not exist, so no checking happens. All is fine.
|
||||
|
ret = ETHASH_IO_MEMO_MISMATCH; |
||||
|
goto free_memo; |
||||
|
} |
||||
|
|
||||
|
if (fread(read_buffer, 1, DAG_MEMO_BYTESIZE, f) != DAG_MEMO_BYTESIZE) { |
||||
|
goto close; |
||||
|
} |
||||
|
|
||||
|
ethash_io_serialize_info(REVISION, seedhash, expect_buffer); |
||||
|
if (memcmp(read_buffer, expect_buffer, DAG_MEMO_BYTESIZE) != 0) { |
||||
|
// we have different memo contents so delete the memo file
|
||||
|
if (unlink(memofile) != 0) { |
||||
|
goto close; |
||||
|
} |
||||
|
ret = ETHASH_IO_MEMO_MISMATCH; |
||||
|
} |
||||
|
|
||||
|
ret = ETHASH_IO_MEMO_MATCH; |
||||
|
|
||||
|
close: |
||||
|
fclose(f); |
||||
|
free_memo: |
||||
|
free(memofile); |
||||
|
end: |
||||
|
return ret; |
||||
|
} |
@ -0,0 +1,73 @@ |
|||||
|
/*
|
||||
|
This file is part of ethash. |
||||
|
|
||||
|
ethash 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 3 of the License, or |
||||
|
(at your option) any later version. |
||||
|
|
||||
|
ethash 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 ethash. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
*/ |
||||
|
/** @file io_win32.c
|
||||
|
* @author Lefteris Karapetsas <lefteris@ethdev.com> |
||||
|
* @date 2015 |
||||
|
*/ |
||||
|
|
||||
|
#include "io.h" |
||||
|
#include <direct.h> |
||||
|
#include <errno.h> |
||||
|
#include <stdio.h> |
||||
|
|
||||
|
enum ethash_io_rc ethash_io_prepare(char const *dirname, ethash_blockhash_t seedhash) |
||||
|
{ |
||||
|
char read_buffer[DAG_MEMO_BYTESIZE]; |
||||
|
char expect_buffer[DAG_MEMO_BYTESIZE]; |
||||
|
enum ethash_io_rc ret = ETHASH_IO_FAIL; |
||||
|
|
||||
|
// assert directory exists
|
||||
|
int rc = _mkdir(dirname); |
||||
|
if (rc == -1 && errno != EEXIST) { |
||||
|
goto end; |
||||
|
} |
||||
|
|
||||
|
char *memofile = ethash_io_create_filename(dirname, DAG_MEMO_NAME, sizeof(DAG_MEMO_NAME)); |
||||
|
if (!memofile) { |
||||
|
goto end; |
||||
|
} |
||||
|
|
||||
|
// try to open memo file
|
||||
|
FILE *f = fopen(memofile, "rb"); |
||||
|
if (!f) { |
||||
|
// file does not exist, so no checking happens. All is fine.
|
||||
|
ret = ETHASH_IO_MEMO_MISMATCH; |
||||
|
goto free_memo; |
||||
|
} |
||||
|
|
||||
|
if (fread(read_buffer, 1, DAG_MEMO_BYTESIZE, f) != DAG_MEMO_BYTESIZE) { |
||||
|
goto close; |
||||
|
} |
||||
|
|
||||
|
ethash_io_serialize_info(REVISION, seedhash, expect_buffer); |
||||
|
if (memcmp(read_buffer, expect_buffer, DAG_MEMO_BYTESIZE) != 0) { |
||||
|
// we have different memo contents so delete the memo file
|
||||
|
if (_unlink(memofile) != 0) { |
||||
|
goto close; |
||||
|
} |
||||
|
ret = ETHASH_IO_MEMO_MISMATCH; |
||||
|
} |
||||
|
|
||||
|
ret = ETHASH_IO_MEMO_MATCH; |
||||
|
|
||||
|
close: |
||||
|
fclose(f); |
||||
|
free_memo: |
||||
|
free(memofile); |
||||
|
end: |
||||
|
return ret; |
||||
|
} |
Loading…
Reference in new issue