mirror of
				https://git.wownero.com/wownero/wownero.git
				synced 2024-08-15 01:03:23 +00:00 
			
		
		
		
	extract some basic code from libcryptonote_core into libcryptonote_basic
This commit is contained in:
		
							parent
							
								
									99ee3fd17e
								
							
						
					
					
						commit
						8027ce0c75
					
				
					 104 changed files with 830 additions and 669 deletions
				
			
		| 
						 | 
				
			
			@ -194,6 +194,7 @@ endif()
 | 
			
		|||
if (BUILD_SHARED_LIBS)
 | 
			
		||||
  message(STATUS "Building internal libraries with position independent code")
 | 
			
		||||
  set(PIC_FLAG "-fPIC")
 | 
			
		||||
  add_definitions("-DBUILD_SHARED_LIBS")
 | 
			
		||||
else()
 | 
			
		||||
  message(STATUS "Building internal libraries as static")
 | 
			
		||||
endif()
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -27,6 +27,7 @@
 | 
			
		|||
# THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
			
		||||
 | 
			
		||||
add_library(epee STATIC http_auth.cpp mlog.cpp)
 | 
			
		||||
 | 
			
		||||
# Build and install libepee if we're building for GUI
 | 
			
		||||
if (BUILD_GUI_DEPS)
 | 
			
		||||
    if(IOS)
 | 
			
		||||
| 
						 | 
				
			
			@ -36,4 +37,11 @@ if (BUILD_GUI_DEPS)
 | 
			
		|||
    endif()
 | 
			
		||||
    install(TARGETS epee
 | 
			
		||||
        ARCHIVE DESTINATION ${lib_folder})
 | 
			
		||||
endif()
 | 
			
		||||
endif()
 | 
			
		||||
 | 
			
		||||
target_link_libraries(epee
 | 
			
		||||
  PUBLIC
 | 
			
		||||
    crypto
 | 
			
		||||
    ${Boost_FILESYSTEM_LIBRARY}
 | 
			
		||||
  PRIVATE
 | 
			
		||||
    ${EXTRA_LIBRARIES})
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -99,6 +99,7 @@ endfunction ()
 | 
			
		|||
add_subdirectory(common)
 | 
			
		||||
add_subdirectory(crypto)
 | 
			
		||||
add_subdirectory(ringct)
 | 
			
		||||
add_subdirectory(cryptonote_basic)
 | 
			
		||||
add_subdirectory(cryptonote_core)
 | 
			
		||||
add_subdirectory(blockchain_db)
 | 
			
		||||
