mirror of
https://github.com/sockspls/badfish
synced 2025-05-01 09:13:08 +00:00
163 lines
No EOL
5 KiB
C++
163 lines
No EOL
5 KiB
C++
// Definition of layer Sum of NNUE evaluation function
|
|
|
|
#ifndef _NNUE_LAYERS_SUM_H_
|
|
#define _NNUE_LAYERS_SUM_H_
|
|
|
|
#if defined(EVAL_NNUE)
|
|
|
|
#include "../nnue_common.h"
|
|
|
|
namespace Eval {
|
|
|
|
namespace NNUE {
|
|
|
|
namespace Layers {
|
|
|
|
// Layer that sums the output of multiple layers
|
|
template <typename FirstPreviousLayer, typename... RemainingPreviousLayers>
|
|
class Sum : public Sum<RemainingPreviousLayers...> {
|
|
private:
|
|
using Head = FirstPreviousLayer;
|
|
using Tail = Sum<RemainingPreviousLayers...>;
|
|
|
|
public:
|
|
// Input/output type
|
|
using InputType = typename Head::OutputType;
|
|
using OutputType = InputType;
|
|
static_assert(std::is_same<InputType, typename Tail::InputType>::value, "");
|
|
|
|
// number of input/output dimensions
|
|
static constexpr IndexType kInputDimensions = Head::kOutputDimensions;
|
|
static constexpr IndexType kOutputDimensions = kInputDimensions;
|
|
static_assert(kInputDimensions == Tail::kInputDimensions ,"");
|
|
|
|
// Size of forward propagation buffer used in this layer
|
|
static constexpr std::size_t kSelfBufferSize =
|
|
CeilToMultiple(kOutputDimensions * sizeof(OutputType), kCacheLineSize);
|
|
|
|
// Size of the forward propagation buffer used from the input layer to this layer
|
|
static constexpr std::size_t kBufferSize =
|
|
std::max(Head::kBufferSize + kSelfBufferSize, Tail::kBufferSize);
|
|
|
|
// Hash value embedded in the evaluation function file
|
|
static constexpr std::uint32_t GetHashValue() {
|
|
std::uint32_t hash_value = 0xBCE400B4u;
|
|
hash_value ^= Head::GetHashValue() >> 1;
|
|
hash_value ^= Head::GetHashValue() << 31;
|
|
hash_value ^= Tail::GetHashValue() >> 2;
|
|
hash_value ^= Tail::GetHashValue() << 30;
|
|
return hash_value;
|
|
}
|
|
|
|
// A string that represents the structure from the input layer to this layer
|
|
static std::string GetStructureString() {
|
|
return "Sum[" +
|
|
std::to_string(kOutputDimensions) + "](" + GetSummandsString() + ")";
|
|
}
|
|
|
|
// read parameters
|
|
bool ReadParameters(std::istream& stream) {
|
|
if (!Tail::ReadParameters(stream)) return false;
|
|
return previous_layer_.ReadParameters(stream);
|
|
}
|
|
|
|
// write parameters
|
|
bool WriteParameters(std::ostream& stream) const {
|
|
if (!Tail::WriteParameters(stream)) return false;
|
|
return previous_layer_.WriteParameters(stream);
|
|
}
|
|
|
|
// forward propagation
|
|
const OutputType* Propagate(
|
|
const TransformedFeatureType* transformed_features, char* buffer) const {
|
|
Tail::Propagate(transformed_features, buffer);
|
|
const auto head_output = previous_layer_.Propagate(
|
|
transformed_features, buffer + kSelfBufferSize);
|
|
const auto output = reinterpret_cast<OutputType*>(buffer);
|
|
for (IndexType i = 0; i <kOutputDimensions; ++i) {
|
|
output[i] += head_output[i];
|
|
}
|
|
return output;
|
|
}
|
|
|
|
protected:
|
|
// A string that represents the list of layers to be summed
|
|
static std::string GetSummandsString() {
|
|
return Head::GetStructureString() + "," + Tail::GetSummandsString();
|
|
}
|
|
|
|
// Make the learning class a friend
|
|
friend class Trainer<Sum>;
|
|
|
|
// the layer immediately before this layer
|
|
FirstPreviousLayer previous_layer_;
|
|
};
|
|
|
|
// Layer that sums the output of multiple layers (when there is one template argument)
|
|
template <typename PreviousLayer>
|
|
class Sum<PreviousLayer> {
|
|
public:
|
|
// Input/output type
|
|
using InputType = typename PreviousLayer::OutputType;
|
|
using OutputType = InputType;
|
|
|
|
// number of input/output dimensions
|
|
static constexpr IndexType kInputDimensions =
|
|
PreviousLayer::kOutputDimensions;
|
|
static constexpr IndexType kOutputDimensions = kInputDimensions;
|
|
|
|
// Size of the forward propagation buffer used from the input layer to this layer
|
|
static constexpr std::size_t kBufferSize = PreviousLayer::kBufferSize;
|
|
|
|
// Hash value embedded in the evaluation function file
|
|
static constexpr std::uint32_t GetHashValue() {
|
|
std::uint32_t hash_value = 0xBCE400B4u;
|
|
hash_value ^= PreviousLayer::GetHashValue() >> 1;
|
|
hash_value ^= PreviousLayer::GetHashValue() << 31;
|
|
return hash_value;
|
|
}
|
|
|
|
// A string that represents the structure from the input layer to this layer
|
|
static std::string GetStructureString() {
|
|
return "Sum[" +
|
|
std::to_string(kOutputDimensions) + "](" + GetSummandsString() + ")";
|
|
}
|
|
|
|
// read parameters
|
|
bool ReadParameters(std::istream& stream) {
|
|
return previous_layer_.ReadParameters(stream);
|
|
}
|
|
|
|
// write parameters
|
|
bool WriteParameters(std::ostream& stream) const {
|
|
return previous_layer_.WriteParameters(stream);
|
|
}
|
|
|
|
// forward propagation
|
|
const OutputType* Propagate(
|
|
const TransformedFeatureType* transformed_features, char* buffer) const {
|
|
return previous_layer_.Propagate(transformed_features, buffer);
|
|
}
|
|
|
|
protected:
|
|
// A string that represents the list of layers to be summed
|
|
static std::string GetSummandsString() {
|
|
return PreviousLayer::GetStructureString();
|
|
}
|
|
|
|
// Make the learning class a friend
|
|
friend class Trainer<Sum>;
|
|
|
|
// the layer immediately before this layer
|
|
PreviousLayer previous_layer_;
|
|
};
|
|
|
|
} // namespace Layers
|
|
|
|
} // namespace NNUE
|
|
|
|
} // namespace Eval
|
|
|
|
#endif // defined(EVAL_NNUE)
|
|
|
|
#endif |