* 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 <gtest/gtest.h>
#include <cmath>
#include <fstream>
#include <stdlib.h>
#include "securec.h"
#include <fcntl.h>
#include <ctime>
#include <string>
#include <sys/mman.h>
#include <unistd.h>
#include "ascendc_pack_kernel.h"
#include "mockcpp/mockcpp.hpp"
std::string GetElfString()
{
std::ifstream file("../../../../tests/tools/utils/elf_tool/elf_tools.txt");
std::stringstream buffer;
std::string content;
if (!file) {
std::cerr << "Cannot open elf_tools.txt\n";
return "";
}
buffer << file.rdbuf();
content = buffer.str();
file.close();
return content;
}
class TEST_PACKKERNEL : public testing::Test {
protected:
void SetUp() {}
void TearDown() { GlobalMockObject::verify(); }
};
uint64_t GetTime()
{
struct timespec ts {};
(void)clock_gettime(CLOCK_MONOTONIC, &ts);
uint64_t milliseconds = (ts.tv_sec * 1000) + (ts.tv_nsec / 1000000);
return milliseconds;
}
bool FileExists(std::string fileName)
{
struct stat buffer;
return stat(fileName.c_str(), &buffer) == 0;
}
std::string GetFileName()
{
std::string fileName = "/tmp/tmpfile_" + std::to_string(GetTime()) + "_" + std::to_string(getpid());
while (FileExists(fileName)) {
fileName = "/tmp/tmpfile_" + std::to_string(GetTime()) + "_" + std::to_string(getpid());
}
return fileName;
}
void GmAllocTemp(std::string fileName)
{
size_t size = 10 * 1024 * 1024;
int fd = open(fileName.c_str(), O_RDWR | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR);
int res = ftruncate(fd, size);
if (res != 0) {
std::cout << "Error to change file size!" << std::endl;
raise(SIGABRT);
}
close(fd);
}
void GetElfBuffer(const char* str, char* buf, int& len)
{
char b2c[16] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};
char c2b[256];
for (int i = 0; i < 16; i++) {
c2b[b2c[i]] = i;
}
int nlen = strlen(str);
for (int i = 0; i < nlen; i += 2) {
buf[i / 2] = c2b[str[i]];
buf[i / 2] |= c2b[str[i + 1]] << 4;
}
len = nlen / 2;
}
TEST_F(TEST_PACKKERNEL, ascendcPackKernelMainTest)
{
std::string src0fileName = GetFileName();
GmAllocTemp(src0fileName);
std::string src1fileName = GetFileName();
GmAllocTemp(src1fileName);
std::string dstfileName = GetFileName();
GmAllocTemp(dstfileName);
char* src0 = new char[src0fileName.length() + 1];
EXPECT_TRUE(src0 != nullptr);
strcpy(src0, src0fileName.c_str());
char* src1 = new char[src1fileName.length() + 1];
EXPECT_TRUE(src1 != nullptr);
strcpy(src1, src1fileName.c_str());
char* dst = new char[dstfileName.length() + 1];
EXPECT_TRUE(dst != nullptr);
strcpy(dst, dstfileName.c_str());
std::string elf_res_str = GetElfString();
const char* elf_tools_test = elf_res_str.c_str();
char buf[618433];
int len;
GetElfBuffer(elf_tools_test, buf, len);
bool ret = WriteFile(src0, buf, len);
EXPECT_TRUE(ret);
ret = WriteFile(src1, buf, len);
EXPECT_TRUE(ret);
char* args[] = {"xxx", src0, src1, "0", dst};
AscendcPackKernelMain(0x5, args);
delete[] src0;
delete[] src1;
delete[] dst;
}
int32_t WrFileStub(int32_t input) { return sizeof(const char*); }
TEST_F(TEST_PACKKERNEL, ascendcPackKernelWRFailTest)
{
std::string src0fileName = GetFileName();
GmAllocTemp(src0fileName);
std::string src1fileName = GetFileName();
GmAllocTemp(src1fileName);
std::string dstfileName = GetFileName();
GmAllocTemp(dstfileName);
char* src0 = new char[src0fileName.length() + 1];
EXPECT_TRUE(src0 != nullptr);
strcpy(src0, src0fileName.c_str());
char* src1 = new char[src1fileName.length() + 1];
EXPECT_TRUE(src1 != nullptr);
strcpy(src1, src1fileName.c_str());
char buf[618433];
int len;
std::string elf_res_str = GetElfString();
const char* elf_tools_test = elf_res_str.c_str();
GetElfBuffer(elf_tools_test, buf, len);
MOCKER(write).stubs().will(invoke(WrFileStub));
bool ret = WriteFile(src0, buf, len);
EXPECT_TRUE(ret);
MOCKER(read).stubs().will(invoke(WrFileStub));
ret = ReadFile(src1, buf, len);
EXPECT_TRUE(ret);
delete[] src0;
delete[] src1;
}
TEST_F(TEST_PACKKERNEL, ascendcPackKernelFseekError)
{
std::string src0fileName = "/tmp/tmp_ascendcPackKernelFseekError.bin";
GmAllocTemp(src0fileName);
MOCKER(fseek).stubs().will(returnValue(1));
size_t retSize = GetFileSize(src0fileName.c_str());
GlobalMockObject::verify();
EXPECT_EQ(retSize, 0);
}
TEST_F(TEST_PACKKERNEL, ascendcPackKernelFopenError)
{
std::string src0fileName = "/tmp/xxxxxeeeedddddd.bin";
size_t retSize = GetFileSize(src0fileName.c_str());
EXPECT_EQ(retSize, 0);
}
TEST_F(TEST_PACKKERNEL, ascendcPackKernelFileIoRoundTrip)
{
std::string fileName = GetFileName();
const char writeBuf[] = "ascendc-pack-kernel";
char readBuf[sizeof(writeBuf)] = {0};
size_t writeSize = WriteFile(fileName.c_str(), const_cast<char*>(writeBuf), sizeof(writeBuf));
EXPECT_EQ(writeSize, sizeof(writeBuf));
EXPECT_EQ(GetFileSize(fileName.c_str()), sizeof(writeBuf));
size_t readSize = ReadFile(fileName.c_str(), readBuf, sizeof(readBuf));
EXPECT_EQ(readSize, sizeof(writeBuf));
EXPECT_STREQ(readBuf, writeBuf);
(void)unlink(fileName.c_str());
}
TEST_F(TEST_PACKKERNEL, ascendcPackKernelErrorType)
{
std::string src0fileName = GetFileName();
GmAllocTemp(src0fileName);
std::string src1fileName = GetFileName();
GmAllocTemp(src1fileName);
std::string dstfileName = GetFileName();
GmAllocTemp(dstfileName);
char* src0 = new char[src0fileName.length() + 1];
strcpy(src0, src0fileName.c_str());
char* src1 = new char[src1fileName.length() + 1];
strcpy(src1, src1fileName.c_str());
char* dst = new char[dstfileName.length() + 1];
strcpy(dst, dstfileName.c_str());
char buf[618433];
int len;
std::string elf_res_str = GetElfString();
const char* elf_tools_test = elf_res_str.c_str();
GetElfBuffer(elf_tools_test, buf, len);
bool ret = WriteFile(src0, buf, len);
ret = WriteFile(src1, buf, len);
char* args[] = {"xxx", src0, src1, "100", dst};
int retMain = AscendcPackKernelMain(0x5, args);
EXPECT_EQ(retMain, 1);
delete[] src0;
delete[] src1;
delete[] dst;
}
TEST_F(TEST_PACKKERNEL, ascendcPackKernelErrorArgs)
{
std::string src0fileName = GetFileName();
GmAllocTemp(src0fileName);
std::string src1fileName = GetFileName();
GmAllocTemp(src1fileName);
std::string dstfileName = GetFileName();
GmAllocTemp(dstfileName);
char* src0 = new char[src0fileName.length() + 1];
strcpy(src0, src0fileName.c_str());
char* src1 = new char[src1fileName.length() + 1];
strcpy(src1, src1fileName.c_str());
char* dst = new char[dstfileName.length() + 1];
strcpy(dst, dstfileName.c_str());
char buf[618433];
int len;
std::string elf_res_str = GetElfString();
const char* elf_tools_test = elf_res_str.c_str();
GetElfBuffer(elf_tools_test, buf, len);
bool ret = WriteFile(src0, buf, len);
ret = WriteFile(src1, buf, len);
char* args[] = {"xxx", src0, src1, "100"};
int retMain = AscendcPackKernelMain(0x4, args);
EXPECT_EQ(retMain, 1);
delete[] src0;
delete[] src1;
delete[] dst;
}