123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228 |
- #define LOG_TAG "GraphDump"
- #include "GraphDump.h"
- #include "HalInterfaces.h"
- #include <android-base/logging.h>
- #include <set>
- #include <iostream>
- #include <sstream>
- namespace android {
- namespace nn {
- namespace {
- class Dumper {
- public:
- Dumper(std::ostream* outStream) : mStream(outStream) { }
- Dumper(const Dumper&) = delete;
- void operator=(const Dumper&) = delete;
- template <typename T>
- Dumper& operator<<(const T& val) {
- mStringStream << val;
- return *this;
- }
- class EndlType { };
- Dumper& operator<<(EndlType) {
- if (mStream) {
- *mStream << mStringStream.str() << std::endl;
- } else {
-
-
-
-
-
-
-
-
-
- LOG(INFO) << mStringStream.str();
- }
- std::ostringstream empty;
- std::swap(mStringStream, empty);
- return *this;
- }
- static const EndlType endl;
- private:
- std::ostream* mStream;
- std::ostringstream mStringStream;
- };
- const Dumper::EndlType Dumper::endl;
- }
- static std::string translate(OperandType type) {
- switch (type) {
- case OperandType::FLOAT32: return "F32";
- case OperandType::INT32: return "I32";
- case OperandType::UINT32: return "U32";
- case OperandType::TENSOR_FLOAT32: return "TF32";
- case OperandType::TENSOR_INT32: return "TI32";
- case OperandType::TENSOR_QUANT8_ASYMM: return "TQ8A";
- case OperandType::OEM: return "OEM";
- case OperandType::TENSOR_OEM_BYTE: return "TOEMB";
- default: return toString(type);
- }
- }
- namespace {
- template<OperandType nnType, typename cppType>
- void tryValueDump(Dumper& dump, const Model& model, const Operand& opnd) {
- if (opnd.type != nnType ||
- opnd.lifetime != OperandLifeTime::CONSTANT_COPY ||
- opnd.location.length != sizeof(cppType)) {
- return;
- }
- cppType val;
- memcpy(&val, &model.operandValues[opnd.location.offset], sizeof(cppType));
- dump << " = " << val;
- }
- }
- void graphDump(const char* name, const Model& model, std::ostream* outStream) {
-
-
-
-
- Dumper dump(outStream);
- dump << "// " << name << Dumper::endl;
- dump << "digraph {" << Dumper::endl;
-
- std::set<uint32_t> modelIO;
- for (unsigned i = 0, e = model.inputIndexes.size(); i < e; i++) {
- modelIO.insert(model.inputIndexes[i]);
- }
- for (unsigned i = 0, e = model.outputIndexes.size(); i < e; i++) {
- modelIO.insert(model.outputIndexes[i]);
- }
-
- for (unsigned i = 0, e = model.operands.size(); i < e; i++) {
- dump << " d" << i << " [";
- if (modelIO.count(i)) {
- dump << "style=filled fillcolor=black fontcolor=white ";
- }
- dump << "label=\"" << i;
- const Operand& opnd = model.operands[i];
- const char* kind = nullptr;
- switch (opnd.lifetime) {
- case OperandLifeTime::CONSTANT_COPY:
- kind = "COPY";
- break;
- case OperandLifeTime::CONSTANT_REFERENCE:
- kind = "REF";
- break;
- case OperandLifeTime::NO_VALUE:
- kind = "NO";
- break;
- default:
-
- break;
- }
- if (kind) {
- dump << ": " << kind;
- }
- dump << "\\n" << translate(opnd.type);
- tryValueDump<OperandType::FLOAT32, float>(dump, model, opnd);
- tryValueDump<OperandType::INT32, int>(dump, model, opnd);
- tryValueDump<OperandType::UINT32, unsigned>(dump, model, opnd);
- if (opnd.dimensions.size()) {
- dump << "(";
- for (unsigned i = 0, e = opnd.dimensions.size(); i < e; i++) {
- if (i > 0) {
- dump << "x";
- }
- dump << opnd.dimensions[i];
- }
- dump << ")";
- }
- dump << "\"]" << Dumper::endl;
- }
-
- for (unsigned i = 0, e = model.operations.size(); i < e; i++) {
- const Operation& operation = model.operations[i];
- dump << " n" << i << " [shape=box";
- const uint32_t maxArity = std::max(operation.inputs.size(), operation.outputs.size());
- if (maxArity > 1) {
- if (maxArity == operation.inputs.size()) {
- dump << " ordering=in";
- } else {
- dump << " ordering=out";
- }
- }
- dump << " label=\"" << i << ": "
- << toString(operation.type) << "\"]" << Dumper::endl;
- {
-
- for (unsigned in = 0, inE = operation.inputs.size(); in < inE; in++) {
- dump << " d" << operation.inputs[in] << " -> n" << i;
- if (inE > 1) {
- dump << " [label=" << in << "]";
- }
- dump << Dumper::endl;
- }
- }
- {
-
- for (unsigned out = 0, outE = operation.outputs.size(); out < outE; out++) {
- dump << " n" << i << " -> d" << operation.outputs[out];
- if (outE > 1) {
- dump << " [label=" << out << "]";
- }
- dump << Dumper::endl;
- }
- }
- }
- dump << "}" << Dumper::endl;
- }
- }
- }
|