// // Copyright 2016 Pixar // // Licensed under the Apache License, Version 2.0 (the "Apache License") // with the following modification; you may not use this file except in // compliance with the Apache License and the following modification to it: // Section 6. Trademarks. is deleted and replaced with: // // 6. Trademarks. This License does not grant permission to use the trade // names, trademarks, service marks, or product names of the Licensor // and its affiliates, except as required to comply with Section 4(c) of // the License and to reproduce the content of the NOTICE file. // // You may obtain a copy of the Apache License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the Apache License with the above modification is // distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY // KIND, either express or implied. See the Apache License for the specific // language governing permissions and limitations under the Apache License. // #include "pxr/pxr.h" #include "pxr/base/tf/mallocTag.h" #include "pxr/base/tf/fileUtils.h" #include "pxr/base/tf/iterator.h" #include "pxr/base/tf/pyResultConversions.h" #include "pxr/base/tf/stringUtils.h" #include "pxr/base/arch/fileSystem.h" #include "pxr/base/arch/symbols.h" #include #include #include #include #include #include #include using std::string; using std::vector; using namespace boost::python; PXR_NAMESPACE_USING_DIRECTIVE namespace { static bool _Initialize() { string reason; return TfMallocTag::Initialize(&reason); } static bool _Initialize2(const std::string& captureTag) { string reason; bool result = TfMallocTag::Initialize(&reason); if (result) { TfMallocTag::SetCapturedMallocStacksMatchList(captureTag); } return result; } static TfMallocTag::CallTree _GetCallTree() { TfMallocTag::CallTree ret; TfMallocTag::GetCallTree(&ret); return ret; } static std::vector _GetCallStacks() { std::vector > stacks = TfMallocTag::GetCapturedMallocStacks(); // Cache address to function name map, one lookup per address. std::map functionNames; TF_FOR_ALL(stack, stacks) { TF_FOR_ALL(func, *stack) { std::string& name = functionNames[*func]; if (name.empty()) { ArchGetAddressInfo(reinterpret_cast(*func), NULL, NULL, &name, NULL); if (name.empty()) { name = ""; } } } } std::vector result; TF_FOR_ALL(stack, stacks) { result.push_back(std::string()); std::string& trace = result.back(); TF_FOR_ALL(func, *stack) { trace += TfStringPrintf(" 0x%016lx: %s\n", (unsigned long)*func, functionNames[*func].c_str()); } trace += '\n'; } return result; } static string _GetPrettyPrintString(TfMallocTag::CallTree const &self) { return self.GetPrettyPrintString(); } static vector _GetCallSites(TfMallocTag::CallTree const &self) { return self.callSites; } static TfMallocTag::CallTree::PathNode _GetRoot(TfMallocTag::CallTree const &self) { return self.root; } static vector _GetChildren(TfMallocTag::CallTree::PathNode const &self) { return self.children; } static void _Report( TfMallocTag::CallTree const &self, std::string const &rootName) { self.Report(std::cout, rootName); } static void _ReportToFile( TfMallocTag::CallTree const &self, std::string const &fileName, std::string const &rootName) { std::ofstream os(fileName.c_str()); self.Report(os, rootName); } static std::string _LogReport( TfMallocTag::CallTree const &self, std::string const &rootName) { string tmpFile; ArchMakeTmpFile(std::string("callSiteReport") + (rootName.empty() ? "" : "_") + rootName, &tmpFile); _ReportToFile(self, tmpFile, rootName); return tmpFile; } } // anonymous namespace void wrapMallocTag() { typedef TfMallocTag This; scope mallocTag = class_("MallocTag", no_init) .def("Initialize", _Initialize2) .def("Initialize", _Initialize).staticmethod("Initialize") .def("IsInitialized", This::IsInitialized).staticmethod("IsInitialized") .def("GetTotalBytes", This::GetTotalBytes).staticmethod("GetTotalBytes") .def("GetMaxTotalBytes", This::GetMaxTotalBytes).staticmethod("GetMaxTotalBytes") .def("GetCallTree", _GetCallTree).staticmethod("GetCallTree") .def("SetCapturedMallocStacksMatchList", This::SetCapturedMallocStacksMatchList) .staticmethod("SetCapturedMallocStacksMatchList") .def("GetCallStacks", _GetCallStacks, return_value_policy()) .staticmethod("GetCallStacks") .def("SetDebugMatchList", This::SetDebugMatchList) .staticmethod("SetDebugMatchList") ; { scope callTree = class_("CallTree", no_init) .def("GetPrettyPrintString", _GetPrettyPrintString) .def("GetCallSites", _GetCallSites, return_value_policy()) .def("GetRoot", _GetRoot) .def("Report", _Report, (arg("rootName")=std::string())) .def("Report", _ReportToFile, (arg("fileName"), arg("rootName")=std::string())) .def("LogReport", _LogReport, (arg("rootName")=std::string())) ; class_("PathNode", no_init) .def_readonly("nBytes", &This::CallTree::PathNode::nBytes) .def_readonly("nBytesDirect", &This::CallTree::PathNode::nBytesDirect) .def_readonly("nAllocations", &This::CallTree::PathNode::nAllocations) .def_readonly("siteName", &This::CallTree::PathNode::siteName) .def("GetChildren", _GetChildren, return_value_policy()) ; class_("CallSite", no_init) .def_readonly("name", &This::CallTree::CallSite::name) .def_readonly("nBytes", &This::CallTree::CallSite::nBytes) ; } }