add_subdirectory(mnemonics)
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -63,6 +63,7 @@ target_link_libraries(blockchain_db
 | 
			
		|||
  PUBLIC
 | 
			
		||||
    common
 | 
			
		||||
    crypto
 | 
			
		||||
    ringct
 | 
			
		||||
    ${LMDB_LIBRARY}
 | 
			
		||||
    ${BDB_LIBRARY}
 | 
			
		||||
    ${Boost_FILESYSTEM_LIBRARY}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -31,7 +31,7 @@
 | 
			
		|||
#include <memory>  // std::unique_ptr
 | 
			
		||||
#include <cstring>  // memcpy
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "profile_tools.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -29,7 +29,7 @@
 | 
			
		|||
#include <boost/range/adaptor/reversed.hpp>
 | 
			
		||||
 | 
			
		||||
#include "blockchain_db.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "profile_tools.h"
 | 
			
		||||
 | 
			
		||||
#undef MONERO_DEFAULT_LOG_CATEGORY
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -34,9 +34,9 @@
 | 
			
		|||
#include <string>
 | 
			
		||||
#include <exception>
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/difficulty.h"
 | 
			
		||||
#include "cryptonote_core/hardfork.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/difficulty.h"
 | 
			
		||||
#include "cryptonote_basic/hardfork.h"
 | 
			
		||||
 | 
			
		||||
/** \file
 | 
			
		||||
 * Cryptonote Blockchain Database Interface
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -34,7 +34,7 @@
 | 
			
		|||
#include <cstring>  // memcpy
 | 
			
		||||
#include <random>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "profile_tools.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -34,7 +34,7 @@
 | 
			
		|||
#include <boost/filesystem.hpp>
 | 
			
		||||
#include "bootstrap_file.h"
 | 
			
		||||
#include "bootstrap_serialization.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "serialization/binary_utils.h" // dump_binary(), parse_binary()
 | 
			
		||||
#include "serialization/json_utils.h" // dump_json()
 | 
			
		||||
#include "include_base_utils.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -34,8 +34,8 @@
 | 
			
		|||
 | 
			
		||||
#include <boost/iostreams/filtering_streambuf.hpp>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_core/blockchain.h"
 | 
			
		||||
#include "blockchain_db/blockchain_db.h"
 | 
			
		||||
#include "blockchain_db/lmdb/db_lmdb.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -34,7 +34,7 @@
 | 
			
		|||
 | 
			
		||||
#include <boost/iostreams/filtering_streambuf.hpp>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/blockchain.h"
 | 
			
		||||
 | 
			
		||||
#include <algorithm>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -28,8 +28,8 @@
 | 
			
		|||
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_core/difficulty.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_basic/difficulty.h"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
namespace cryptonote
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -26,8 +26,8 @@
 | 
			
		|||
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
 | 
			
		||||
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/tx_extra.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/tx_extra.h"
 | 
			
		||||
#include "cryptonote_core/blockchain.h"
 | 
			
		||||
#include "blockchain_utilities.h"
 | 
			
		||||
#include "common/command_line.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -2,12 +2,19 @@
 | 
			
		|||
 | 
			
		||||
#if defined(__APPLE__)
 | 
			
		||||
#include <mach-o/getsect.h>
 | 
			
		||||
 | 
			
		||||
#ifdef BUILD_SHARED_LIBS
 | 
			
		||||
#if !defined(__LP64__)
 | 
			
		||||
const struct mach_header _mh_execute_header;
 | 
			
		||||
#else
 | 
			
		||||
const struct mach_header_64 _mh_execute_header;
 | 
			
		||||
#endif
 | 
			
		||||
#else
 | 
			
		||||
#if !defined(__LP64__)
 | 
			
		||||
extern const struct mach_header _mh_execute_header;
 | 
			
		||||
#else
 | 
			
		||||
extern const struct mach_header_64 _mh_execute_header;
 | 
			
		||||
#endif
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
const unsigned char *get_blocks_dat_start(int testnet)
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -69,6 +69,7 @@ monero_add_library(common
 | 
			
		|||
  ${common_private_headers})
 | 
			
		||||
target_link_libraries(common
 | 
			
		||||
  PUBLIC
 | 
			
		||||
    epee
 | 
			
		||||
    crypto
 | 
			
		||||
    ${UNBOUND_LIBRARY}
 | 
			
		||||
    ${LIBUNWIND_LIBRARIES}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -29,7 +29,7 @@
 | 
			
		|||
#include "common/command_line.h"
 | 
			
		||||
#include "common/i18n.h"
 | 
			
		||||
#include "common/dns_utils.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include <cstring>
 | 
			
		||||
#include <sstream>
 | 
			
		||||
// check local first (in the event of static or in-source compilation of libunbound)
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -29,7 +29,7 @@
 | 
			
		|||
 | 
			
		||||
#include <vector>
 | 
			
		||||
#include <string>
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
 | 
			
		||||
namespace tools
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										73
									
								
								src/cryptonote_basic/CMakeLists.txt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								src/cryptonote_basic/CMakeLists.txt
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,73 @@
 | 
			
		|||
# Copyright (c) 2014-2016, The Monero Project
 | 
			
		||||
#
 | 
			
		||||
# All rights reserved.
 | 
			
		||||
#
 | 
			
		||||
# Redistribution and use in source and binary forms, with or without modification, are
 | 
			
		||||
# permitted provided that the following conditions are met:
 | 
			
		||||
#
 | 
			
		||||
# 1. Redistributions of source code must retain the above copyright notice, this list of
 | 
			
		||||
#    conditions and the following disclaimer.
 | 
			
		||||
#
 | 
			
		||||
# 2. Redistributions in binary form must reproduce the above copyright notice, this list
 | 
			
		||||
#    of conditions and the following disclaimer in the documentation and/or other
 | 
			
		||||
#    materials provided with the distribution.
 | 
			
		||||
#
 | 
			
		||||
# 3. Neither the name of the copyright holder nor the names of its contributors may be
 | 
			
		||||
#    used to endorse or promote products derived from this software without specific
 | 
			
		||||
#    prior written permission.
 | 
			
		||||
#
 | 
			
		||||
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
 | 
			
		||||
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
 | 
			
		||||
# MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
 | 
			
		||||
# THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 | 
			
		||||
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 | 
			
		||||
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 | 
			
		||||
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 | 
			
		||||
# STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
 | 
			
		||||
# THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
			
		||||
 | 
			
		||||
set(cryptonote_basic_sources
 | 
			
		||||
  account.cpp
 | 
			
		||||
  checkpoints.cpp
 | 
			
		||||
  cryptonote_basic_impl.cpp
 | 
			
		||||
  cryptonote_format_utils.cpp
 | 
			
		||||
  difficulty.cpp
 | 
			
		||||
  miner.cpp
 | 
			
		||||
  hardfork.cpp)
 | 
			
		||||
 | 
			
		||||
set(cryptonote_basic_headers)
 | 
			
		||||
 | 
			
		||||
set(cryptonote_basic_private_headers
 | 
			
		||||
  account.h
 | 
			
		||||
  account_boost_serialization.h
 | 
			
		||||
  checkpoints.h
 | 
			
		||||
  connection_context.h
 | 
			
		||||
  cryptonote_basic.h
 | 
			
		||||
  cryptonote_basic_impl.h
 | 
			
		||||
  cryptonote_boost_serialization.h
 | 
			
		||||
  cryptonote_format_utils.h
 | 
			
		||||
  cryptonote_stat_info.h
 | 
			
		||||
  difficulty.h
 | 
			
		||||
  miner.h
 | 
			
		||||
  tx_extra.h
 | 
			
		||||
  verification_context.h
 | 
			
		||||
  hardfork.h)
 | 
			
		||||
 | 
			
		||||
monero_private_headers(cryptonote_basic
 | 
			
		||||
  ${crypto_private_headers})
 | 
			
		||||
monero_add_library(cryptonote_basic
 | 
			
		||||
  ${cryptonote_basic_sources}
 | 
			
		||||
  ${cryptonote_basic_headers}
 | 
			
		||||
  ${cryptonote_basic_private_headers})
 | 
			
		||||
target_link_libraries(cryptonote_basic
 | 
			
		||||
  PUBLIC
 | 
			
		||||
    common
 | 
			
		||||
    crypto
 | 
			
		||||
    ${Boost_DATE_TIME_LIBRARY}
 | 
			
		||||
    ${Boost_PROGRAM_OPTIONS_LIBRARY}
 | 
			
		||||
    ${Boost_SERIALIZATION_LIBRARY}
 | 
			
		||||
    ${Boost_FILESYSTEM_LIBRARY}
 | 
			
		||||
    ${Boost_SYSTEM_LIBRARY}
 | 
			
		||||
    ${Boost_THREAD_LIBRARY}
 | 
			
		||||
  PRIVATE
 | 
			
		||||
    ${EXTRA_LIBRARIES})
 | 
			
		||||
| 
						 | 
				
			
			@ -38,8 +38,8 @@ extern "C"
 | 
			
		|||
{
 | 
			
		||||
#include "crypto/keccak.h"
 | 
			
		||||
}
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_format_utils.h"
 | 
			
		||||
 | 
			
		||||
#undef MONERO_DEFAULT_LOG_CATEGORY
 | 
			
		||||
#define MONERO_DEFAULT_LOG_CATEGORY "account"
 | 
			
		||||
| 
						 | 
				
			
			@ -30,7 +30,7 @@
 | 
			
		|||
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic.h"
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "serialization/keyvalue_serialization.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -31,7 +31,7 @@
 | 
			
		|||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_boost_serialization.h"
 | 
			
		||||
 | 
			
		||||
//namespace cryptonote {
 | 
			
		||||
namespace boost
 | 
			
		||||
| 
						 | 
				
			
			@ -33,7 +33,6 @@ using namespace epee;
 | 
			
		|||
 | 
			
		||||
#include "cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_config.h"
 | 
			
		||||
#include "miner.h"
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "ringct/rctSigs.h"
 | 
			
		||||
| 
						 | 
				
			
			@ -108,102 +107,6 @@ namespace cryptonote
 | 
			
		|||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool construct_miner_tx(size_t height, size_t median_size, uint64_t already_generated_coins, size_t current_block_size, uint64_t fee, const account_public_address &miner_address, transaction& tx, const blobdata& extra_nonce, size_t max_outs, uint8_t hard_fork_version) {
 | 
			
		||||
    tx.vin.clear();
 | 
			
		||||
    tx.vout.clear();
 | 
			
		||||
    tx.extra.clear();
 | 
			
		||||
 | 
			
		||||
    keypair txkey = keypair::generate();
 | 
			
		||||
    add_tx_pub_key_to_extra(tx, txkey.pub);
 | 
			
		||||
    if(!extra_nonce.empty())
 | 
			
		||||
      if(!add_extra_nonce_to_tx_extra(tx.extra, extra_nonce))
 | 
			
		||||
        return false;
 | 
			
		||||
 | 
			
		||||
    txin_gen in;
 | 
			
		||||
    in.height = height;
 | 
			
		||||
 | 
			
		||||
    uint64_t block_reward;
 | 
			
		||||
    if(!get_block_reward(median_size, current_block_size, already_generated_coins, block_reward, hard_fork_version))
 | 
			
		||||
    {
 | 
			
		||||
      LOG_PRINT_L0("Block is too big");
 | 
			
		||||
      return false;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
#if defined(DEBUG_CREATE_BLOCK_TEMPLATE)
 | 
			
		||||
    LOG_PRINT_L1("Creating block template: reward " << block_reward <<
 | 
			
		||||
      ", fee " << fee);
 | 
			
		||||
#endif
 | 
			
		||||
    block_reward += fee;
 | 
			
		||||
 | 
			
		||||
    // from hard fork 2, we cut out the low significant digits. This makes the tx smaller, and
 | 
			
		||||
    // keeps the paid amount almost the same. The unpaid remainder gets pushed back to the
 | 
			
		||||
    // emission schedule
 | 
			
		||||
    // from hard fork 4, we use a single "dusty" output. This makes the tx even smaller,
 | 
			
		||||
    // and avoids the quantization. These outputs will be added as rct outputs with identity
 | 
			
		||||
    // masks, to they can be used as rct inputs.
 | 
			
		||||
    if (hard_fork_version >= 2 && hard_fork_version < 4) {
 | 
			
		||||
      block_reward = block_reward - block_reward % ::config::BASE_REWARD_CLAMP_THRESHOLD;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    std::vector<uint64_t> out_amounts;
 | 
			
		||||
    decompose_amount_into_digits(block_reward, hard_fork_version >= 2 ? 0 : ::config::DEFAULT_DUST_THRESHOLD,
 | 
			
		||||
      [&out_amounts](uint64_t a_chunk) { out_amounts.push_back(a_chunk); },
 | 
			
		||||
      [&out_amounts](uint64_t a_dust) { out_amounts.push_back(a_dust); });
 | 
			
		||||
 | 
			
		||||
    CHECK_AND_ASSERT_MES(1 <= max_outs, false, "max_out must be non-zero");
 | 
			
		||||
    if (height == 0 || hard_fork_version >= 4)
 | 
			
		||||
    {
 | 
			
		||||
      // the genesis block was not decomposed, for unknown reasons
 | 
			
		||||
      while (max_outs < out_amounts.size())
 | 
			
		||||
      {
 | 
			
		||||
        //out_amounts[out_amounts.size() - 2] += out_amounts.back();
 | 
			
		||||
        //out_amounts.resize(out_amounts.size() - 1);
 | 
			
		||||
        out_amounts[1] += out_amounts[0];
 | 
			
		||||
        for (size_t n = 1; n < out_amounts.size(); ++n)
 | 
			
		||||
          out_amounts[n - 1] = out_amounts[n];
 | 
			
		||||
        out_amounts.resize(out_amounts.size() - 1);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    else
 | 
			
		||||
    {
 | 
			
		||||
      CHECK_AND_ASSERT_MES(max_outs >= out_amounts.size(), false, "max_out exceeded");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    uint64_t summary_amounts = 0;
 | 
			
		||||
    for (size_t no = 0; no < out_amounts.size(); no++)
 | 
			
		||||
    {
 | 
			
		||||
      crypto::key_derivation derivation = AUTO_VAL_INIT(derivation);;
 | 
			
		||||
      crypto::public_key out_eph_public_key = AUTO_VAL_INIT(out_eph_public_key);
 | 
			
		||||
      bool r = crypto::generate_key_derivation(miner_address.m_view_public_key, txkey.sec, derivation);
 | 
			
		||||
      CHECK_AND_ASSERT_MES(r, false, "while creating outs: failed to generate_key_derivation(" << miner_address.m_view_public_key << ", " << txkey.sec << ")");
 | 
			
		||||
 | 
			
		||||
      r = crypto::derive_public_key(derivation, no, miner_address.m_spend_public_key, out_eph_public_key);
 | 
			
		||||
      CHECK_AND_ASSERT_MES(r, false, "while creating outs: failed to derive_public_key(" << derivation << ", " << no << ", "<< miner_address.m_spend_public_key << ")");
 | 
			
		||||
 | 
			
		||||
      txout_to_key tk;
 | 
			
		||||
      tk.key = out_eph_public_key;
 | 
			
		||||
 | 
			
		||||
      tx_out out;
 | 
			
		||||
      summary_amounts += out.amount = out_amounts[no];
 | 
			
		||||
      out.target = tk;
 | 
			
		||||
      tx.vout.push_back(out);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    CHECK_AND_ASSERT_MES(summary_amounts == block_reward, false, "Failed to construct miner tx, summary_amounts = " << summary_amounts << " not equal block_reward = " << block_reward);
 | 
			
		||||
 | 
			
		||||
    if (hard_fork_version >= 4)
 | 
			
		||||
      tx.version = 2;
 | 
			
		||||
    else
 | 
			
		||||
      tx.version = 1;
 | 
			
		||||
 | 
			
		||||
    //lock
 | 
			
		||||
    tx.unlock_time = height + CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW;
 | 
			
		||||
    tx.vin.push_back(in);
 | 
			
		||||
    //LOG_PRINT("MINER_TX generated ok, block_reward=" << print_money(block_reward) << "("  << print_money(block_reward - fee) << "+" << print_money(fee)
 | 
			
		||||
    //  << "), current_block_size=" << current_block_size << ", already_generated_coins=" << already_generated_coins << ", tx_id=" << get_transaction_hash(tx), LOG_LEVEL_2);
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool generate_key_image_helper(const account_keys& ack, const crypto::public_key& tx_public_key, size_t real_output_index, keypair& in_ephemeral, crypto::key_image& ki)
 | 
			
		||||
  {
 | 
			
		||||
    crypto::key_derivation recv_derivation = AUTO_VAL_INIT(recv_derivation);
 | 
			
		||||
| 
						 | 
				
			
			@ -435,22 +338,6 @@ namespace cryptonote
 | 
			
		|||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  crypto::public_key get_destination_view_key_pub(const std::vector<tx_destination_entry> &destinations, const account_keys &sender_keys)
 | 
			
		||||
  {
 | 
			
		||||
    if (destinations.empty())
 | 
			
		||||
      return null_pkey;
 | 
			
		||||
    for (size_t n = 1; n < destinations.size(); ++n)
 | 
			
		||||
    {
 | 
			
		||||
      if (!memcmp(&destinations[n].addr, &sender_keys.m_account_address, sizeof(destinations[0].addr)))
 | 
			
		||||
        continue;
 | 
			
		||||
      if (destinations[n].amount == 0)
 | 
			
		||||
        continue;
 | 
			
		||||
      if (memcmp(&destinations[n].addr, &destinations[0].addr, sizeof(destinations[0].addr)))
 | 
			
		||||
        return null_pkey;
 | 
			
		||||
    }
 | 
			
		||||
    return destinations[0].addr.m_view_public_key;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool encrypt_payment_id(crypto::hash8 &payment_id, const crypto::public_key &public_key, const crypto::secret_key &secret_key)
 | 
			
		||||
  {
 | 
			
		||||
    crypto::key_derivation derivation;
 | 
			
		||||
| 
						 | 
				
			
			@ -475,313 +362,6 @@ namespace cryptonote
 | 
			
		|||
    return encrypt_payment_id(payment_id, public_key, secret_key);
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool construct_tx_and_get_tx_key(const account_keys& sender_account_keys, const std::vector<tx_source_entry>& sources, const std::vector<tx_destination_entry>& destinations, std::vector<uint8_t> extra, transaction& tx, uint64_t unlock_time, crypto::secret_key &tx_key, bool rct)
 | 
			
		||||
  {
 | 
			
		||||
    std::vector<rct::key> amount_keys;
 | 
			
		||||
    tx.set_null();
 | 
			
		||||
    amount_keys.clear();
 | 
			
		||||
 | 
			
		||||
    tx.version = rct ? 2 : 1;
 | 
			
		||||
    tx.unlock_time = unlock_time;
 | 
			
		||||
 | 
			
		||||
    tx.extra = extra;
 | 
			
		||||
    keypair txkey = keypair::generate();
 | 
			
		||||
    remove_field_from_tx_extra(tx.extra, typeid(tx_extra_pub_key));
 | 
			
		||||
    add_tx_pub_key_to_extra(tx, txkey.pub);
 | 
			
		||||
    tx_key = txkey.sec;
 | 
			
		||||
 | 
			
		||||
    // if we have a stealth payment id, find it and encrypt it with the tx key now
 | 
			
		||||
    std::vector<tx_extra_field> tx_extra_fields;
 | 
			
		||||
    if (parse_tx_extra(tx.extra, tx_extra_fields))
 | 
			
		||||
    {
 | 
			
		||||
      tx_extra_nonce extra_nonce;
 | 
			
		||||
      if (find_tx_extra_field_by_type(tx_extra_fields, extra_nonce))
 | 
			
		||||
      {
 | 
			
		||||
        crypto::hash8 payment_id = null_hash8;
 | 
			
		||||
        if (get_encrypted_payment_id_from_tx_extra_nonce(extra_nonce.nonce, payment_id))
 | 
			
		||||
        {
 | 
			
		||||
          LOG_PRINT_L2("Encrypting payment id " << payment_id);
 | 
			
		||||
          crypto::public_key view_key_pub = get_destination_view_key_pub(destinations, sender_account_keys);
 | 
			
		||||
          if (view_key_pub == null_pkey)
 | 
			
		||||
          {
 | 
			
		||||
            LOG_ERROR("Destinations have to have exactly one output to support encrypted payment ids");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          if (!encrypt_payment_id(payment_id, view_key_pub, txkey.sec))
 | 
			
		||||
          {
 | 
			
		||||
            LOG_ERROR("Failed to encrypt payment id");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          std::string extra_nonce;
 | 
			
		||||
          set_encrypted_payment_id_to_tx_extra_nonce(extra_nonce, payment_id);
 | 
			
		||||
          remove_field_from_tx_extra(tx.extra, typeid(tx_extra_nonce));
 | 
			
		||||
          if (!add_extra_nonce_to_tx_extra(tx.extra, extra_nonce))
 | 
			
		||||
          {
 | 
			
		||||
            LOG_ERROR("Failed to add encrypted payment id to tx extra");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
          LOG_PRINT_L1("Encrypted payment ID: " << payment_id);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    else
 | 
			
		||||
    {
 | 
			
		||||
      LOG_ERROR("Failed to parse tx extra");
 | 
			
		||||
      return false;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    struct input_generation_context_data
 | 
			
		||||
    {
 | 
			
		||||
      keypair in_ephemeral;
 | 
			
		||||
    };
 | 
			
		||||
    std::vector<input_generation_context_data> in_contexts;
 | 
			
		||||
 | 
			
		||||
    uint64_t summary_inputs_money = 0;
 | 
			
		||||
    //fill inputs
 | 
			
		||||
    int idx = -1;
 | 
			
		||||
    for(const tx_source_entry& src_entr:  sources)
 | 
			
		||||
    {
 | 
			
		||||
      ++idx;
 | 
			
		||||
      if(src_entr.real_output >= src_entr.outputs.size())
 | 
			
		||||
      {
 | 
			
		||||
        LOG_ERROR("real_output index (" << src_entr.real_output << ")bigger than output_keys.size()=" << src_entr.outputs.size());
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
      summary_inputs_money += src_entr.amount;
 | 
			
		||||
 | 
			
		||||
      //key_derivation recv_derivation;
 | 
			
		||||
      in_contexts.push_back(input_generation_context_data());
 | 
			
		||||
      keypair& in_ephemeral = in_contexts.back().in_ephemeral;
 | 
			
		||||
      crypto::key_image img;
 | 
			
		||||
      if(!generate_key_image_helper(sender_account_keys, src_entr.real_out_tx_key, src_entr.real_output_in_tx_index, in_ephemeral, img))
 | 
			
		||||
        return false;
 | 
			
		||||
 | 
			
		||||
      //check that derivated key is equal with real output key
 | 
			
		||||
      if( !(in_ephemeral.pub == src_entr.outputs[src_entr.real_output].second.dest) )
 | 
			
		||||
      {
 | 
			
		||||
        LOG_ERROR("derived public key mismatch with output public key at index " << idx << ", real out " << src_entr.real_output << "! "<< ENDL << "derived_key:"
 | 
			
		||||
          << string_tools::pod_to_hex(in_ephemeral.pub) << ENDL << "real output_public_key:"
 | 
			
		||||
          << string_tools::pod_to_hex(src_entr.outputs[src_entr.real_output].second) );
 | 
			
		||||
        LOG_ERROR("amount " << src_entr.amount << ", rct " << src_entr.rct);
 | 
			
		||||
        LOG_ERROR("tx pubkey " << src_entr.real_out_tx_key << ", real_output_in_tx_index " << src_entr.real_output_in_tx_index);
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      //put key image into tx input
 | 
			
		||||
      txin_to_key input_to_key;
 | 
			
		||||
      input_to_key.amount = src_entr.amount;
 | 
			
		||||
      input_to_key.k_image = img;
 | 
			
		||||
 | 
			
		||||
      //fill outputs array and use relative offsets
 | 
			
		||||
      for(const tx_source_entry::output_entry& out_entry: src_entr.outputs)
 | 
			
		||||
        input_to_key.key_offsets.push_back(out_entry.first);
 | 
			
		||||
 | 
			
		||||
      input_to_key.key_offsets = absolute_output_offsets_to_relative(input_to_key.key_offsets);
 | 
			
		||||
      tx.vin.push_back(input_to_key);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // "Shuffle" outs
 | 
			
		||||
    std::vector<tx_destination_entry> shuffled_dsts(destinations);
 | 
			
		||||
    std::sort(shuffled_dsts.begin(), shuffled_dsts.end(), [](const tx_destination_entry& de1, const tx_destination_entry& de2) { return de1.amount < de2.amount; } );
 | 
			
		||||
 | 
			
		||||
    uint64_t summary_outs_money = 0;
 | 
			
		||||
    //fill outputs
 | 
			
		||||
    size_t output_index = 0;
 | 
			
		||||
    for(const tx_destination_entry& dst_entr:  shuffled_dsts)
 | 
			
		||||
    {
 | 
			
		||||
      CHECK_AND_ASSERT_MES(dst_entr.amount > 0 || tx.version > 1, false, "Destination with wrong amount: " << dst_entr.amount);
 | 
			
		||||
      crypto::key_derivation derivation;
 | 
			
		||||
      crypto::public_key out_eph_public_key;
 | 
			
		||||
      bool r = crypto::generate_key_derivation(dst_entr.addr.m_view_public_key, txkey.sec, derivation);
 | 
			
		||||
      CHECK_AND_ASSERT_MES(r, false, "at creation outs: failed to generate_key_derivation(" << dst_entr.addr.m_view_public_key << ", " << txkey.sec << ")");
 | 
			
		||||
 | 
			
		||||
      if (tx.version > 1)
 | 
			
		||||
      {
 | 
			
		||||
        crypto::secret_key scalar1;
 | 
			
		||||
        crypto::derivation_to_scalar(derivation, output_index, scalar1);
 | 
			
		||||
        amount_keys.push_back(rct::sk2rct(scalar1));
 | 
			
		||||
      }
 | 
			
		||||
      r = crypto::derive_public_key(derivation, output_index, dst_entr.addr.m_spend_public_key, out_eph_public_key);
 | 
			
		||||
      CHECK_AND_ASSERT_MES(r, false, "at creation outs: failed to derive_public_key(" << derivation << ", " << output_index << ", "<< dst_entr.addr.m_spend_public_key << ")");
 | 
			
		||||
 | 
			
		||||
      tx_out out;
 | 
			
		||||
      out.amount = dst_entr.amount;
 | 
			
		||||
      txout_to_key tk;
 | 
			
		||||
      tk.key = out_eph_public_key;
 | 
			
		||||
      out.target = tk;
 | 
			
		||||
      tx.vout.push_back(out);
 | 
			
		||||
      output_index++;
 | 
			
		||||
      summary_outs_money += dst_entr.amount;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    //check money
 | 
			
		||||
    if(summary_outs_money > summary_inputs_money )
 | 
			
		||||
    {
 | 
			
		||||
      LOG_ERROR("Transaction inputs money ("<< summary_inputs_money << ") less than outputs money (" << summary_outs_money << ")");
 | 
			
		||||
      return false;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // check for watch only wallet
 | 
			
		||||
    bool zero_secret_key = true;
 | 
			
		||||
    for (size_t i = 0; i < sizeof(sender_account_keys.m_spend_secret_key); ++i)
 | 
			
		||||
      zero_secret_key &= (sender_account_keys.m_spend_secret_key.data[i] == 0);
 | 
			
		||||
    if (zero_secret_key)
 | 
			
		||||
    {
 | 
			
		||||
      MDEBUG("Null secret key, skipping signatures");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (tx.version == 1)
 | 
			
		||||
    {
 | 
			
		||||
      //generate ring signatures
 | 
			
		||||
      crypto::hash tx_prefix_hash;
 | 
			
		||||
      get_transaction_prefix_hash(tx, tx_prefix_hash);
 | 
			
		||||
 | 
			
		||||
      std::stringstream ss_ring_s;
 | 
			
		||||
      size_t i = 0;
 | 
			
		||||
      for(const tx_source_entry& src_entr:  sources)
 | 
			
		||||
      {
 | 
			
		||||
        ss_ring_s << "pub_keys:" << ENDL;
 | 
			
		||||
        std::vector<const crypto::public_key*> keys_ptrs;
 | 
			
		||||
        std::vector<crypto::public_key> keys(src_entr.outputs.size());
 | 
			
		||||
        size_t ii = 0;
 | 
			
		||||
        for(const tx_source_entry::output_entry& o: src_entr.outputs)
 | 
			
		||||
        {
 | 
			
		||||
          keys[ii] = rct2pk(o.second.dest);
 | 
			
		||||
          keys_ptrs.push_back(&keys[ii]);
 | 
			
		||||
          ss_ring_s << o.second.dest << ENDL;
 | 
			
		||||
          ++ii;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        tx.signatures.push_back(std::vector<crypto::signature>());
 | 
			
		||||
        std::vector<crypto::signature>& sigs = tx.signatures.back();
 | 
			
		||||
        sigs.resize(src_entr.outputs.size());
 | 
			
		||||
        if (!zero_secret_key)
 | 
			
		||||
          crypto::generate_ring_signature(tx_prefix_hash, boost::get<txin_to_key>(tx.vin[i]).k_image, keys_ptrs, in_contexts[i].in_ephemeral.sec, src_entr.real_output, sigs.data());
 | 
			
		||||
        ss_ring_s << "signatures:" << ENDL;
 | 
			
		||||
        std::for_each(sigs.begin(), sigs.end(), [&](const crypto::signature& s){ss_ring_s << s << ENDL;});
 | 
			
		||||
        ss_ring_s << "prefix_hash:" << tx_prefix_hash << ENDL << "in_ephemeral_key: " << in_contexts[i].in_ephemeral.sec << ENDL << "real_output: " << src_entr.real_output;
 | 
			
		||||
        i++;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      MCINFO("construct_tx", "transaction_created: " << get_transaction_hash(tx) << ENDL << obj_to_json_str(tx) << ENDL << ss_ring_s.str());
 | 
			
		||||
    }
 | 
			
		||||
    else
 | 
			
		||||
    {
 | 
			
		||||
      size_t n_total_outs = sources[0].outputs.size(); // only for non-simple rct
 | 
			
		||||
 | 
			
		||||
      // the non-simple version is slightly smaller, but assumes all real inputs
 | 
			
		||||
      // are on the same index, so can only be used if there just one ring.
 | 
			
		||||
      bool use_simple_rct = sources.size() > 1;
 | 
			
		||||
 | 
			
		||||
      if (!use_simple_rct)
 | 
			
		||||
      {
 | 
			
		||||
        // non simple ringct requires all real inputs to be at the same index for all inputs
 | 
			
		||||
        for(const tx_source_entry& src_entr:  sources)
 | 
			
		||||
        {
 | 
			
		||||
          if(src_entr.real_output != sources.begin()->real_output)
 | 
			
		||||
          {
 | 
			
		||||
            LOG_ERROR("All inputs must have the same index for non-simple ringct");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // enforce same mixin for all outputs
 | 
			
		||||
        for (size_t i = 1; i < sources.size(); ++i) {
 | 
			
		||||
          if (n_total_outs != sources[i].outputs.size()) {
 | 
			
		||||
            LOG_ERROR("Non-simple ringct transaction has varying mixin");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      uint64_t amount_in = 0, amount_out = 0;
 | 
			
		||||
      rct::ctkeyV inSk;
 | 
			
		||||
      // mixRing indexing is done the other way round for simple
 | 
			
		||||
      rct::ctkeyM mixRing(use_simple_rct ? sources.size() : n_total_outs);
 | 
			
		||||
      rct::keyV destinations;
 | 
			
		||||
      std::vector<uint64_t> inamounts, outamounts;
 | 
			
		||||
      std::vector<unsigned int> index;
 | 
			
		||||
      for (size_t i = 0; i < sources.size(); ++i)
 | 
			
		||||
      {
 | 
			
		||||
        rct::ctkey ctkey;
 | 
			
		||||
        amount_in += sources[i].amount;
 | 
			
		||||
        inamounts.push_back(sources[i].amount);
 | 
			
		||||
        index.push_back(sources[i].real_output);
 | 
			
		||||
        // inSk: (secret key, mask)
 | 
			
		||||
        ctkey.dest = rct::sk2rct(in_contexts[i].in_ephemeral.sec);
 | 
			
		||||
        ctkey.mask = sources[i].mask;
 | 
			
		||||
        inSk.push_back(ctkey);
 | 
			
		||||
        // inPk: (public key, commitment)
 | 
			
		||||
        // will be done when filling in mixRing
 | 
			
		||||
      }
 | 
			
		||||
      for (size_t i = 0; i < tx.vout.size(); ++i)
 | 
			
		||||
      {
 | 
			
		||||
        destinations.push_back(rct::pk2rct(boost::get<txout_to_key>(tx.vout[i].target).key));
 | 
			
		||||
        outamounts.push_back(tx.vout[i].amount);
 | 
			
		||||
        amount_out += tx.vout[i].amount;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (use_simple_rct)
 | 
			
		||||
      {
 | 
			
		||||
        // mixRing indexing is done the other way round for simple
 | 
			
		||||
        for (size_t i = 0; i < sources.size(); ++i)
 | 
			
		||||
        {
 | 
			
		||||
          mixRing[i].resize(sources[i].outputs.size());
 | 
			
		||||
          for (size_t n = 0; n < sources[i].outputs.size(); ++n)
 | 
			
		||||
          {
 | 
			
		||||
            mixRing[i][n] = sources[i].outputs[n].second;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      else
 | 
			
		||||
      {
 | 
			
		||||
        for (size_t i = 0; i < n_total_outs; ++i) // same index assumption
 | 
			
		||||
        {
 | 
			
		||||
          mixRing[i].resize(sources.size());
 | 
			
		||||
          for (size_t n = 0; n < sources.size(); ++n)
 | 
			
		||||
          {
 | 
			
		||||
            mixRing[i][n] = sources[n].outputs[i].second;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // fee
 | 
			
		||||
      if (!use_simple_rct && amount_in > amount_out)
 | 
			
		||||
        outamounts.push_back(amount_in - amount_out);
 | 
			
		||||
 | 
			
		||||
      // zero out all amounts to mask rct outputs, real amounts are now encrypted
 | 
			
		||||
      for (size_t i = 0; i < tx.vin.size(); ++i)
 | 
			
		||||
      {
 | 
			
		||||
        if (sources[i].rct)
 | 
			
		||||
          boost::get<txin_to_key>(tx.vin[i]).amount = 0;
 | 
			
		||||
      }
 | 
			
		||||
      for (size_t i = 0; i < tx.vout.size(); ++i)
 | 
			
		||||
        tx.vout[i].amount = 0;
 | 
			
		||||
 | 
			
		||||
      crypto::hash tx_prefix_hash;
 | 
			
		||||
      get_transaction_prefix_hash(tx, tx_prefix_hash);
 | 
			
		||||
      rct::ctkeyV outSk;
 | 
			
		||||
      if (use_simple_rct)
 | 
			
		||||
        tx.rct_signatures = rct::genRctSimple(rct::hash2rct(tx_prefix_hash), inSk, destinations, inamounts, outamounts, amount_in - amount_out, mixRing, amount_keys, index, outSk);
 | 
			
		||||
      else
 | 
			
		||||
        tx.rct_signatures = rct::genRct(rct::hash2rct(tx_prefix_hash), inSk, destinations, outamounts, mixRing, amount_keys, sources[0].real_output, outSk); // same index assumption
 | 
			
		||||
 | 
			
		||||
      CHECK_AND_ASSERT_MES(tx.vout.size() == outSk.size(), false, "outSk size does not match vout");
 | 
			
		||||
 | 
			
		||||
      MCINFO("construct_tx", "transaction_created: " << get_transaction_hash(tx) << ENDL << obj_to_json_str(tx) << ENDL);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool construct_tx(const account_keys& sender_account_keys, const std::vector<tx_source_entry>& sources, const std::vector<tx_destination_entry>& destinations, std::vector<uint8_t> extra, transaction& tx, uint64_t unlock_time)
 | 
			
		||||
  {
 | 
			
		||||
     crypto::secret_key tx_key;
 | 
			
		||||
     return construct_tx_and_get_tx_key(sender_account_keys, sources, destinations, extra, tx, unlock_time, tx_key);
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool get_inputs_money_amount(const transaction& tx, uint64_t& money)
 | 
			
		||||
  {
 | 
			
		||||
    money = 0;
 | 
			
		||||
| 
						 | 
				
			
			@ -1057,36 +637,6 @@ namespace cryptonote
 | 
			
		|||
    return p;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool generate_genesis_block(
 | 
			
		||||
      block& bl
 | 
			
		||||
    , std::string const & genesis_tx
 | 
			
		||||
    , uint32_t nonce
 | 
			
		||||
    )
 | 
			
		||||
  {
 | 
			
		||||
    //genesis block
 | 
			
		||||
    bl = boost::value_initialized<block>();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    account_public_address ac = boost::value_initialized<account_public_address>();
 | 
			
		||||
    std::vector<size_t> sz;
 | 
			
		||||
    construct_miner_tx(0, 0, 0, 0, 0, ac, bl.miner_tx); // zero fee in genesis
 | 
			
		||||
    blobdata txb = tx_to_blob(bl.miner_tx);
 | 
			
		||||
    std::string hex_tx_represent = string_tools::buff_to_hex_nodelimer(txb);
 | 
			
		||||
 | 
			
		||||
    std::string genesis_coinbase_tx_hex = config::GENESIS_TX;
 | 
			
		||||
 | 
			
		||||
    blobdata tx_bl;
 | 
			
		||||
    string_tools::parse_hexstr_to_binbuff(genesis_coinbase_tx_hex, tx_bl);
 | 
			
		||||
    bool r = parse_and_validate_tx_from_blob(tx_bl, bl.miner_tx);
 | 
			
		||||
    CHECK_AND_ASSERT_MES(r, false, "failed to parse coinbase tx from hard coded blob");
 | 
			
		||||
    bl.major_version = CURRENT_BLOCK_MAJOR_VERSION;
 | 
			
		||||
    bl.minor_version = CURRENT_BLOCK_MINOR_VERSION;
 | 
			
		||||
    bl.timestamp = 0;
 | 
			
		||||
    bl.nonce = nonce;
 | 
			
		||||
    miner::find_nonce_for_given_block(bl, 1, 0);
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool get_block_longhash(const block& b, crypto::hash& res, uint64_t height)
 | 
			
		||||
  {
 | 
			
		||||
    // block 202612 bug workaround
 | 
			
		||||
| 
						 | 
				
			
			@ -30,14 +30,12 @@
 | 
			
		|||
 | 
			
		||||
#pragma once
 | 
			
		||||
#include "cryptonote_protocol/cryptonote_protocol_defs.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic_impl.h"
 | 
			
		||||
#include "account.h"
 | 
			
		||||
#include "include_base_utils.h"
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "ringct/rctOps.h"
 | 
			
		||||
#include <boost/serialization/vector.hpp>
 | 
			
		||||
#include <boost/serialization/utility.hpp>
 | 
			
		||||
 | 
			
		||||
namespace cryptonote
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			@ -46,43 +44,9 @@ namespace cryptonote
 | 
			
		|||
  crypto::hash get_transaction_prefix_hash(const transaction_prefix& tx);
 | 
			
		||||
  bool parse_and_validate_tx_from_blob(const blobdata& tx_blob, transaction& tx, crypto::hash& tx_hash, crypto::hash& tx_prefix_hash);
 | 
			
		||||
  bool parse_and_validate_tx_from_blob(const blobdata& tx_blob, transaction& tx);
 | 
			
		||||
  bool construct_miner_tx(size_t height, size_t median_size, uint64_t already_generated_coins, size_t current_block_size, uint64_t fee, const account_public_address &miner_address, transaction& tx, const blobdata& extra_nonce = blobdata(), size_t max_outs = 999, uint8_t hard_fork_version = 1);
 | 
			
		||||
  bool encrypt_payment_id(crypto::hash8 &payment_id, const crypto::public_key &public_key, const crypto::secret_key &secret_key);
 | 
			
		||||
  bool decrypt_payment_id(crypto::hash8 &payment_id, const crypto::public_key &public_key, const crypto::secret_key &secret_key);
 | 
			
		||||
 | 
			
		||||
  struct tx_source_entry
 | 
			
		||||
  {
 | 
			
		||||
    typedef std::pair<uint64_t, rct::ctkey> output_entry;
 | 
			
		||||
 | 
			
		||||
    std::vector<output_entry> outputs;  //index + key + optional ringct commitment
 | 
			
		||||
    size_t real_output;                 //index in outputs vector of real output_entry
 | 
			
		||||
    crypto::public_key real_out_tx_key; //incoming real tx public key
 | 
			
		||||
    size_t real_output_in_tx_index;     //index in transaction outputs vector
 | 
			
		||||
    uint64_t amount;                    //money
 | 
			
		||||
    bool rct;                           //true if the output is rct
 | 
			
		||||
    rct::key mask;                      //ringct amount mask
 | 
			
		||||
 | 
			
		||||
    void push_output(uint64_t idx, const crypto::public_key &k, uint64_t amount) { outputs.push_back(std::make_pair(idx, rct::ctkey({rct::pk2rct(k), rct::zeroCommit(amount)}))); }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  struct tx_destination_entry
 | 
			
		||||
  {
 | 
			
		||||
    uint64_t amount;                    //money
 | 
			
		||||
    account_public_address addr;        //destination address
 | 
			
		||||
 | 
			
		||||
    tx_destination_entry() : amount(0), addr(AUTO_VAL_INIT(addr)) { }
 | 
			
		||||
    tx_destination_entry(uint64_t a, const account_public_address &ad) : amount(a), addr(ad) { }
 | 
			
		||||
 | 
			
		||||
    BEGIN_SERIALIZE_OBJECT()
 | 
			
		||||
      VARINT_FIELD(amount)
 | 
			
		||||
      FIELD(addr)
 | 
			
		||||
    END_SERIALIZE()
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool construct_tx(const account_keys& sender_account_keys, const std::vector<tx_source_entry>& sources, const std::vector<tx_destination_entry>& destinations, std::vector<uint8_t> extra, transaction& tx, uint64_t unlock_time);
 | 
			
		||||
  bool construct_tx_and_get_tx_key(const account_keys& sender_account_keys, const std::vector<tx_source_entry>& sources, const std::vector<tx_destination_entry>& destinations, std::vector<uint8_t> extra, transaction& tx, uint64_t unlock_time, crypto::secret_key &tx_key, bool rct = false);
 | 
			
		||||
 | 
			
		||||
  template<typename T>
 | 
			
		||||
  bool find_tx_extra_field_by_type(const std::vector<tx_extra_field>& tx_extra_fields, T& field, size_t index = 0)
 | 
			
		||||
  {
 | 
			
		||||
| 
						 | 
				
			
			@ -125,11 +89,6 @@ namespace cryptonote
 | 
			
		|||
  crypto::hash get_block_hash(const block& b);
 | 
			
		||||
  bool get_block_longhash(const block& b, crypto::hash& res, uint64_t height);
 | 
			
		||||
  crypto::hash get_block_longhash(const block& b, uint64_t height);
 | 
			
		||||
  bool generate_genesis_block(
 | 
			
		||||
      block& bl
 | 
			
		||||
    , std::string const & genesis_tx
 | 
			
		||||
    , uint32_t nonce
 | 
			
		||||
    );
 | 
			
		||||
  bool parse_and_validate_block_from_blob(const blobdata& b_blob, block& b);
 | 
			
		||||
  bool get_inputs_money_amount(const transaction& tx, uint64_t& money);
 | 
			
		||||
  uint64_t get_outs_money_amount(const transaction& tx);
 | 
			
		||||
| 
						 | 
				
			
			@ -252,23 +211,3 @@ namespace cryptonote
 | 
			
		|||
  specific_type& variable_name = boost::get<specific_type>(variant_var);
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
BOOST_CLASS_VERSION(cryptonote::tx_source_entry, 0)
 | 
			
		||||
 | 
			
		||||
namespace boost
 | 
			
		||||
{
 | 
			
		||||
  namespace serialization
 | 
			
		||||
  {
 | 
			
		||||
    template <class Archive>
 | 
			
		||||
    inline void serialize(Archive &a, cryptonote::tx_source_entry &x, const boost::serialization::version_type ver)
 | 
			
		||||
    {
 | 
			
		||||
      a & x.outputs;
 | 
			
		||||
      a & x.real_output;
 | 
			
		||||
      a & x.real_out_tx_key;
 | 
			
		||||
      a & x.real_output_in_tx_index;
 | 
			
		||||
      a & x.amount;
 | 
			
		||||
      a & x.rct;
 | 
			
		||||
      a & x.mask;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -29,7 +29,7 @@
 | 
			
		|||
#include <algorithm>
 | 
			
		||||
#include <cstdio>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic.h"
 | 
			
		||||
#include "blockchain_db/blockchain_db.h"
 | 
			
		||||
#include "hardfork.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -29,7 +29,7 @@
 | 
			
		|||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "syncobj.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic.h"
 | 
			
		||||
 | 
			
		||||
namespace cryptonote
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			@ -27,38 +27,19 @@
 | 
			
		|||
# THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
			
		||||
 | 
			
		||||
set(cryptonote_core_sources
 | 
			
		||||
  account.cpp
 | 
			
		||||
  blockchain.cpp
 | 
			
		||||
  checkpoints.cpp
 | 
			
		||||
  cryptonote_basic_impl.cpp
 | 
			
		||||
  cryptonote_core.cpp
 | 
			
		||||
  cryptonote_format_utils.cpp
 | 
			
		||||
  difficulty.cpp
 | 
			
		||||
  miner.cpp
 | 
			
		||||
  tx_pool.cpp
 | 
			
		||||
  hardfork.cpp)
 | 
			
		||||
  cryptonote_tx_utils.cpp)
 | 
			
		||||
 | 
			
		||||
set(cryptonote_core_headers)
 | 
			
		||||
 | 
			
		||||
set(cryptonote_core_private_headers
 | 
			
		||||
  account.h
 | 
			
		||||
  account_boost_serialization.h
 | 
			
		||||
  blockchain_storage_boost_serialization.h
 | 
			
		||||
  blockchain.h
 | 
			
		||||
  checkpoints.h
 | 
			
		||||
  connection_context.h
 | 
			
		||||
  cryptonote_basic.h
 | 
			
		||||
  cryptonote_basic_impl.h
 | 
			
		||||
  cryptonote_boost_serialization.h
 | 
			
		||||
  cryptonote_core.h
 | 
			
		||||
  cryptonote_format_utils.h
 | 
			
		||||
  cryptonote_stat_info.h
 | 
			
		||||
  difficulty.h
 | 
			
		||||
  miner.h
 | 
			
		||||
  tx_extra.h
 | 
			
		||||
  tx_pool.h
 | 
			
		||||
  verification_context.h
 | 
			
		||||
  hardfork.h)
 | 
			
		||||
  cryptonote_tx_utils.h)
 | 
			
		||||
 | 
			
		||||
if(PER_BLOCK_CHECKPOINT)
 | 
			
		||||
  set(Blocks "blocks")
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -34,14 +34,13 @@
 | 
			
		|||
#include <boost/range/adaptor/reversed.hpp>
 | 
			
		||||
 | 
			
		||||
#include "include_base_utils.h"
 | 
			
		||||
#include "cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "tx_pool.h"
 | 
			
		||||
#include "blockchain.h"
 | 
			
		||||
#include "blockchain_db/blockchain_db.h"
 | 
			
		||||
#include "cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_config.h"
 | 
			
		||||
#include "miner.h"
 | 
			
		||||
#include "cryptonote_basic/miner.h"
 | 
			
		||||
#include "misc_language.h"
 | 
			
		||||
#include "profile_tools.h"
 | 
			
		||||
#include "file_io_utils.h"
 | 
			
		||||
| 
						 | 
				
			
			@ -49,8 +48,8 @@
 | 
			
		|||
#include "common/boost_serialization_helper.h"
 | 
			
		||||
#include "warnings.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "cryptonote_core/checkpoints.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_core.h"
 | 
			
		||||
#include "cryptonote_basic/checkpoints.h"
 | 
			
		||||
#include "cryptonote_core.h"
 | 
			
		||||
#include "ringct/rctSigs.h"
 | 
			
		||||
#include "common/perf_timer.h"
 | 
			
		||||
#if defined(PER_BLOCK_CHECKPOINT)
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -42,16 +42,16 @@
 | 
			
		|||
 | 
			
		||||
#include "syncobj.h"
 | 
			
		||||
#include "string_tools.h"
 | 
			
		||||
#include "cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "common/util.h"
 | 
			
		||||
#include "cryptonote_protocol/cryptonote_protocol_defs.h"
 | 
			
		||||
#include "rpc/core_rpc_server_commands_defs.h"
 | 
			
		||||
#include "difficulty.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "verification_context.h"
 | 
			
		||||
#include "cryptonote_basic/difficulty.h"
 | 
			
		||||
#include "cryptonote_tx_utils.h"
 | 
			
		||||
#include "cryptonote_basic/verification_context.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "checkpoints.h"
 | 
			
		||||
#include "hardfork.h"
 | 
			
		||||
#include "cryptonote_basic/checkpoints.h"
 | 
			
		||||
#include "cryptonote_basic/hardfork.h"
 | 
			
		||||
#include "blockchain_db/blockchain_db.h"
 | 
			
		||||
 | 
			
		||||
namespace cryptonote
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -38,11 +38,11 @@ using namespace epee;
 | 
			
		|||
#include "warnings.h"
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "cryptonote_config.h"
 | 
			
		||||
#include "cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_tx_utils.h"
 | 
			
		||||
#include "misc_language.h"
 | 
			
		||||
#include <csignal>
 | 
			
		||||
#include <p2p/net_node.h>
 | 
			
		||||
#include "cryptonote_core/checkpoints.h"
 | 
			
		||||
#include "cryptonote_basic/checkpoints.h"
 | 
			
		||||
#include "ringct/rctTypes.h"
 | 
			
		||||
#include "blockchain_db/blockchain_db.h"
 | 
			
		||||
#include "blockchain_db/lmdb/db_lmdb.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -41,9 +41,9 @@
 | 
			
		|||
#include "storages/portable_storage_template_helper.h"
 | 
			
		||||
#include "tx_pool.h"
 | 
			
		||||
#include "blockchain.h"
 | 
			
		||||
#include "miner.h"
 | 
			
		||||
#include "connection_context.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_stat_info.h"
 | 
			
		||||
#include "cryptonote_basic/miner.h"
 | 
			
		||||
#include "cryptonote_basic/connection_context.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_stat_info.h"
 | 
			
		||||
#include "warnings.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										493
									
								
								src/cryptonote_core/cryptonote_tx_utils.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										493
									
								
								src/cryptonote_core/cryptonote_tx_utils.cpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,493 @@
 | 
			
		|||
// Copyright (c) 2014-2016, The Monero Project
 | 
			
		||||
// 
 | 
			
		||||
// All rights reserved.
 | 
			
		||||
// 
 | 
			
		||||
// Redistribution and use in source and binary forms, with or without modification, are
 | 
			
		||||
// permitted provided that the following conditions are met:
 | 
			
		||||
// 
 | 
			
		||||
// 1. Redistributions of source code must retain the above copyright notice, this list of
 | 
			
		||||
//    conditions and the following disclaimer.
 | 
			
		||||
// 
 | 
			
		||||
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
 | 
			
		||||
//    of conditions and the following disclaimer in the documentation and/or other
 | 
			
		||||
//    materials provided with the distribution.
 | 
			
		||||
// 
 | 
			
		||||
// 3. Neither the name of the copyright holder nor the names of its contributors may be
 | 
			
		||||
//    used to endorse or promote products derived from this software without specific
 | 
			
		||||
//    prior written permission.
 | 
			
		||||
// 
 | 
			
		||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
 | 
			
		||||
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
 | 
			
		||||
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
 | 
			
		||||
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 | 
			
		||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 | 
			
		||||
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 | 
			
		||||
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 | 
			
		||||
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
 | 
			
		||||
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
			
		||||
// 
 | 
			
		||||
// Parts of this file are originally copyright (c) 2012-2013 The Cryptonote developers
 | 
			
		||||
 | 
			
		||||
#include "include_base_utils.h"
 | 
			
		||||
using namespace epee;
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_tx_utils.h"
 | 
			
		||||
#include "cryptonote_config.h"
 | 
			
		||||
#include "cryptonote_basic/miner.h"
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "ringct/rctSigs.h"
 | 
			
		||||
 | 
			
		||||
namespace cryptonote
 | 
			
		||||
{
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool construct_miner_tx(size_t height, size_t median_size, uint64_t already_generated_coins, size_t current_block_size, uint64_t fee, const account_public_address &miner_address, transaction& tx, const blobdata& extra_nonce, size_t max_outs, uint8_t hard_fork_version) {
 | 
			
		||||
    tx.vin.clear();
 | 
			
		||||
    tx.vout.clear();
 | 
			
		||||
    tx.extra.clear();
 | 
			
		||||
 | 
			
		||||
    keypair txkey = keypair::generate();
 | 
			
		||||
    add_tx_pub_key_to_extra(tx, txkey.pub);
 | 
			
		||||
    if(!extra_nonce.empty())
 | 
			
		||||
      if(!add_extra_nonce_to_tx_extra(tx.extra, extra_nonce))
 | 
			
		||||
        return false;
 | 
			
		||||
 | 
			
		||||
    txin_gen in;
 | 
			
		||||
    in.height = height;
 | 
			
		||||
 | 
			
		||||
    uint64_t block_reward;
 | 
			
		||||
    if(!get_block_reward(median_size, current_block_size, already_generated_coins, block_reward, hard_fork_version))
 | 
			
		||||
    {
 | 
			
		||||
      LOG_PRINT_L0("Block is too big");
 | 
			
		||||
      return false;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
#if defined(DEBUG_CREATE_BLOCK_TEMPLATE)
 | 
			
		||||
    LOG_PRINT_L1("Creating block template: reward " << block_reward <<
 | 
			
		||||
      ", fee " << fee);
 | 
			
		||||
#endif
 | 
			
		||||
    block_reward += fee;
 | 
			
		||||
 | 
			
		||||
    // from hard fork 2, we cut out the low significant digits. This makes the tx smaller, and
 | 
			
		||||
    // keeps the paid amount almost the same. The unpaid remainder gets pushed back to the
 | 
			
		||||
    // emission schedule
 | 
			
		||||
    // from hard fork 4, we use a single "dusty" output. This makes the tx even smaller,
 | 
			
		||||
    // and avoids the quantization. These outputs will be added as rct outputs with identity
 | 
			
		||||
    // masks, to they can be used as rct inputs.
 | 
			
		||||
    if (hard_fork_version >= 2 && hard_fork_version < 4) {
 | 
			
		||||
      block_reward = block_reward - block_reward % ::config::BASE_REWARD_CLAMP_THRESHOLD;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    std::vector<uint64_t> out_amounts;
 | 
			
		||||
    decompose_amount_into_digits(block_reward, hard_fork_version >= 2 ? 0 : ::config::DEFAULT_DUST_THRESHOLD,
 | 
			
		||||
      [&out_amounts](uint64_t a_chunk) { out_amounts.push_back(a_chunk); },
 | 
			
		||||
      [&out_amounts](uint64_t a_dust) { out_amounts.push_back(a_dust); });
 | 
			
		||||
 | 
			
		||||
    CHECK_AND_ASSERT_MES(1 <= max_outs, false, "max_out must be non-zero");
 | 
			
		||||
    if (height == 0 || hard_fork_version >= 4)
 | 
			
		||||
    {
 | 
			
		||||
      // the genesis block was not decomposed, for unknown reasons
 | 
			
		||||
      while (max_outs < out_amounts.size())
 | 
			
		||||
      {
 | 
			
		||||
        //out_amounts[out_amounts.size() - 2] += out_amounts.back();
 | 
			
		||||
        //out_amounts.resize(out_amounts.size() - 1);
 | 
			
		||||
        out_amounts[1] += out_amounts[0];
 | 
			
		||||
        for (size_t n = 1; n < out_amounts.size(); ++n)
 | 
			
		||||
          out_amounts[n - 1] = out_amounts[n];
 | 
			
		||||
        out_amounts.resize(out_amounts.size() - 1);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    else
 | 
			
		||||
    {
 | 
			
		||||
      CHECK_AND_ASSERT_MES(max_outs >= out_amounts.size(), false, "max_out exceeded");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    uint64_t summary_amounts = 0;
 | 
			
		||||
    for (size_t no = 0; no < out_amounts.size(); no++)
 | 
			
		||||
    {
 | 
			
		||||
      crypto::key_derivation derivation = AUTO_VAL_INIT(derivation);;
 | 
			
		||||
      crypto::public_key out_eph_public_key = AUTO_VAL_INIT(out_eph_public_key);
 | 
			
		||||
      bool r = crypto::generate_key_derivation(miner_address.m_view_public_key, txkey.sec, derivation);
 | 
			
		||||
      CHECK_AND_ASSERT_MES(r, false, "while creating outs: failed to generate_key_derivation(" << miner_address.m_view_public_key << ", " << txkey.sec << ")");
 | 
			
		||||
 | 
			
		||||
      r = crypto::derive_public_key(derivation, no, miner_address.m_spend_public_key, out_eph_public_key);
 | 
			
		||||
      CHECK_AND_ASSERT_MES(r, false, "while creating outs: failed to derive_public_key(" << derivation << ", " << no << ", "<< miner_address.m_spend_public_key << ")");
 | 
			
		||||
 | 
			
		||||
      txout_to_key tk;
 | 
			
		||||
      tk.key = out_eph_public_key;
 | 
			
		||||
 | 
			
		||||
      tx_out out;
 | 
			
		||||
      summary_amounts += out.amount = out_amounts[no];
 | 
			
		||||
      out.target = tk;
 | 
			
		||||
      tx.vout.push_back(out);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    CHECK_AND_ASSERT_MES(summary_amounts == block_reward, false, "Failed to construct miner tx, summary_amounts = " << summary_amounts << " not equal block_reward = " << block_reward);
 | 
			
		||||
 | 
			
		||||
    if (hard_fork_version >= 4)
 | 
			
		||||
      tx.version = 2;
 | 
			
		||||
    else
 | 
			
		||||
      tx.version = 1;
 | 
			
		||||
 | 
			
		||||
    //lock
 | 
			
		||||
    tx.unlock_time = height + CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW;
 | 
			
		||||
    tx.vin.push_back(in);
 | 
			
		||||
    //LOG_PRINT("MINER_TX generated ok, block_reward=" << print_money(block_reward) << "("  << print_money(block_reward - fee) << "+" << print_money(fee)
 | 
			
		||||
    //  << "), current_block_size=" << current_block_size << ", already_generated_coins=" << already_generated_coins << ", tx_id=" << get_transaction_hash(tx), LOG_LEVEL_2);
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  crypto::public_key get_destination_view_key_pub(const std::vector<tx_destination_entry> &destinations, const account_keys &sender_keys)
 | 
			
		||||
  {
 | 
			
		||||
    if (destinations.empty())
 | 
			
		||||
      return null_pkey;
 | 
			
		||||
    for (size_t n = 1; n < destinations.size(); ++n)
 | 
			
		||||
    {
 | 
			
		||||
      if (!memcmp(&destinations[n].addr, &sender_keys.m_account_address, sizeof(destinations[0].addr)))
 | 
			
		||||
        continue;
 | 
			
		||||
      if (destinations[n].amount == 0)
 | 
			
		||||
        continue;
 | 
			
		||||
      if (memcmp(&destinations[n].addr, &destinations[0].addr, sizeof(destinations[0].addr)))
 | 
			
		||||
        return null_pkey;
 | 
			
		||||
    }
 | 
			
		||||
    return destinations[0].addr.m_view_public_key;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool construct_tx_and_get_tx_key(const account_keys& sender_account_keys, const std::vector<tx_source_entry>& sources, const std::vector<tx_destination_entry>& destinations, std::vector<uint8_t> extra, transaction& tx, uint64_t unlock_time, crypto::secret_key &tx_key, bool rct)
 | 
			
		||||
  {
 | 
			
		||||
    std::vector<rct::key> amount_keys;
 | 
			
		||||
    tx.set_null();
 | 
			
		||||
    amount_keys.clear();
 | 
			
		||||
 | 
			
		||||
    tx.version = rct ? 2 : 1;
 | 
			
		||||
    tx.unlock_time = unlock_time;
 | 
			
		||||
 | 
			
		||||
    tx.extra = extra;
 | 
			
		||||
    keypair txkey = keypair::generate();
 | 
			
		||||
    remove_field_from_tx_extra(tx.extra, typeid(tx_extra_pub_key));
 | 
			
		||||
    add_tx_pub_key_to_extra(tx, txkey.pub);
 | 
			
		||||
    tx_key = txkey.sec;
 | 
			
		||||
 | 
			
		||||
    // if we have a stealth payment id, find it and encrypt it with the tx key now
 | 
			
		||||
    std::vector<tx_extra_field> tx_extra_fields;
 | 
			
		||||
    if (parse_tx_extra(tx.extra, tx_extra_fields))
 | 
			
		||||
    {
 | 
			
		||||
      tx_extra_nonce extra_nonce;
 | 
			
		||||
      if (find_tx_extra_field_by_type(tx_extra_fields, extra_nonce))
 | 
			
		||||
      {
 | 
			
		||||
        crypto::hash8 payment_id = null_hash8;
 | 
			
		||||
        if (get_encrypted_payment_id_from_tx_extra_nonce(extra_nonce.nonce, payment_id))
 | 
			
		||||
        {
 | 
			
		||||
          LOG_PRINT_L2("Encrypting payment id " << payment_id);
 | 
			
		||||
          crypto::public_key view_key_pub = get_destination_view_key_pub(destinations, sender_account_keys);
 | 
			
		||||
          if (view_key_pub == null_pkey)
 | 
			
		||||
          {
 | 
			
		||||
            LOG_ERROR("Destinations have to have exactly one output to support encrypted payment ids");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          if (!encrypt_payment_id(payment_id, view_key_pub, txkey.sec))
 | 
			
		||||
          {
 | 
			
		||||
            LOG_ERROR("Failed to encrypt payment id");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          std::string extra_nonce;
 | 
			
		||||
          set_encrypted_payment_id_to_tx_extra_nonce(extra_nonce, payment_id);
 | 
			
		||||
          remove_field_from_tx_extra(tx.extra, typeid(tx_extra_nonce));
 | 
			
		||||
          if (!add_extra_nonce_to_tx_extra(tx.extra, extra_nonce))
 | 
			
		||||
          {
 | 
			
		||||
            LOG_ERROR("Failed to add encrypted payment id to tx extra");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
          LOG_PRINT_L1("Encrypted payment ID: " << payment_id);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    else
 | 
			
		||||
    {
 | 
			
		||||
      LOG_ERROR("Failed to parse tx extra");
 | 
			
		||||
      return false;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    struct input_generation_context_data
 | 
			
		||||
    {
 | 
			
		||||
      keypair in_ephemeral;
 | 
			
		||||
    };
 | 
			
		||||
    std::vector<input_generation_context_data> in_contexts;
 | 
			
		||||
 | 
			
		||||
    uint64_t summary_inputs_money = 0;
 | 
			
		||||
    //fill inputs
 | 
			
		||||
    int idx = -1;
 | 
			
		||||
    for(const tx_source_entry& src_entr:  sources)
 | 
			
		||||
    {
 | 
			
		||||
      ++idx;
 | 
			
		||||
      if(src_entr.real_output >= src_entr.outputs.size())
 | 
			
		||||
      {
 | 
			
		||||
        LOG_ERROR("real_output index (" << src_entr.real_output << ")bigger than output_keys.size()=" << src_entr.outputs.size());
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
      summary_inputs_money += src_entr.amount;
 | 
			
		||||
 | 
			
		||||
      //key_derivation recv_derivation;
 | 
			
		||||
      in_contexts.push_back(input_generation_context_data());
 | 
			
		||||
      keypair& in_ephemeral = in_contexts.back().in_ephemeral;
 | 
			
		||||
      crypto::key_image img;
 | 
			
		||||
      if(!generate_key_image_helper(sender_account_keys, src_entr.real_out_tx_key, src_entr.real_output_in_tx_index, in_ephemeral, img))
 | 
			
		||||
        return false;
 | 
			
		||||
 | 
			
		||||
      //check that derivated key is equal with real output key
 | 
			
		||||
      if( !(in_ephemeral.pub == src_entr.outputs[src_entr.real_output].second.dest) )
 | 
			
		||||
      {
 | 
			
		||||
        LOG_ERROR("derived public key mismatch with output public key at index " << idx << ", real out " << src_entr.real_output << "! "<< ENDL << "derived_key:"
 | 
			
		||||
          << string_tools::pod_to_hex(in_ephemeral.pub) << ENDL << "real output_public_key:"
 | 
			
		||||
          << string_tools::pod_to_hex(src_entr.outputs[src_entr.real_output].second) );
 | 
			
		||||
        LOG_ERROR("amount " << src_entr.amount << ", rct " << src_entr.rct);
 | 
			
		||||
        LOG_ERROR("tx pubkey " << src_entr.real_out_tx_key << ", real_output_in_tx_index " << src_entr.real_output_in_tx_index);
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      //put key image into tx input
 | 
			
		||||
      txin_to_key input_to_key;
 | 
			
		||||
      input_to_key.amount = src_entr.amount;
 | 
			
		||||
      input_to_key.k_image = img;
 | 
			
		||||
 | 
			
		||||
      //fill outputs array and use relative offsets
 | 
			
		||||
      for(const tx_source_entry::output_entry& out_entry: src_entr.outputs)
 | 
			
		||||
        input_to_key.key_offsets.push_back(out_entry.first);
 | 
			
		||||
 | 
			
		||||
      input_to_key.key_offsets = absolute_output_offsets_to_relative(input_to_key.key_offsets);
 | 
			
		||||
      tx.vin.push_back(input_to_key);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // "Shuffle" outs
 | 
			
		||||
    std::vector<tx_destination_entry> shuffled_dsts(destinations);
 | 
			
		||||
    std::sort(shuffled_dsts.begin(), shuffled_dsts.end(), [](const tx_destination_entry& de1, const tx_destination_entry& de2) { return de1.amount < de2.amount; } );
 | 
			
		||||
 | 
			
		||||
    uint64_t summary_outs_money = 0;
 | 
			
		||||
    //fill outputs
 | 
			
		||||
    size_t output_index = 0;
 | 
			
		||||
    for(const tx_destination_entry& dst_entr:  shuffled_dsts)
 | 
			
		||||
    {
 | 
			
		||||
      CHECK_AND_ASSERT_MES(dst_entr.amount > 0 || tx.version > 1, false, "Destination with wrong amount: " << dst_entr.amount);
 | 
			
		||||
      crypto::key_derivation derivation;
 | 
			
		||||
      crypto::public_key out_eph_public_key;
 | 
			
		||||
      bool r = crypto::generate_key_derivation(dst_entr.addr.m_view_public_key, txkey.sec, derivation);
 | 
			
		||||
      CHECK_AND_ASSERT_MES(r, false, "at creation outs: failed to generate_key_derivation(" << dst_entr.addr.m_view_public_key << ", " << txkey.sec << ")");
 | 
			
		||||
 | 
			
		||||
      if (tx.version > 1)
 | 
			
		||||
      {
 | 
			
		||||
        crypto::secret_key scalar1;
 | 
			
		||||
        crypto::derivation_to_scalar(derivation, output_index, scalar1);
 | 
			
		||||
        amount_keys.push_back(rct::sk2rct(scalar1));
 | 
			
		||||
      }
 | 
			
		||||
      r = crypto::derive_public_key(derivation, output_index, dst_entr.addr.m_spend_public_key, out_eph_public_key);
 | 
			
		||||
      CHECK_AND_ASSERT_MES(r, false, "at creation outs: failed to derive_public_key(" << derivation << ", " << output_index << ", "<< dst_entr.addr.m_spend_public_key << ")");
 | 
			
		||||
 | 
			
		||||
      tx_out out;
 | 
			
		||||
      out.amount = dst_entr.amount;
 | 
			
		||||
      txout_to_key tk;
 | 
			
		||||
      tk.key = out_eph_public_key;
 | 
			
		||||
      out.target = tk;
 | 
			
		||||
      tx.vout.push_back(out);
 | 
			
		||||
      output_index++;
 | 
			
		||||
      summary_outs_money += dst_entr.amount;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    //check money
 | 
			
		||||
    if(summary_outs_money > summary_inputs_money )
 | 
			
		||||
    {
 | 
			
		||||
      LOG_ERROR("Transaction inputs money ("<< summary_inputs_money << ") less than outputs money (" << summary_outs_money << ")");
 | 
			
		||||
      return false;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // check for watch only wallet
 | 
			
		||||
    bool zero_secret_key = true;
 | 
			
		||||
    for (size_t i = 0; i < sizeof(sender_account_keys.m_spend_secret_key); ++i)
 | 
			
		||||
      zero_secret_key &= (sender_account_keys.m_spend_secret_key.data[i] == 0);
 | 
			
		||||
    if (zero_secret_key)
 | 
			
		||||
    {
 | 
			
		||||
      MDEBUG("Null secret key, skipping signatures");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (tx.version == 1)
 | 
			
		||||
    {
 | 
			
		||||
      //generate ring signatures
 | 
			
		||||
      crypto::hash tx_prefix_hash;
 | 
			
		||||
      get_transaction_prefix_hash(tx, tx_prefix_hash);
 | 
			
		||||
 | 
			
		||||
      std::stringstream ss_ring_s;
 | 
			
		||||
      size_t i = 0;
 | 
			
		||||
      for(const tx_source_entry& src_entr:  sources)
 | 
			
		||||
      {
 | 
			
		||||
        ss_ring_s << "pub_keys:" << ENDL;
 | 
			
		||||
        std::vector<const crypto::public_key*> keys_ptrs;
 | 
			
		||||
        std::vector<crypto::public_key> keys(src_entr.outputs.size());
 | 
			
		||||
        size_t ii = 0;
 | 
			
		||||
        for(const tx_source_entry::output_entry& o: src_entr.outputs)
 | 
			
		||||
        {
 | 
			
		||||
          keys[ii] = rct2pk(o.second.dest);
 | 
			
		||||
          keys_ptrs.push_back(&keys[ii]);
 | 
			
		||||
          ss_ring_s << o.second.dest << ENDL;
 | 
			
		||||
          ++ii;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        tx.signatures.push_back(std::vector<crypto::signature>());
 | 
			
		||||
        std::vector<crypto::signature>& sigs = tx.signatures.back();
 | 
			
		||||
        sigs.resize(src_entr.outputs.size());
 | 
			
		||||
        if (!zero_secret_key)
 | 
			
		||||
          crypto::generate_ring_signature(tx_prefix_hash, boost::get<txin_to_key>(tx.vin[i]).k_image, keys_ptrs, in_contexts[i].in_ephemeral.sec, src_entr.real_output, sigs.data());
 | 
			
		||||
        ss_ring_s << "signatures:" << ENDL;
 | 
			
		||||
        std::for_each(sigs.begin(), sigs.end(), [&](const crypto::signature& s){ss_ring_s << s << ENDL;});
 | 
			
		||||
        ss_ring_s << "prefix_hash:" << tx_prefix_hash << ENDL << "in_ephemeral_key: " << in_contexts[i].in_ephemeral.sec << ENDL << "real_output: " << src_entr.real_output;
 | 
			
		||||
        i++;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      MCINFO("construct_tx", "transaction_created: " << get_transaction_hash(tx) << ENDL << obj_to_json_str(tx) << ENDL << ss_ring_s.str());
 | 
			
		||||
    }
 | 
			
		||||
    else
 | 
			
		||||
    {
 | 
			
		||||
      size_t n_total_outs = sources[0].outputs.size(); // only for non-simple rct
 | 
			
		||||
 | 
			
		||||
      // the non-simple version is slightly smaller, but assumes all real inputs
 | 
			
		||||
      // are on the same index, so can only be used if there just one ring.
 | 
			
		||||
      bool use_simple_rct = sources.size() > 1;
 | 
			
		||||
 | 
			
		||||
      if (!use_simple_rct)
 | 
			
		||||
      {
 | 
			
		||||
        // non simple ringct requires all real inputs to be at the same index for all inputs
 | 
			
		||||
        for(const tx_source_entry& src_entr:  sources)
 | 
			
		||||
        {
 | 
			
		||||
          if(src_entr.real_output != sources.begin()->real_output)
 | 
			
		||||
          {
 | 
			
		||||
            LOG_ERROR("All inputs must have the same index for non-simple ringct");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // enforce same mixin for all outputs
 | 
			
		||||
        for (size_t i = 1; i < sources.size(); ++i) {
 | 
			
		||||
          if (n_total_outs != sources[i].outputs.size()) {
 | 
			
		||||
            LOG_ERROR("Non-simple ringct transaction has varying mixin");
 | 
			
		||||
            return false;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      uint64_t amount_in = 0, amount_out = 0;
 | 
			
		||||
      rct::ctkeyV inSk;
 | 
			
		||||
      // mixRing indexing is done the other way round for simple
 | 
			
		||||
      rct::ctkeyM mixRing(use_simple_rct ? sources.size() : n_total_outs);
 | 
			
		||||
      rct::keyV destinations;
 | 
			
		||||
      std::vector<uint64_t> inamounts, outamounts;
 | 
			
		||||
      std::vector<unsigned int> index;
 | 
			
		||||
      for (size_t i = 0; i < sources.size(); ++i)
 | 
			
		||||
      {
 | 
			
		||||
        rct::ctkey ctkey;
 | 
			
		||||
        amount_in += sources[i].amount;
 | 
			
		||||
        inamounts.push_back(sources[i].amount);
 | 
			
		||||
        index.push_back(sources[i].real_output);
 | 
			
		||||
        // inSk: (secret key, mask)
 | 
			
		||||
        ctkey.dest = rct::sk2rct(in_contexts[i].in_ephemeral.sec);
 | 
			
		||||
        ctkey.mask = sources[i].mask;
 | 
			
		||||
        inSk.push_back(ctkey);
 | 
			
		||||
        // inPk: (public key, commitment)
 | 
			
		||||
        // will be done when filling in mixRing
 | 
			
		||||
      }
 | 
			
		||||
      for (size_t i = 0; i < tx.vout.size(); ++i)
 | 
			
		||||
      {
 | 
			
		||||
        destinations.push_back(rct::pk2rct(boost::get<txout_to_key>(tx.vout[i].target).key));
 | 
			
		||||
        outamounts.push_back(tx.vout[i].amount);
 | 
			
		||||
        amount_out += tx.vout[i].amount;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (use_simple_rct)
 | 
			
		||||
      {
 | 
			
		||||
        // mixRing indexing is done the other way round for simple
 | 
			
		||||
        for (size_t i = 0; i < sources.size(); ++i)
 | 
			
		||||
        {
 | 
			
		||||
          mixRing[i].resize(sources[i].outputs.size());
 | 
			
		||||
          for (size_t n = 0; n < sources[i].outputs.size(); ++n)
 | 
			
		||||
          {
 | 
			
		||||
            mixRing[i][n] = sources[i].outputs[n].second;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      else
 | 
			
		||||
      {
 | 
			
		||||
        for (size_t i = 0; i < n_total_outs; ++i) // same index assumption
 | 
			
		||||
        {
 | 
			
		||||
          mixRing[i].resize(sources.size());
 | 
			
		||||
          for (size_t n = 0; n < sources.size(); ++n)
 | 
			
		||||
          {
 | 
			
		||||
            mixRing[i][n] = sources[n].outputs[i].second;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // fee
 | 
			
		||||
      if (!use_simple_rct && amount_in > amount_out)
 | 
			
		||||
        outamounts.push_back(amount_in - amount_out);
 | 
			
		||||
 | 
			
		||||
      // zero out all amounts to mask rct outputs, real amounts are now encrypted
 | 
			
		||||
      for (size_t i = 0; i < tx.vin.size(); ++i)
 | 
			
		||||
      {
 | 
			
		||||
        if (sources[i].rct)
 | 
			
		||||
          boost::get<txin_to_key>(tx.vin[i]).amount = 0;
 | 
			
		||||
      }
 | 
			
		||||
      for (size_t i = 0; i < tx.vout.size(); ++i)
 | 
			
		||||
        tx.vout[i].amount = 0;
 | 
			
		||||
 | 
			
		||||
      crypto::hash tx_prefix_hash;
 | 
			
		||||
      get_transaction_prefix_hash(tx, tx_prefix_hash);
 | 
			
		||||
      rct::ctkeyV outSk;
 | 
			
		||||
      if (use_simple_rct)
 | 
			
		||||
        tx.rct_signatures = rct::genRctSimple(rct::hash2rct(tx_prefix_hash), inSk, destinations, inamounts, outamounts, amount_in - amount_out, mixRing, amount_keys, index, outSk);
 | 
			
		||||
      else
 | 
			
		||||
        tx.rct_signatures = rct::genRct(rct::hash2rct(tx_prefix_hash), inSk, destinations, outamounts, mixRing, amount_keys, sources[0].real_output, outSk); // same index assumption
 | 
			
		||||
 | 
			
		||||
      CHECK_AND_ASSERT_MES(tx.vout.size() == outSk.size(), false, "outSk size does not match vout");
 | 
			
		||||
 | 
			
		||||
      MCINFO("construct_tx", "transaction_created: " << get_transaction_hash(tx) << ENDL << obj_to_json_str(tx) << ENDL);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool construct_tx(const account_keys& sender_account_keys, const std::vector<tx_source_entry>& sources, const std::vector<tx_destination_entry>& destinations, std::vector<uint8_t> extra, transaction& tx, uint64_t unlock_time)
 | 
			
		||||
  {
 | 
			
		||||
     crypto::secret_key tx_key;
 | 
			
		||||
     return construct_tx_and_get_tx_key(sender_account_keys, sources, destinations, extra, tx, unlock_time, tx_key);
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool generate_genesis_block(
 | 
			
		||||
      block& bl
 | 
			
		||||
    , std::string const & genesis_tx
 | 
			
		||||
    , uint32_t nonce
 | 
			
		||||
    )
 | 
			
		||||
  {
 | 
			
		||||
    //genesis block
 | 
			
		||||
    bl = boost::value_initialized<block>();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    account_public_address ac = boost::value_initialized<account_public_address>();
 | 
			
		||||
    std::vector<size_t> sz;
 | 
			
		||||
    construct_miner_tx(0, 0, 0, 0, 0, ac, bl.miner_tx); // zero fee in genesis
 | 
			
		||||
    blobdata txb = tx_to_blob(bl.miner_tx);
 | 
			
		||||
    std::string hex_tx_represent = string_tools::buff_to_hex_nodelimer(txb);
 | 
			
		||||
 | 
			
		||||
    std::string genesis_coinbase_tx_hex = config::GENESIS_TX;
 | 
			
		||||
 | 
			
		||||
    blobdata tx_bl;
 | 
			
		||||
    string_tools::parse_hexstr_to_binbuff(genesis_coinbase_tx_hex, tx_bl);
 | 
			
		||||
    bool r = parse_and_validate_tx_from_blob(tx_bl, bl.miner_tx);
 | 
			
		||||
    CHECK_AND_ASSERT_MES(r, false, "failed to parse coinbase tx from hard coded blob");
 | 
			
		||||
    bl.major_version = CURRENT_BLOCK_MAJOR_VERSION;
 | 
			
		||||
    bl.minor_version = CURRENT_BLOCK_MINOR_VERSION;
 | 
			
		||||
    bl.timestamp = 0;
 | 
			
		||||
    bl.nonce = nonce;
 | 
			
		||||
    miner::find_nonce_for_given_block(bl, 1, 0);
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										101
									
								
								src/cryptonote_core/cryptonote_tx_utils.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								src/cryptonote_core/cryptonote_tx_utils.h
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,101 @@
 | 
			
		|||
// Copyright (c) 2014-2016, The Monero Project
 | 
			
		||||
// 
 | 
			
		||||
// All rights reserved.
 | 
			
		||||
// 
 | 
			
		||||
// Redistribution and use in source and binary forms, with or without modification, are
 | 
			
		||||
// permitted provided that the following conditions are met:
 | 
			
		||||
// 
 | 
			
		||||
// 1. Redistributions of source code must retain the above copyright notice, this list of
 | 
			
		||||
//    conditions and the following disclaimer.
 | 
			
		||||
// 
 | 
			
		||||
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
 | 
			
		||||
//    of conditions and the following disclaimer in the documentation and/or other
 | 
			
		||||
//    materials provided with the distribution.
 | 
			
		||||
// 
 | 
			
		||||
// 3. Neither the name of the copyright holder nor the names of its contributors may be
 | 
			
		||||
//    used to endorse or promote products derived from this software without specific
 | 
			
		||||
//    prior written permission.
 | 
			
		||||
// 
 | 
			
		||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
 | 
			
		||||
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
 | 
			
		||||
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
 | 
			
		||||
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 | 
			
		||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 | 
			
		||||
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 | 
			
		||||
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 | 
			
		||||
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
 | 
			
		||||
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
			
		||||
// 
 | 
			
		||||
// Parts of this file are originally copyright (c) 2012-2013 The Cryptonote developers
 | 
			
		||||
 | 
			
		||||
#pragma once
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include <boost/serialization/vector.hpp>
 | 
			
		||||
#include <boost/serialization/utility.hpp>
 | 
			
		||||
 | 
			
		||||
namespace cryptonote
 | 
			
		||||
{
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  bool construct_miner_tx(size_t height, size_t median_size, uint64_t already_generated_coins, size_t current_block_size, uint64_t fee, const account_public_address &miner_address, transaction& tx, const blobdata& extra_nonce = blobdata(), size_t max_outs = 999, uint8_t hard_fork_version = 1);
 | 
			
		||||
 | 
			
		||||
  struct tx_source_entry
 | 
			
		||||
  {
 | 
			
		||||
    typedef std::pair<uint64_t, rct::ctkey> output_entry;
 | 
			
		||||
 | 
			
		||||
    std::vector<output_entry> outputs;  //index + key + optional ringct commitment
 | 
			
		||||
    size_t real_output;                 //index in outputs vector of real output_entry
 | 
			
		||||
    crypto::public_key real_out_tx_key; //incoming real tx public key
 | 
			
		||||
    size_t real_output_in_tx_index;     //index in transaction outputs vector
 | 
			
		||||
    uint64_t amount;                    //money
 | 
			
		||||
    bool rct;                           //true if the output is rct
 | 
			
		||||
    rct::key mask;                      //ringct amount mask
 | 
			
		||||
 | 
			
		||||
    void push_output(uint64_t idx, const crypto::public_key &k, uint64_t amount) { outputs.push_back(std::make_pair(idx, rct::ctkey({rct::pk2rct(k), rct::zeroCommit(amount)}))); }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  struct tx_destination_entry
 | 
			
		||||
  {
 | 
			
		||||
    uint64_t amount;                    //money
 | 
			
		||||
    account_public_address addr;        //destination address
 | 
			
		||||
 | 
			
		||||
    tx_destination_entry() : amount(0), addr(AUTO_VAL_INIT(addr)) { }
 | 
			
		||||
    tx_destination_entry(uint64_t a, const account_public_address &ad) : amount(a), addr(ad) { }
 | 
			
		||||
 | 
			
		||||
    BEGIN_SERIALIZE_OBJECT()
 | 
			
		||||
      VARINT_FIELD(amount)
 | 
			
		||||
      FIELD(addr)
 | 
			
		||||
    END_SERIALIZE()
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  //---------------------------------------------------------------
 | 
			
		||||
  crypto::public_key get_destination_view_key_pub(const std::vector<tx_destination_entry> &destinations, const account_keys &sender_keys);
 | 
			
		||||
  bool construct_tx(const account_keys& sender_account_keys, const std::vector<tx_source_entry>& sources, const std::vector<tx_destination_entry>& destinations, std::vector<uint8_t> extra, transaction& tx, uint64_t unlock_time);
 | 
			
		||||
  bool construct_tx_and_get_tx_key(const account_keys& sender_account_keys, const std::vector<tx_source_entry>& sources, const std::vector<tx_destination_entry>& destinations, std::vector<uint8_t> extra, transaction& tx, uint64_t unlock_time, crypto::secret_key &tx_key, bool rct = false);
 | 
			
		||||
 | 
			
		||||
  bool generate_genesis_block(
 | 
			
		||||
      block& bl
 | 
			
		||||
    , std::string const & genesis_tx
 | 
			
		||||
    , uint32_t nonce
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
BOOST_CLASS_VERSION(cryptonote::tx_source_entry, 0)
 | 
			
		||||
 | 
			
		||||
namespace boost
 | 
			
		||||
{
 | 
			
		||||
  namespace serialization
 | 
			
		||||
  {
 | 
			
		||||
    template <class Archive>
 | 
			
		||||
    inline void serialize(Archive &a, cryptonote::tx_source_entry &x, const boost::serialization::version_type ver)
 | 
			
		||||
    {
 | 
			
		||||
      a & x.outputs;
 | 
			
		||||
      a & x.real_output;
 | 
			
		||||
      a & x.real_out_tx_key;
 | 
			
		||||
      a & x.real_output_in_tx_index;
 | 
			
		||||
      a & x.amount;
 | 
			
		||||
      a & x.rct;
 | 
			
		||||
      a & x.mask;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -34,8 +34,8 @@
 | 
			
		|||
#include <vector>
 | 
			
		||||
 | 
			
		||||
#include "tx_pool.h"
 | 
			
		||||
#include "cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_tx_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_config.h"
 | 
			
		||||
#include "blockchain.h"
 | 
			
		||||
#include "common/boost_serialization_helper.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -41,8 +41,8 @@
 | 
			
		|||
#include "string_tools.h"
 | 
			
		||||
#include "syncobj.h"
 | 
			
		||||
#include "math_helper.h"
 | 
			
		||||
#include "cryptonote_basic_impl.h"
 | 
			
		||||
#include "verification_context.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/verification_context.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "rpc/core_rpc_server_commands_defs.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -35,6 +35,8 @@ source_group(cryptonote_protocol FILES ${CRYPTONOTE_PROTOCOL})
 | 
			
		|||
#monero_private_headers(cryptonote_protocol ${CRYPTONOTE_PROTOCOL})
 | 
			
		||||
monero_add_library(cryptonote_protocol ${CRYPTONOTE_PROTOCOL})
 | 
			
		||||
target_link_libraries(cryptonote_protocol
 | 
			
		||||
  PUBLIC
 | 
			
		||||
    p2p
 | 
			
		||||
  PRIVATE
 | 
			
		||||
    ${EXTRA_LIBRARIES})
 | 
			
		||||
add_dependencies(cryptonote_protocol
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,7 +32,7 @@
 | 
			
		|||
 | 
			
		||||
#include <list>
 | 
			
		||||
#include "serialization/keyvalue_serialization.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_protocol/blobdatatype.h"
 | 
			
		||||
namespace cryptonote
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -42,9 +42,9 @@
 | 
			
		|||
#include "warnings.h"
 | 
			
		||||
#include "cryptonote_protocol_defs.h"
 | 
			
		||||
#include "cryptonote_protocol_handler_common.h"
 | 
			
		||||
#include "cryptonote_core/connection_context.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_stat_info.h"
 | 
			
		||||
#include "cryptonote_core/verification_context.h"
 | 
			
		||||
#include "cryptonote_basic/connection_context.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_stat_info.h"
 | 
			
		||||
#include "cryptonote_basic/verification_context.h"
 | 
			
		||||
// #include <netinet/in.h>
 | 
			
		||||
#include <boost/circular_buffer.hpp>
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -39,7 +39,7 @@
 | 
			
		|||
#include <list>
 | 
			
		||||
#include <unordered_map>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "profile_tools.h"
 | 
			
		||||
#include "../../src/p2p/network_throttle-detail.hpp"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,7 +32,7 @@
 | 
			
		|||
 | 
			
		||||
#include "p2p/net_node_common.h"
 | 
			
		||||
#include "cryptonote_protocol/cryptonote_protocol_defs.h"
 | 
			
		||||
#include "cryptonote_core/connection_context.h"
 | 
			
		||||
#include "cryptonote_basic/connection_context.h"
 | 
			
		||||
namespace cryptonote
 | 
			
		||||
{
 | 
			
		||||
  /************************************************************************/
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,7 +32,7 @@
 | 
			
		|||
#include "common/scoped_message_writer.h"
 | 
			
		||||
#include "common/util.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_core.h"
 | 
			
		||||
#include "cryptonote_core/miner.h"
 | 
			
		||||
#include "cryptonote_basic/miner.h"
 | 
			
		||||
#include "daemon/command_server.h"
 | 
			
		||||
#include "daemon/daemon.h"
 | 
			
		||||
#include "daemon/executor.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -33,7 +33,7 @@
 | 
			
		|||
#include "daemon/rpc_command_executor.h"
 | 
			
		||||
#include "rpc/core_rpc_server_commands_defs.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_core.h"
 | 
			
		||||
#include "cryptonote_core/hardfork.h"
 | 
			
		||||
#include "cryptonote_basic/hardfork.h"
 | 
			
		||||
#include <boost/format.hpp>
 | 
			
		||||
#include <ctime>
 | 
			
		||||
#include <string>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -51,5 +51,7 @@ monero_add_library(mnemonics
 | 
			
		|||
  ${mnemonics_headers}
 | 
			
		||||
  ${mnemonics_private_headers})
 | 
			
		||||
