Fix convert async dump files failed issue and refactor convert_async.py
This commit is contained in:
parent
5a851daf2f
commit
b17b2bc687
|
@ -26,6 +26,7 @@
|
||||||
#include <unordered_set>
|
#include <unordered_set>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include "pybind11/embed.h"
|
#include "pybind11/embed.h"
|
||||||
|
#include "pybind11/stl.h"
|
||||||
#ifdef ONLINE_DBG_MODE
|
#ifdef ONLINE_DBG_MODE
|
||||||
#include "debug/common.h"
|
#include "debug/common.h"
|
||||||
#include "debug/debugger/debugger.h"
|
#include "debug/debugger/debugger.h"
|
||||||
|
@ -549,6 +550,7 @@ void DebugServices::ConvertToHostFormat(const std::map<std::string, std::vector<
|
||||||
std::string file_format = "npy";
|
std::string file_format = "npy";
|
||||||
for (auto const &d : dir_to_files_map) {
|
for (auto const &d : dir_to_files_map) {
|
||||||
std::vector<std::string> files_to_convert_in_dir;
|
std::vector<std::string> files_to_convert_in_dir;
|
||||||
|
std::vector<std::string> files_after_convert_in_dir;
|
||||||
std::string dump_key = d.first;
|
std::string dump_key = d.first;
|
||||||
for (auto const &file_name : d.second) {
|
for (auto const &file_name : d.second) {
|
||||||
bool already_converted = false;
|
bool already_converted = false;
|
||||||
|
@ -567,26 +569,19 @@ void DebugServices::ConvertToHostFormat(const std::map<std::string, std::vector<
|
||||||
}
|
}
|
||||||
if (!already_converted) {
|
if (!already_converted) {
|
||||||
files_to_convert_in_dir.push_back(dump_key + "/" + file_name);
|
files_to_convert_in_dir.push_back(dump_key + "/" + file_name);
|
||||||
|
files_after_convert_in_dir.push_back(dump_key + "/" + file_name_without_scope);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
std::ostringstream input_file_o;
|
MS_LOG(INFO) << "Number of files to convert: " << files_to_convert_in_dir.size();
|
||||||
const char *const delim = " ";
|
if (!files_to_convert_in_dir.empty()) {
|
||||||
std::copy(files_to_convert_in_dir.begin(), files_to_convert_in_dir.end(),
|
|
||||||
std::ostream_iterator<std::string>(input_file_o, delim));
|
|
||||||
std::string input_files = input_file_o.str();
|
|
||||||
MS_LOG(INFO) << "Ops to convert: " << input_files;
|
|
||||||
if (input_files != "") {
|
|
||||||
// Look for the installation path to the conver_async package. If not found, throw exception and terminate the
|
// Look for the installation path to the conver_async package. If not found, throw exception and terminate the
|
||||||
// later task.
|
// later task.
|
||||||
try {
|
try {
|
||||||
auto pkg = pybind11::module::import("mindspore.offline_debug.convert_async");
|
auto pkg = pybind11::module::import("mindspore.offline_debug.convert_async");
|
||||||
std::string convert_pkg_path = pkg.attr("__file__").cast<std::string>();
|
auto convert_obj = pkg.attr("AsyncDumpConverter")(pybind11::cast(files_to_convert_in_dir), dump_key);
|
||||||
MS_LOG(INFO) << "The file for converting async dump data is in " << convert_pkg_path;
|
(void)convert_obj.attr("convert_files")();
|
||||||
std::string convert_command = "python " + convert_pkg_path + " -out " + dump_key + " -t " + file_format +
|
|
||||||
" -d " + dump_key + " -f NCHW -l " + input_files;
|
|
||||||
(void)(system(convert_command.c_str()) + 1);
|
|
||||||
} catch (pybind11::error_already_set &e) {
|
} catch (pybind11::error_already_set &e) {
|
||||||
MS_LOG(EXCEPTION) << "Can't find package mindspore.offline_debug.convert_async";
|
MS_LOG(EXCEPTION) << "Failed to convert async dump data: " << e.what();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string abspath = RealPath(dump_key);
|
std::string abspath = RealPath(dump_key);
|
||||||
|
@ -599,7 +594,7 @@ void DebugServices::ConvertToHostFormat(const std::map<std::string, std::vector<
|
||||||
while ((dir = readdir(d_handle)) != nullptr) {
|
while ((dir = readdir(d_handle)) != nullptr) {
|
||||||
if (dir->d_type == DT_REG) {
|
if (dir->d_type == DT_REG) {
|
||||||
std::string candidate = dir->d_name;
|
std::string candidate = dir->d_name;
|
||||||
for (const std::string &file_to_find : files_to_convert_in_dir) {
|
for (const std::string &file_to_find : files_after_convert_in_dir) {
|
||||||
std::string file_n = file_to_find.substr(file_to_find.find_last_of("\\/") + 1);
|
std::string file_n = file_to_find.substr(file_to_find.find_last_of("\\/") + 1);
|
||||||
if (candidate.find(file_n) != std::string::npos && candidate.rfind(file_format) != std::string::npos) {
|
if (candidate.find(file_n) != std::string::npos && candidate.rfind(file_format) != std::string::npos) {
|
||||||
// we found a converted file for this op
|
// we found a converted file for this op
|
||||||
|
|
|
@ -12,30 +12,167 @@
|
||||||
# See the License for the specific language governing permissions and
|
# See the License for the specific language governing permissions and
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
# ============================================================================
|
# ============================================================================
|
||||||
"""Module to provide conversion capabalities from .timestamp async dump files to .npy."""
|
"""
|
||||||
import site
|
Module to provide conversion capabalities from .timestamp async dump files to .npy.
|
||||||
|
It's an internal module for debugger backend but not exposed to users.
|
||||||
|
"""
|
||||||
import os
|
import os
|
||||||
DIR_PATH = "/usr/local/Ascend/toolkit/tools/operator_cmp/compare/"
|
import glob
|
||||||
if not os.path.exists(DIR_PATH):
|
import stat
|
||||||
raise ValueError("Directory " + DIR_PATH + " does not exist. Please install Ascend toolkit.")
|
import sys
|
||||||
site.addsitedir(DIR_PATH)
|
from pathlib import Path
|
||||||
#pylint: disable=wrong-import-position
|
from importlib import import_module
|
||||||
import argparse
|
from collections import namedtuple
|
||||||
import csv
|
|
||||||
from dump_data_parser import DumpDataParser
|
import numpy as np
|
||||||
from shape_conversion import FormatConversionMain
|
|
||||||
import utils
|
|
||||||
#pylint: enable=wrong-import-position
|
|
||||||
|
|
||||||
|
|
||||||
def handle_multi_process(convert_obj, files):
|
class ConvertToolLoader:
|
||||||
"""Convert async format files to npy in a multithreaded manner"""
|
"""Module to load CANN conversion tool."""
|
||||||
#pylint: disable=W0212
|
|
||||||
return_code = utils.VECTOR_COMPARISON_NONE_ERROR
|
def __init__(self):
|
||||||
convert_obj.progress = utils.Progress(len(files))
|
self.utils = None
|
||||||
|
self.common = None
|
||||||
|
self.dump_data_parser = None
|
||||||
|
self.format_conversion = None
|
||||||
|
self.load_convert_tool()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def find_toolkit_path():
|
||||||
|
"""Find the path to Ascend toolkit."""
|
||||||
|
ascend_install_path = "/usr/local/Ascend"
|
||||||
|
if not os.path.exists(ascend_install_path):
|
||||||
|
ascend_toolkit_path = os.getenv("ASCEND_TOOLKIT_PATH")
|
||||||
|
if not ascend_toolkit_path:
|
||||||
|
raise ValueError(
|
||||||
|
"Failed to get $ASCEND_TOOLKIT_PATH in environment. Please install run packages " +
|
||||||
|
"and set the environment variable correctly.")
|
||||||
|
ascend_install_path = ascend_toolkit_path
|
||||||
|
ascend_install_path = Path(ascend_install_path).resolve()
|
||||||
|
msaccucmp_file_list = list(ascend_install_path.rglob('msaccucmp.py*'))
|
||||||
|
if not msaccucmp_file_list:
|
||||||
|
raise ValueError("Failed to find msaccucmp.py or msaccucmp.pyc file under " +
|
||||||
|
ascend_install_path + ". Please install Ascend toolkit.")
|
||||||
|
return msaccucmp_file_list[0].parent
|
||||||
|
|
||||||
|
def load_convert_tool(self):
|
||||||
|
"""load CANN conversion tool from the toolkit path."""
|
||||||
|
toolkit_path = self.find_toolkit_path()
|
||||||
|
if str(toolkit_path) not in sys.path:
|
||||||
|
sys.path.append(str(toolkit_path))
|
||||||
|
try:
|
||||||
|
self.utils = import_module('utils')
|
||||||
|
self.common = import_module('common')
|
||||||
|
self.dump_data_parser = import_module(
|
||||||
|
'dump_data_parser').DumpDataParser
|
||||||
|
self.format_conversion = import_module(
|
||||||
|
'shape_conversion').FormatConversionMain
|
||||||
|
except ModuleNotFoundError:
|
||||||
|
raise ModuleNotFoundError(
|
||||||
|
"Failed to load CANN conversion tools under " + toolkit_path + ". Please make sure Ascend " +
|
||||||
|
"toolkit has been installed properly.")
|
||||||
|
|
||||||
|
|
||||||
|
def parse_args(file_list, output_path):
|
||||||
|
"""Helper function to parse the input argument for the conversion configuration."""
|
||||||
|
args_dict = dict()
|
||||||
|
args_dict['dump_version'] = '2.0'
|
||||||
|
args_dict['format'] = 'NCHW'
|
||||||
|
args_dict['output_file_type'] = 'npy'
|
||||||
|
args_dict['dump_path'] = output_path
|
||||||
|
args_dict['output_path'] = output_path
|
||||||
|
args_dict['file_list'] = file_list
|
||||||
|
args_dict['input'] = None
|
||||||
|
args_dict['output'] = None
|
||||||
|
args_dict['shape'] = None
|
||||||
|
args_dict['custom_script_path'] = None
|
||||||
|
args_parser = namedtuple("args_parser", args_dict.keys())
|
||||||
|
return args_parser(**args_dict)
|
||||||
|
|
||||||
|
|
||||||
|
class AsyncDumpConverter:
|
||||||
|
"""Convert the target async dump data into npy files."""
|
||||||
|
|
||||||
|
def __init__(self, file_list, output_path):
|
||||||
|
# check input path
|
||||||
|
for file_item in file_list:
|
||||||
|
file_item = os.path.realpath(file_item)
|
||||||
|
output_path = os.path.realpath(output_path)
|
||||||
|
|
||||||
|
self.convert_tool = ConvertToolLoader()
|
||||||
|
self.args = parse_args(file_list, output_path)
|
||||||
|
self.files_to_convert = self.args.file_list
|
||||||
|
self.output_path = self.args.output_path
|
||||||
|
self.failed_file_path = os.path.join(
|
||||||
|
self.output_path, 'convert_failed_file_list.txt')
|
||||||
|
self.clear_failed_list_file()
|
||||||
|
|
||||||
|
def clear_failed_list_file(self):
|
||||||
|
"""Remove existing failed txt file."""
|
||||||
|
if self.failed_file_path and os.path.exists(self.failed_file_path):
|
||||||
|
os.remove(self.failed_file_path)
|
||||||
|
|
||||||
|
def convert_files(self):
|
||||||
|
"""Main entry of the converter to convert async dump files into npy format."""
|
||||||
|
self.convert_tool.utils.print_info_log('Start to convert async dump files.')
|
||||||
|
ret_code = self.convert_tool.utils.VECTOR_COMPARISON_NONE_ERROR
|
||||||
|
if self.args.format is not None:
|
||||||
|
convert = self.convert_tool.format_conversion(self.args)
|
||||||
|
else:
|
||||||
|
convert = self.convert_tool.dump_data_parser(self.args)
|
||||||
|
ret_code = self.handle_multi_process(convert, self.files_to_convert)
|
||||||
|
self._rename_generated_npy_files()
|
||||||
|
if ret_code != self.convert_tool.utils.VECTOR_COMPARISON_NONE_ERROR:
|
||||||
|
if os.path.exists(self.failed_file_path):
|
||||||
|
self.convert_failed_tensors()
|
||||||
|
self.convert_tool.utils.print_info_log('Finish to convert async dump files.')
|
||||||
|
|
||||||
|
def convert_failed_tensors(self):
|
||||||
|
"""Convert the failed tensor recorded in the failed txt file."""
|
||||||
|
self.convert_tool.utils.print_info_log(
|
||||||
|
'Start to convert failed tensors recorded in ' + self.failed_file_path + '.')
|
||||||
|
with open(self.failed_file_path) as failed_lines:
|
||||||
|
for failed_line in failed_lines:
|
||||||
|
try:
|
||||||
|
failed_line_list = failed_line.rstrip().split(',')
|
||||||
|
self.convert_one_failed_tensor(failed_line_list)
|
||||||
|
except (ValueError, OSError, AttributeError, self.convert_tool.utils.CompareError) as err:
|
||||||
|
self.convert_tool.utils.print_error_log(
|
||||||
|
'Failed to convert ' + failed_line + ' to Host format: ' + str(err))
|
||||||
|
|
||||||
|
def convert_one_failed_tensor(self, failed_tensor):
|
||||||
|
"""Convert failed operator one by one."""
|
||||||
|
if len(failed_tensor) <= 1:
|
||||||
|
raise ValueError(
|
||||||
|
"Invalid tensor info in convert_failed_file_list.txt")
|
||||||
|
file_path = failed_tensor[0]
|
||||||
|
type_index = failed_tensor[1:]
|
||||||
|
op_data = self.convert_tool.utils.parse_dump_file(
|
||||||
|
file_path, self.args.dump_version)
|
||||||
|
for type_index_item in type_index:
|
||||||
|
tensor_type, index = type_index_item.split(':')
|
||||||
|
index = int(index)
|
||||||
|
tensor = getattr(op_data, tensor_type)[index]
|
||||||
|
dump_data_array = self.convert_tool.utils.deserialize_dump_data_to_array(tensor)
|
||||||
|
array = dump_data_array.reshape(tensor.shape.dim)
|
||||||
|
self._save_tensor_to_npy_file(
|
||||||
|
file_path, tensor_type, index, tensor.format, array)
|
||||||
|
|
||||||
|
def handle_multi_process(self, convert_obj, files):
|
||||||
|
"""Convert async format files to npy in a multithreaded manner."""
|
||||||
|
return_code = self.convert_tool.utils.VECTOR_COMPARISON_NONE_ERROR
|
||||||
|
# try looking for function in compatibility with the toolkit package version.
|
||||||
|
if hasattr(convert_obj, 'multi_process'):
|
||||||
|
_ = setattr(convert_obj.multi_process, '_progress', self.convert_tool.utils.Progress(len(files)))
|
||||||
|
else:
|
||||||
|
_ = setattr(convert_obj, 'progress', self.convert_tool.utils.Progress(len(files)))
|
||||||
multi_process_file_list = []
|
multi_process_file_list = []
|
||||||
big_file_list = []
|
big_file_list = []
|
||||||
max_file_size = convert_obj._get_max_file_size()
|
max_file_size = 0
|
||||||
|
if hasattr(convert_obj, 'multi_process'):
|
||||||
|
max_file_size = getattr(convert_obj.multi_process, 'get_max_file_size')()
|
||||||
|
else:
|
||||||
|
max_file_size = getattr(convert_obj, '_get_max_file_size')()
|
||||||
for cur_file in files:
|
for cur_file in files:
|
||||||
cur_path = cur_file
|
cur_path = cur_file
|
||||||
if os.path.isfile(cur_path):
|
if os.path.isfile(cur_path):
|
||||||
|
@ -43,61 +180,61 @@ def handle_multi_process(convert_obj, files):
|
||||||
big_file_list.append(cur_path)
|
big_file_list.append(cur_path)
|
||||||
else:
|
else:
|
||||||
multi_process_file_list.append(cur_path)
|
multi_process_file_list.append(cur_path)
|
||||||
|
|
||||||
if multi_process_file_list:
|
if multi_process_file_list:
|
||||||
ret = convert_obj._do_multi_process(multi_process_file_list)
|
ret_mp = self.convert_tool.utils.VECTOR_COMPARISON_NONE_ERROR
|
||||||
if ret != utils.VECTOR_COMPARISON_NONE_ERROR:
|
if hasattr(convert_obj, 'multi_process'):
|
||||||
return_code = ret
|
ret_mp = getattr(convert_obj.multi_process, '_do_multi_process')(multi_process_file_list)
|
||||||
|
else:
|
||||||
|
ret_mp = getattr(convert_obj, '_do_multi_process')(multi_process_file_list)
|
||||||
|
if ret_mp != self.convert_tool.utils.VECTOR_COMPARISON_NONE_ERROR:
|
||||||
|
return_code = ret_mp
|
||||||
for big_file in big_file_list:
|
for big_file in big_file_list:
|
||||||
ret, _ = convert_obj.convert_format_for_one_file(big_file)
|
ret_bf = self.convert_tool.utils.VECTOR_COMPARISON_NONE_ERROR
|
||||||
convert_obj._handle_result_callback([ret, big_file])
|
if hasattr(convert_obj, '_convert_format_for_one_file'):
|
||||||
if ret != utils.VECTOR_COMPARISON_NONE_ERROR:
|
ret_bf, _ = getattr(convert_obj, '_convert_format_for_one_file')(big_file)
|
||||||
return_code = ret
|
else:
|
||||||
|
ret_bf, _ = getattr(convert_obj, 'convert_format_for_one_file')(big_file)
|
||||||
if return_code != utils.VECTOR_COMPARISON_NONE_ERROR:
|
if hasattr(convert_obj, 'multi_process'):
|
||||||
error_file_path = os.path.join(
|
getattr(convert_obj.multi_process, '_handle_result_callback')([ret_bf, big_file])
|
||||||
convert_obj.output_path, utils.CONVERT_FAILED_FILE_LIST_NAME)
|
else:
|
||||||
if os.path.exists(error_file_path):
|
getattr(convert_obj, '_handle_result_callback')([ret_bf, big_file])
|
||||||
utils.print_info_log(
|
if ret_bf != self.convert_tool.utils.VECTOR_COMPARISON_NONE_ERROR:
|
||||||
'The list of file that failed to convert has been written to "' + error_file_path + '".')
|
return_code = ret_bf
|
||||||
# pylint: enable=W0212
|
if return_code != self.convert_tool.utils.VECTOR_COMPARISON_NONE_ERROR:
|
||||||
|
if os.path.exists(self.failed_file_path):
|
||||||
|
self.convert_tool.utils.print_info_log(
|
||||||
|
'The list of file that failed to convert has been written to "'
|
||||||
|
+ self.failed_file_path + '".')
|
||||||
return return_code
|
return return_code
|
||||||
|
|
||||||
if __name__ == "__main__":
|
def _save_tensor_to_npy_file(self, file_path, tensor_type, idx, tensor_format, dump_data_array):
|
||||||
convert_parser = argparse.ArgumentParser()
|
"""Save tensor file into npy format."""
|
||||||
convert_parser.add_argument(
|
file_name = os.path.basename(file_path)
|
||||||
'-d', '--dump_file', dest='dump_path', default='', required=True)
|
name_splits = file_name.split('.')
|
||||||
convert_parser.add_argument(
|
name_splits[1] = name_splits[1].split('_')[-1]
|
||||||
'-l', '--file_list', nargs="*", dest='file_list', default='')
|
file_name_no_scope = '.'.join(name_splits)
|
||||||
convert_parser.add_argument('-f', '--format', dest='format', default=None)
|
out_file_name = "%s.%s.%d.%s.npy" % (
|
||||||
convert_parser.add_argument(
|
file_name_no_scope,
|
||||||
'-v', '--version', dest='dump_version', choices=[1, 2], type=int, default=2)
|
tensor_type,
|
||||||
convert_parser.add_argument('-s', '--shape', dest='shape', default=None)
|
idx,
|
||||||
convert_parser.add_argument('-o', '--output_tensor',
|
self.convert_tool.common.get_format_string(tensor_format)
|
||||||
dest='output', default=None)
|
)
|
||||||
convert_parser.add_argument('-i', '--input_tensor', dest='input', default=None)
|
out_path = os.path.join(self.output_path, out_file_name)
|
||||||
convert_parser.add_argument(
|
np.save(out_path, dump_data_array)
|
||||||
'-c', '--custom_script_path', dest='custom_script_path', default=None)
|
os.chmod(out_path, stat.S_IRUSR)
|
||||||
convert_parser.add_argument('-out', '--output', dest='output_path', default='')
|
|
||||||
convert_parser.add_argument(
|
|
||||||
'-t', '--type', dest='output_file_type', choices=['npy', 'bin'], default='npy')
|
|
||||||
|
|
||||||
args = convert_parser.parse_args()
|
def _rename_generated_npy_files(self):
|
||||||
dump_failed = os.path.abspath(args.dump_path) + "/convert_failed_file_list.txt"
|
"""In order to follow dump naming convention, rename npy files generated by CANN conversion tool."""
|
||||||
if os.path.exists(dump_failed):
|
target_file_list = []
|
||||||
os.remove(dump_failed)
|
for in_file in self.files_to_convert:
|
||||||
file_list = args.file_list
|
target_file_list.extend(glob.glob(in_file + "*.npy"))
|
||||||
if args.format is not None:
|
for target_file in target_file_list:
|
||||||
convert = FormatConversionMain(args)
|
old_filename = os.path.basename(target_file)
|
||||||
else:
|
name_splits = old_filename.split('.')
|
||||||
convert = DumpDataParser(args)
|
name_splits[1] = name_splits[1].split('_')[-1]
|
||||||
if args.file_list == "":
|
name_splits[-2] = self.args.format
|
||||||
file_list = os.listdir(args.dump_path)
|
new_file_name = '.'.join(name_splits)
|
||||||
handle_multi_process(convert, file_list)
|
out_path = os.path.join(self.output_path, new_file_name)
|
||||||
if os.path.exists(dump_failed):
|
os.rename(target_file, out_path)
|
||||||
with open(dump_failed, newline='') as failed_ops:
|
os.chmod(out_path, stat.S_IRUSR)
|
||||||
file_reader = csv.reader(failed_ops, delimiter=',')
|
self.convert_tool.utils.print_info_log("Rename file " + target_file + " to " + out_path)
|
||||||
file_list = [os.path.abspath(row[0]) for row in file_reader]
|
|
||||||
args.format = None
|
|
||||||
convert = DumpDataParser(args)
|
|
||||||
handle_multi_process(convert, file_list)
|
|
||||||
|
|
|
@ -1,12 +0,0 @@
|
||||||
{
|
|
||||||
"common_dump_settings": {
|
|
||||||
"dump_mode": 0,
|
|
||||||
"path": "/test",
|
|
||||||
"net_name": "Net",
|
|
||||||
"iteration": "0",
|
|
||||||
"input_output": 2,
|
|
||||||
"kernels": ["Default/TensorAdd-op3"],
|
|
||||||
"support_device": [0,1,2,3,4,5,6,7],
|
|
||||||
"op_debug_mode": 0
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,90 @@
|
||||||
|
# Copyright 2021 Huawei Technologies Co., Ltd
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
# ============================================================================
|
||||||
|
"""
|
||||||
|
Utils for testing dump feature.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import json
|
||||||
|
|
||||||
|
async_dump_dict = {
|
||||||
|
"common_dump_settings": {
|
||||||
|
"dump_mode": 0,
|
||||||
|
"path": "",
|
||||||
|
"net_name": "Net",
|
||||||
|
"iteration": "0",
|
||||||
|
"input_output": 2,
|
||||||
|
"kernels": ["Default/TensorAdd-op3"],
|
||||||
|
"support_device": [0, 1, 2, 3, 4, 5, 6, 7],
|
||||||
|
"op_debug_mode": 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
e2e_dump_dict = {
|
||||||
|
"common_dump_settings": {
|
||||||
|
"dump_mode": 0,
|
||||||
|
"path": "",
|
||||||
|
"net_name": "Net",
|
||||||
|
"iteration": "0",
|
||||||
|
"input_output": 0,
|
||||||
|
"kernels": ["Default/Conv-op12"],
|
||||||
|
"support_device": [0, 1, 2, 3, 4, 5, 6, 7],
|
||||||
|
"op_debug_mode": 0
|
||||||
|
},
|
||||||
|
"e2e_dump_settings": {
|
||||||
|
"enable": True,
|
||||||
|
"trans_flag": False
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async_dump_dict_2 = {
|
||||||
|
"common_dump_settings": {
|
||||||
|
"dump_mode": 0,
|
||||||
|
"path": "/tmp/async_dump/test_async_dump_net_multi_layer_mode1",
|
||||||
|
"net_name": "test",
|
||||||
|
"iteration": "0",
|
||||||
|
"input_output": 2,
|
||||||
|
"kernels": [
|
||||||
|
"default/TensorAdd-op10",
|
||||||
|
"Gradients/Default/network-WithLossCell/_backbone-ReLUReduceMeanDenseRelu/dense-Dense/gradBiasAdd/"\
|
||||||
|
"BiasAddGrad-op8",
|
||||||
|
"Default/network-WithLossCell/_loss_fn-SoftmaxCrossEntropyWithLogits/SoftmaxCrossEntropyWithLogits-op5",
|
||||||
|
"Default/optimizer-Momentum/tuple_getitem-op29",
|
||||||
|
"Default/optimizer-Momentum/ApplyMomentum-op12"
|
||||||
|
],
|
||||||
|
"support_device": [0, 1, 2, 3, 4, 5, 6, 7],
|
||||||
|
"op_debug_mode": 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def generate_dump_json(dump_path, json_file_name, test_key):
|
||||||
|
"""
|
||||||
|
Util function to generate dump configuration json file.
|
||||||
|
"""
|
||||||
|
data = dict()
|
||||||
|
if test_key == "test_async_dump":
|
||||||
|
data = async_dump_dict
|
||||||
|
data["common_dump_settings"]["path"] = dump_path
|
||||||
|
elif test_key == "test_e2e_dump":
|
||||||
|
data = e2e_dump_dict
|
||||||
|
data["common_dump_settings"]["path"] = dump_path
|
||||||
|
elif test_key == "test_async_dump_net_multi_layer_mode1":
|
||||||
|
data = async_dump_dict_2
|
||||||
|
data["common_dump_settings"]["path"] = dump_path
|
||||||
|
else:
|
||||||
|
raise ValueError(
|
||||||
|
"Failed to generate dump json file. The test name value " + test_key + " is invalid.")
|
||||||
|
with open(json_file_name, 'w') as f:
|
||||||
|
json.dump(data, f)
|
|
@ -1,16 +0,0 @@
|
||||||
{
|
|
||||||
"common_dump_settings": {
|
|
||||||
"dump_mode": 0,
|
|
||||||
"path": "/test",
|
|
||||||
"net_name": "Net",
|
|
||||||
"iteration": "0",
|
|
||||||
"input_output": 0,
|
|
||||||
"kernels": ["Default/Conv-op12"],
|
|
||||||
"support_device": [0,1,2,3,4,5,6,7],
|
|
||||||
"op_debug_mode": 0
|
|
||||||
},
|
|
||||||
"e2e_dump_settings": {
|
|
||||||
"enable": true,
|
|
||||||
"trans_flag": false
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,18 +0,0 @@
|
||||||
{
|
|
||||||
"common_dump_settings":{
|
|
||||||
"dump_mode": 0,
|
|
||||||
"path": "/tmp/async_dump/test_async_dump_net_multi_layer_mode1",
|
|
||||||
"net_name": "test",
|
|
||||||
"iteration": "0",
|
|
||||||
"input_output": 2,
|
|
||||||
"kernels": [
|
|
||||||
"default/TensorAdd-op10",
|
|
||||||
"Gradients/Default/network-WithLossCell/_backbone-ReLUReduceMeanDenseRelu/dense-Dense/gradBiasAdd/BiasAddGrad-op8",
|
|
||||||
"Default/network-WithLossCell/_loss_fn-SoftmaxCrossEntropyWithLogits/SoftmaxCrossEntropyWithLogits-op5",
|
|
||||||
"Default/optimizer-Momentum/tuple_getitem-op29",
|
|
||||||
"Default/optimizer-Momentum/ApplyMomentum-op12"
|
|
||||||
],
|
|
||||||
"support_device": [0,1,2,3,4,5,6,7],
|
|
||||||
"op_debug_mode": 0
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -13,13 +13,13 @@
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
# ============================================================================
|
# ============================================================================
|
||||||
import os
|
import os
|
||||||
import json
|
|
||||||
import sys
|
import sys
|
||||||
import tempfile
|
import tempfile
|
||||||
import time
|
import time
|
||||||
import shutil
|
import shutil
|
||||||
import glob
|
import glob
|
||||||
|
from importlib import import_module
|
||||||
|
from pathlib import Path
|
||||||
import numpy as np
|
import numpy as np
|
||||||
import pytest
|
import pytest
|
||||||
import mindspore.context as context
|
import mindspore.context as context
|
||||||
|
@ -32,6 +32,7 @@ from mindspore.nn import SoftmaxCrossEntropyWithLogits
|
||||||
from mindspore.nn import Momentum
|
from mindspore.nn import Momentum
|
||||||
from mindspore.nn import TrainOneStepCell
|
from mindspore.nn import TrainOneStepCell
|
||||||
from mindspore.nn import WithLossCell
|
from mindspore.nn import WithLossCell
|
||||||
|
from dump_test_utils import generate_dump_json
|
||||||
|
|
||||||
|
|
||||||
class Net(nn.Cell):
|
class Net(nn.Cell):
|
||||||
|
@ -47,14 +48,6 @@ x = np.array([[1, 2, 3], [4, 5, 6]]).astype(np.float32)
|
||||||
y = np.array([[7, 8, 9], [10, 11, 12]]).astype(np.float32)
|
y = np.array([[7, 8, 9], [10, 11, 12]]).astype(np.float32)
|
||||||
|
|
||||||
|
|
||||||
def change_current_dump_json(file_name, dump_path, dump_config_path):
|
|
||||||
with open(file_name, 'r+') as f:
|
|
||||||
data = json.load(f)
|
|
||||||
data["common_dump_settings"]["path"] = dump_path
|
|
||||||
with open(dump_config_path, 'w') as f:
|
|
||||||
json.dump(data, f)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.level1
|
@pytest.mark.level1
|
||||||
@pytest.mark.platform_arm_ascend_training
|
@pytest.mark.platform_arm_ascend_training
|
||||||
@pytest.mark.platform_x86_ascend_training
|
@pytest.mark.platform_x86_ascend_training
|
||||||
|
@ -65,7 +58,7 @@ def test_async_dump():
|
||||||
with tempfile.TemporaryDirectory(dir=pwd) as tmp_dir:
|
with tempfile.TemporaryDirectory(dir=pwd) as tmp_dir:
|
||||||
dump_path = os.path.join(tmp_dir, 'async_dump')
|
dump_path = os.path.join(tmp_dir, 'async_dump')
|
||||||
dump_config_path = os.path.join(tmp_dir, 'async_dump.json')
|
dump_config_path = os.path.join(tmp_dir, 'async_dump.json')
|
||||||
change_current_dump_json('async_dump.json', dump_path, dump_config_path)
|
generate_dump_json(dump_path, dump_config_path, 'test_async_dump')
|
||||||
os.environ['MINDSPORE_DUMP_CONFIG'] = dump_config_path
|
os.environ['MINDSPORE_DUMP_CONFIG'] = dump_config_path
|
||||||
dump_file_path = os.path.join(dump_path, 'rank_0', 'Net', '0', '0')
|
dump_file_path = os.path.join(dump_path, 'rank_0', 'Net', '0', '0')
|
||||||
if os.path.isdir(dump_path):
|
if os.path.isdir(dump_path):
|
||||||
|
@ -83,7 +76,7 @@ def run_e2e_dump():
|
||||||
with tempfile.TemporaryDirectory(dir=pwd) as tmp_dir:
|
with tempfile.TemporaryDirectory(dir=pwd) as tmp_dir:
|
||||||
dump_path = os.path.join(tmp_dir, 'e2e_dump')
|
dump_path = os.path.join(tmp_dir, 'e2e_dump')
|
||||||
dump_config_path = os.path.join(tmp_dir, 'e2e_dump.json')
|
dump_config_path = os.path.join(tmp_dir, 'e2e_dump.json')
|
||||||
change_current_dump_json('e2e_dump.json', dump_path, dump_config_path)
|
generate_dump_json(dump_path, dump_config_path, 'test_e2e_dump')
|
||||||
os.environ['MINDSPORE_DUMP_CONFIG'] = dump_config_path
|
os.environ['MINDSPORE_DUMP_CONFIG'] = dump_config_path
|
||||||
dump_file_path = os.path.join(dump_path, 'rank_0', 'Net', '0', '0')
|
dump_file_path = os.path.join(dump_path, 'rank_0', 'Net', '0', '0')
|
||||||
if os.path.isdir(dump_path):
|
if os.path.isdir(dump_path):
|
||||||
|
@ -178,32 +171,18 @@ class ReluReduceMeanDenseRelu(Cell):
|
||||||
return x_
|
return x_
|
||||||
|
|
||||||
|
|
||||||
def search_path(path, keyword):
|
|
||||||
content = os.listdir(path)
|
|
||||||
for each in content:
|
|
||||||
each_path = path + os.sep + each
|
|
||||||
if keyword in each:
|
|
||||||
return each_path
|
|
||||||
read_write = os.access(each_path, os.W_OK) and os.access(each_path, os.R_OK)
|
|
||||||
if not read_write:
|
|
||||||
continue
|
|
||||||
if os.path.isdir(each_path):
|
|
||||||
search_path(each_path, keyword)
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.level0
|
@pytest.mark.level0
|
||||||
@pytest.mark.platform_arm_ascend_training
|
@pytest.mark.platform_arm_ascend_training
|
||||||
@pytest.mark.platform_x86_ascend_training
|
@pytest.mark.platform_x86_ascend_training
|
||||||
@pytest.mark.env_onecard
|
@pytest.mark.env_onecard
|
||||||
def test_async_dump_net_multi_layer_mode1():
|
def test_async_dump_net_multi_layer_mode1():
|
||||||
context.set_context(mode=context.GRAPH_MODE, device_target="Ascend")
|
context.set_context(mode=context.GRAPH_MODE, device_target="Ascend")
|
||||||
test_name = "test_async_dump_net_multi_layer_mode1"
|
pwd = os.getcwd()
|
||||||
json_file = os.path.join(os.getcwd(), "{}.json".format(test_name))
|
with tempfile.TemporaryDirectory(dir=pwd) as tmp_dir:
|
||||||
rank_id = 0
|
dump_path = os.path.join(tmp_dir, 'async_dump_net_multi_layer_mode1')
|
||||||
dump_full_path = os.path.join("/tmp/async_dump/", "{}_{}".format(test_name, rank_id))
|
json_file_path = os.path.join(tmp_dir, "test_async_dump_net_multi_layer_mode1.json")
|
||||||
os.system("rm -rf {}/*".format(dump_full_path))
|
generate_dump_json(dump_path, json_file_path, 'test_async_dump_net_multi_layer_mode1')
|
||||||
os.environ["MINDSPORE_DUMP_CONFIG"] = json_file
|
os.environ['MINDSPORE_DUMP_CONFIG'] = json_file_path
|
||||||
weight = Tensor(np.ones((1000, 2048)).astype(np.float32))
|
weight = Tensor(np.ones((1000, 2048)).astype(np.float32))
|
||||||
bias = Tensor(np.ones((1000,)).astype(np.float32))
|
bias = Tensor(np.ones((1000,)).astype(np.float32))
|
||||||
net = ReluReduceMeanDenseRelu(weight, bias, 2048, 1000)
|
net = ReluReduceMeanDenseRelu(weight, bias, 2048, 1000)
|
||||||
|
@ -216,31 +195,23 @@ def test_async_dump_net_multi_layer_mode1():
|
||||||
inputs = Tensor(np.random.randn(32, 2048, 7, 7).astype(np.float32))
|
inputs = Tensor(np.random.randn(32, 2048, 7, 7).astype(np.float32))
|
||||||
label = Tensor(np.zeros(shape=(32, 1000)).astype(np.float32))
|
label = Tensor(np.zeros(shape=(32, 1000)).astype(np.float32))
|
||||||
net_dict = train_network(inputs, label)
|
net_dict = train_network(inputs, label)
|
||||||
|
dump_file_path = os.path.join(dump_path, 'rank_0', 'test', '0', '0')
|
||||||
dump_path = "/tmp/async_dump/{}/rank_{}/test/0/0/".format(test_name, rank_id)
|
dump_file_name = list(Path(dump_file_path).rglob("*SoftmaxCrossEntropyWithLogits*"))[0]
|
||||||
dump_file = os.listdir(dump_path)
|
dump_file_full_path = os.path.join(dump_file_path, dump_file_name)
|
||||||
dump_file_name = ""
|
npy_path = os.path.join(dump_path, "npy_files")
|
||||||
for file in dump_file:
|
|
||||||
if "SoftmaxCrossEntropyWithLogits" in file:
|
|
||||||
dump_file_name = file
|
|
||||||
dump_file_full_path = os.path.join(dump_path, dump_file_name)
|
|
||||||
npy_path = os.path.join(os.getcwd(), "./{}".format(test_name))
|
|
||||||
if os.path.exists(npy_path):
|
if os.path.exists(npy_path):
|
||||||
shutil.rmtree(npy_path)
|
shutil.rmtree(npy_path)
|
||||||
os.mkdir(npy_path)
|
os.mkdir(npy_path)
|
||||||
tool_path = search_path('/usr/local/Ascend', 'msaccucmp.pyc')
|
tool_path_search_list = list(Path('/usr/local/Ascend').rglob('msaccucmp.py*'))
|
||||||
if tool_path:
|
if tool_path_search_list:
|
||||||
cmd = "python {0} convert -d {1} -out {2}".format(tool_path, dump_file_full_path, npy_path)
|
converter = import_module("mindspore.offline_debug.convert_async")
|
||||||
os.system(cmd)
|
converter.AsyncDumpConverter([dump_file_full_path], npy_path).convert_files()
|
||||||
npy_file_list = os.listdir(npy_path)
|
npy_result_file = list(Path(npy_path).rglob("*output.0.*.npy"))[0]
|
||||||
dump_result = {}
|
dump_result = np.load(os.path.join(npy_path, npy_result_file))
|
||||||
for file in npy_file_list:
|
|
||||||
if "output.0.npy" in file:
|
|
||||||
dump_result["output0"] = np.load(os.path.join(npy_path, file))
|
|
||||||
for index, value in enumerate(net_dict):
|
for index, value in enumerate(net_dict):
|
||||||
assert value.asnumpy() == dump_result["output0"][index]
|
assert value.asnumpy() == dump_result[index]
|
||||||
else:
|
else:
|
||||||
print('not find convert tools msaccucmp.pyc')
|
print('Failed to find hisi convert tools: msaccucmp.py or msaccucmp.pyc.')
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.level0
|
@pytest.mark.level0
|
||||||
|
@ -256,7 +227,7 @@ def test_dump_with_diagnostic_path():
|
||||||
pwd = os.getcwd()
|
pwd = os.getcwd()
|
||||||
with tempfile.TemporaryDirectory(dir=pwd) as tmp_dir:
|
with tempfile.TemporaryDirectory(dir=pwd) as tmp_dir:
|
||||||
dump_config_path = os.path.join(tmp_dir, 'e2e_dump.json')
|
dump_config_path = os.path.join(tmp_dir, 'e2e_dump.json')
|
||||||
change_current_dump_json('e2e_dump.json', '', dump_config_path)
|
generate_dump_json('', dump_config_path, 'test_e2e_dump')
|
||||||
os.environ['MINDSPORE_DUMP_CONFIG'] = dump_config_path
|
os.environ['MINDSPORE_DUMP_CONFIG'] = dump_config_path
|
||||||
diagnose_path = os.path.join(tmp_dir, 'e2e_dump')
|
diagnose_path = os.path.join(tmp_dir, 'e2e_dump')
|
||||||
os.environ['MS_DIAGNOSTIC_DATA_PATH'] = diagnose_path
|
os.environ['MS_DIAGNOSTIC_DATA_PATH'] = diagnose_path
|
||||||
|
|
Loading…
Reference in New Issue