* Copyright (c) 2025 Huawei Technologies Co., Ltd.
* This program is free software, you can redistribute it and/or modify it under the terms and conditions of
* CANN Open Software License Agreement Version 2.0 (the "License").
* Please refer to the License for details. You may not use this file except in compliance with the License.
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED,
* INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY, OR FITNESS FOR A PARTICULAR PURPOSE.
* See LICENSE in the root of the software repository for the full text of the License.
*/
#include <sstream>
#include "ascir_ops.h"
#include "common_utils.h"
#include "common/ge_common/debug/log.h"
#include "graph/ascendc_ir/utils/asc_tensor_utils.h"
#include "common/checker.h"
#include "../utils/api_call_factory.h"
#include "../utils/api_call_utils.h"
#include "store_api_call.h"
using namespace af::ops;
using namespace af::ascir_op;
using namespace ascgen_utils;
namespace {
constexpr uint64_t kDmaMaxLen = 2U;
}
namespace codegen {
Status StoreApiCall::ParseAttr(const ascir::NodeView &node) {
repeats_ = node->outputs[0U].attr.repeats;
(void)node->attr.ir_attr->GetAttrValue("offset", offset_);
return ge::SUCCESS;
}
Status StoreApiCall::PreProcess(const TPipe &tpipe, const std::vector<ascir::AxisId> ¤t_axis,
const std::vector<std::reference_wrapper<const Tensor>> &outputs,
std::string &result) const {
GE_ASSERT_TRUE(!outputs.empty());
const_cast<Tensor &>(outputs.front().get()).axis_size = repeats_;
GE_ASSERT_SUCCESS(ApiCall::PreProcess(tpipe, current_axis, outputs, result));
return ge::SUCCESS;
}
Status StoreApiCall::Generate(const TPipe &tpipe, const std::vector<ascir::AxisId> ¤t_axis,
const std::vector<std::reference_wrapper<const Tensor>> &inputs,
const std::vector<std::reference_wrapper<const Tensor>> &outputs,
std::string &result) const {
std::stringstream ss;
const auto &gm = outputs[0].get();
const auto &ub = inputs[0].get();
DataCopyParams param;
bool status = CalculateDmaParams(tpipe, gm, gm, param);
GE_ASSERT_TRUE(status, "CalculateDmaParams failed");
std::string gm_offset = tpipe.tiler.Offset(current_axis, gm.axis, gm.axis_strides);
if (param.repeats.size() <= kDmaMaxLen) {
DmaParams dma_param;
SetDmaParams(tpipe, param, dma_param, false);
ss << "DataCopyPadExtend(" << gm << "[" << gm_offset << " + " << tpipe.tiler.Size(offset_) << "], " << ub << ", "
<< dma_param.block_count << ", " << dma_param.block_len << ", " << dma_param.src_stride << ", "
<< dma_param.dst_stride << ");" << std::endl;
} else {
CreateDmaCall(tpipe, ub, gm, gm_offset, param, offset_, ss, false);
}
if (IsUnitLastRead(*(this->inputs[0])) && ub.is_load_link_store_and_vec) {
std::string offset = offset_.Str().get();
offset = GenValidName(offset);
GenerateLinkStoreEventCode(ub, offset, ss);
}
result = ss.str();
return ge::SUCCESS;
}
static ApiCallRegister<StoreApiCall> register_store_api_call("StoreApiCall");
}