target_link_libraries(mnemonics
 | 
			
		||||
  PUBLIC
 | 
			
		||||
    ${Boost_SYSTEM_LIBRARY}
 | 
			
		||||
  PRIVATE
 | 
			
		||||
    ${EXTRA_LIBRARIES})
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -38,6 +38,7 @@ source_group(p2p FILES ${P2P})
 | 
			
		|||
monero_add_library(p2p ${P2P})
 | 
			
		||||
target_link_libraries(p2p
 | 
			
		||||
  PUBLIC
 | 
			
		||||
    epee
 | 
			
		||||
    ${UPNP_LIBRARIES}
 | 
			
		||||
    ${Boost_CHRONO_LIBRARY}
 | 
			
		||||
    ${Boost_PROGRAM_OPTIONS_LIBRARY}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -49,5 +49,6 @@ target_link_libraries(ringct
 | 
			
		|||
  PUBLIC
 | 
			
		||||
    common
 | 
			
		||||
    crypto
 | 
			
		||||
    cryptonote_basic
 | 
			
		||||
  PRIVATE
 | 
			
		||||
    ${EXTRA_LIBRARIES})
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -34,7 +34,7 @@
 | 
			
		|||
#include "common/thread_group.h"
 | 
			
		||||
#include "common/util.h"
 | 
			
		||||
#include "rctSigs.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
 | 
			
		||||
using namespace crypto;
 | 
			
		||||
using namespace std;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -47,6 +47,7 @@ target_link_libraries(rpc
 | 
			
		|||
    cryptonote_core
 | 
			
		||||
    cryptonote_protocol
 | 
			
		||||
    epee
 | 
			
		||||
    ${Boost_REGEX_LIBRARY}
 | 
			
		||||
    ${Boost_THREAD_LIBRARY}
 | 
			
		||||
  PRIVATE
 | 
			
		||||
    ${EXTRA_LIBRARIES})
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -33,9 +33,9 @@ using namespace epee;
 | 
			
		|||
 | 
			
		||||
#include "core_rpc_server.h"
 | 
			
		||||
#include "common/command_line.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "misc_language.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "core_rpc_server_error_codes.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,8 +30,8 @@
 | 
			
		|||
 | 
			
		||||
#pragma once
 | 
			
		||||
#include "cryptonote_protocol/cryptonote_protocol_defs.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/difficulty.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/difficulty.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
 | 
			
		||||
namespace cryptonote
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -50,7 +50,7 @@
 | 
			
		|||
#include "p2p/net_node.h"
 | 
			
		||||
#include "cryptonote_protocol/cryptonote_protocol_handler.h"
 | 
			
		||||
#include "simplewallet.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "storages/http_abstract_invoke.h"
 | 
			
		||||
#include "rpc/core_rpc_server_commands_defs.h"
 | 
			
		||||
#include "crypto/crypto.h"  // for crypto::secret_key definition
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -40,8 +40,8 @@
 | 
			
		|||
#include <boost/optional/optional.hpp>
 | 
			
		||||
#include <boost/program_options/variables_map.hpp>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "wallet/wallet2.h"
 | 
			
		||||
#include "console_handler.h"
 | 
			
		||||
#include "wallet/password_container.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,9 +32,8 @@
 | 
			
		|||
#include "wallet.h"
 | 
			
		||||
#include "common_defines.h"
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
 | 
			
		||||
#include <memory>
 | 
			
		||||
#include <vector>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,9 +32,8 @@
 | 
			
		|||
#include "wallet.h"
 | 
			
		||||
#include "common_defines.h"
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
 | 
			
		||||
#include <memory>
 | 
			
		||||
#include <vector>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -39,10 +39,10 @@ using namespace epee;
 | 
			
		|||
#include "cryptonote_config.h"
 | 
			
		||||
#include "wallet2.h"
 | 
			
		||||
#include "wallet2_api.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "rpc/core_rpc_server_commands_defs.h"
 | 
			
		||||
#include "misc_language.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "common/boost_serialization_helper.h"
 | 
			
		||||
#include "common/command_line.h"
 | 
			
		||||
#include "profile_tools.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -39,13 +39,14 @@
 | 
			
		|||
#include <atomic>
 | 
			
		||||
 | 
			
		||||
#include "include_base_utils.h"
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/account_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_basic/account_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "net/http_client.h"
 | 
			
		||||
#include "storages/http_abstract_invoke.h"
 | 
			
		||||
#include "rpc/core_rpc_server_commands_defs.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_tx_utils.h"
 | 
			
		||||
#include "common/unordered_containers_boost_serialization.h"
 | 
			
		||||
#include "crypto/chacha8.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -35,7 +35,7 @@
 | 
			
		|||
#include <string>
 | 
			
		||||
#include <vector>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "rpc/core_rpc_server_commands_defs.h"
 | 
			
		||||
#include "include_base_utils.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -38,8 +38,8 @@ using namespace epee;
 | 
			
		|||
#include "common/command_line.h"
 | 
			
		||||
#include "common/i18n.h"
 | 
			
		||||
#include "common/util.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "wallet_rpc_server_commands_defs.h"
 | 
			
		||||
#include "misc_language.h"
 | 
			
		||||
#include "string_coding.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,7 +30,7 @@
 | 
			
		|||
 | 
			
		||||
#pragma once
 | 
			
		||||
#include "cryptonote_protocol/cryptonote_protocol_defs.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "wallet_rpc_server_error_codes.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,8 +32,8 @@
 | 
			
		|||
 | 
			
		||||
#include <boost/program_options/variables_map.hpp>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_core/verification_context.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/verification_context.h"
 | 
			
		||||
#include <unordered_map>
 | 
			
		||||
 | 
			
		||||
namespace tests
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -37,10 +37,10 @@
 | 
			
		|||
#include "console_handler.h"
 | 
			
		||||
 | 
			
		||||
#include "p2p/net_node.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_core/miner.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/miner.h"
 | 
			
		||||
 | 
			
		||||
#include "chaingen.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -44,12 +44,12 @@
 | 
			
		|||
#include "common/boost_serialization_helper.h"
 | 
			
		||||
#include "common/command_line.h"
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/account_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/account_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_core.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_boost_serialization.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_boost_serialization.h"
 | 
			
		||||
#include "misc_language.h"
 | 
			
		||||
 | 
			
		||||
#undef MONERO_DEFAULT_LOG_CATEGORY
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -35,8 +35,8 @@
 | 
			
		|||
 | 
			
		||||
#include "console_handler.h"
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
 | 
			
		||||
#include "chaingen.h"
 | 
			
		||||
#include "chaingen_tests_list.h"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -29,9 +29,9 @@
 | 
			
		|||
// Parts of this file are originally copyright (c) 2012-2013 The Cryptonote developers
 | 
			
		||||
 | 
			
		||||
#include "include_base_utils.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_tx_utils.h"
 | 
			
		||||
#include "misc_language.h"
 | 
			
		||||
 | 
			
		||||
using namespace cryptonote;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,7 +32,7 @@
 | 
			
		|||
#include <sstream>
 | 
			
		||||
#include "wallet/wallet.h"
 | 
			
		||||
#include "rpc/core_rpc_server.h"
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "net/http_client_abstract_invoke.h"
 | 
			
		||||
using namespace std;
 | 
			
		||||
using namespace epee::misc_utils;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -36,7 +36,7 @@
 | 
			
		|||
#include <algorithm>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_config.h"
 | 
			
		||||
#include "cryptonote_core/difficulty.h"
 | 
			
		||||
#include "cryptonote_basic/difficulty.h"
 | 
			
		||||
 | 
			
		||||
using namespace std;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -33,7 +33,7 @@
 | 
			
		|||
#include <cstring>
 | 
			
		||||
#include <limits>
 | 
			
		||||
#include "crypto/hash.h"
 | 
			
		||||
#include "cryptonote_core/difficulty.h"
 | 
			
		||||
#include "cryptonote_basic/difficulty.h"
 | 
			
		||||
 | 
			
		||||
using namespace std;
 | 
			
		||||
using cryptonote::check_hash;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,9 +32,9 @@
 | 
			
		|||
 | 
			
		||||
#include <vector>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_tx_utils.h"
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "ringct/rctSigs.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -31,7 +31,7 @@
 | 
			
		|||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
 | 
			
		||||
class test_cn_slow_hash
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,9 +30,9 @@
 | 
			
		|||
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_tx_utils.h"
 | 
			
		||||
 | 
			
		||||
#include "multi_tx_test_base.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -31,7 +31,7 @@
 | 
			
		|||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
 | 
			
		||||
#include "single_tx_test_base.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -31,7 +31,7 @@
 | 
			
		|||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
 | 
			
		||||
#include "single_tx_test_base.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -31,7 +31,7 @@
 | 
			
		|||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
 | 
			
		||||
#include "single_tx_test_base.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -31,7 +31,7 @@
 | 
			
		|||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
 | 
			
		||||
#include "single_tx_test_base.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -31,7 +31,7 @@
 | 
			
		|||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
 | 
			
		||||
#include "single_tx_test_base.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,9 +30,9 @@
 | 
			
		|||
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
 | 
			
		||||
#include "single_tx_test_base.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -31,7 +31,7 @@
 | 
			
		|||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
 | 
			
		||||
class test_generate_keypair
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,9 +30,9 @@
 | 
			
		|||
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_tx_utils.h"
 | 
			
		||||
 | 
			
		||||
#include "single_tx_test_base.h"
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,9 +32,9 @@
 | 
			
		|||
 | 
			
		||||
#include <vector>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_tx_utils.h"
 | 
			
		||||
#include "crypto/crypto.h"
 | 
			
		||||
 | 
			
		||||
template<size_t a_ring_size>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,9 +30,9 @@
 | 
			
		|||
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/account.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/account.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_tx_utils.h"
 | 
			
		||||
 | 
			
		||||
class single_tx_test_base
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -49,7 +49,7 @@ set(unit_tests_sources
 | 
			
		|||
  parse_amount.cpp
 | 
			
		||||
  serialization.cpp
 | 
			
		||||
  slow_memmem.cpp
 | 
			
		||||
  test_format_utils.cpp
 | 
			
		||||
  test_tx_utils.cpp
 | 
			
		||||
  test_peerlist.cpp
 | 
			
		||||
  test_protocol_pack.cpp
 | 
			
		||||
  thread_group.cpp
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -33,7 +33,7 @@
 | 
			
		|||
#include <cstdint>
 | 
			
		||||
 | 
			
		||||
#include "common/base58.cpp"
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
#include "serialization/binary_utils.h"
 | 
			
		||||
 | 
			
		||||
using namespace tools;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,7 +30,7 @@
 | 
			
		|||
 | 
			
		||||
#include "gtest/gtest.h"
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_basic_impl.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_basic_impl.h"
 | 
			
		||||
 | 
			
		||||
using namespace cryptonote;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -40,7 +40,7 @@
 | 
			
		|||
#ifdef BERKELEY_DB
 | 
			
		||||
#include "blockchain_db/berkeleydb/db_bdb.h"
 | 
			
		||||
#endif
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
 | 
			
		||||
using namespace cryptonote;
 | 
			
		||||
using epee::string_tools::pod_to_hex;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -29,7 +29,7 @@
 | 
			
		|||
// Parts of this file are originally copyright (c) 2012-2013 The Cryptonote developers
 | 
			
		||||
 | 
			
		||||
#include "gtest/gtest.h"
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
 | 
			
		||||
static const uint64_t valid_decomposed_outputs[] = {
 | 
			
		||||
  (uint64_t)1, (uint64_t)2, (uint64_t)3, (uint64_t)4, (uint64_t)5, (uint64_t)6, (uint64_t)7, (uint64_t)8, (uint64_t)9, // 1 piconero
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,7 +30,7 @@
 | 
			
		|||
 | 
			
		||||
#include "gtest/gtest.h"
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/checkpoints.cpp"
 | 
			
		||||
#include "cryptonote_basic/checkpoints.cpp"
 | 
			
		||||
 | 
			
		||||
using namespace cryptonote;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -33,7 +33,7 @@
 | 
			
		|||
#include <cstdint>
 | 
			
		||||
#include <vector>
 | 
			
		||||
 | 
			
		||||
#include "cryptonote_core/cryptonote_format_utils.h"
 | 
			
		||||
#include "cryptonote_basic/cryptonote_format_utils.h"
 | 
			
		||||
 | 
			
		||||
#define VEC_FROM_ARR(vec)                                               \
 | 
			
		||||
  std::vector<uint64_t> vec;                                            \
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,7 +32,7 @@
 | 
			
		|||
#include "gtest/gtest.h"
 | 
			
		||||
 | 
			
		||||
#include "blockchain_db/lmdb/db_lmdb.h"
 | 
			
		||||
#include "cryptonote_core/hardfork.h"
 | 
			
		||||
#include "cryptonote_basic/hardfork.h"
 | 
			
		||||
 | 
			
		||||
using namespace cryptonote;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
Some files were not shown because too many files have changed in this diff Show more
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue