/* Copyright (c) Huawei Technologies Co., Ltd. 2024. All rights reserved.

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
        limitations under the License.
==============================================================================*/

#include "initializer.h"

#include <string>
#include "log/logger.h"
#include "constant_initializer/constant_initializer.h"
#include "random_normal_initializer/random_normal_initializer.h"
#include "truncated_normal_initializer/truncated_normal_initializer.h"

using namespace MxRec;

ConstantInitializerInfo::ConstantInitializerInfo(float constantValue, float initK)
    : constantValue(constantValue), initK(initK)
{}

NormalInitializerInfo::NormalInitializerInfo(float mean, float stddev, int seed, float initK)
    : mean(mean), stddev(stddev), seed(seed), initK(initK)
{}

InitializeInfo::InitializeInfo(string &name, int start, int len,
                               ConstantInitializerInfo constantInitializerInfo)
    : name(name), start(start), len(len), constantInitializerInfo(constantInitializerInfo)
{
    if (name == "constant_initializer") {
        initializerType = InitializerType::CONSTANT;
        initializer = make_shared<ConstantInitializer>(start, len, constantInitializerInfo.constantValue,
                                                       constantInitializerInfo.initK);
    } else {
        throw invalid_argument("Invalid Initializer Type.");
    }
}

InitializeInfo::InitializeInfo(string &name, int start, int len, NormalInitializerInfo normalInitializerInfo)
    : name(name), start(start), len(len), normalInitializerInfo(normalInitializerInfo)
{
    if (name == "truncated_normal_initializer") {
        initializerType = InitializerType::TRUNCATED_NORMAL;
        initializer = make_shared<TruncatedNormalInitializer>(start, len, normalInitializerInfo);
    } else if (name == "random_normal_initializer") {
        initializerType = InitializerType::RANDOM_NORMAL;
        initializer = make_shared<RandomNormalInitializer>(start, len, normalInitializerInfo);
    } else {
        throw invalid_argument("Invalid Initializer Type.");
    }
}

bool Initializer::RangeValidate(int begin, int lens)
{
    if (begin < 0) { // 新增:检查start非负
        LOG_ERROR("Start index cannot be negative: {}", begin);
        return false;
    }
    
    if (lens <= 0) { // 新增:检查len > 0
        LOG_ERROR("Length must be positive: {}", lens);
        return false;
    }

    // 检查start + len是否会整数溢出
    if (begin > std::numeric_limits<int>::max() - lens) {
        LOG_ERROR("Integer overflow detected: start {} + len {} exceeds maximum int value.", begin, lens);
        return false;
    }

    return true;
}