1 /** 2 Copyright: Copyright (c) 2017, Joakim Brännström. All rights reserved. 3 License: MPL-2 4 Author: Joakim Brännström (joakim.brannstrom@gmx.com) 5 6 This Source Code Form is subject to the terms of the Mozilla Public License, 7 v.2.0. If a copy of the MPL was not distributed with this file, You can obtain 8 one at http://mozilla.org/MPL/2.0/. 9 */ 10 module dextool.plugin.analyze.visitor; 11 12 import libclang_ast.ast : Visitor; 13 import dextool.type : AbsolutePath; 14 15 @safe: 16 17 /** Calculate McCabe per file and function. 18 */ 19 final class TUVisitor : Visitor { 20 import std.string : startsWith; 21 import dsrcgen.cpp; 22 import libclang_ast.ast; 23 import libclang_ast.cursor_logger : logNode, mixinNodeLog; 24 import cpptooling.data.symbol : Container; 25 26 alias visit = Visitor.visit; 27 28 mixin generateIndentIncrDecr; 29 30 alias CallbackT(T) = void delegate(const(T) v) @safe; 31 32 private static string makeCallback(string kind) { 33 string s; 34 string alias_name = "On" ~ kind ~ "T"; 35 s = "alias " ~ alias_name ~ " = CallbackT!" ~ kind ~ ";\n"; 36 s ~= alias_name ~ "[] on" ~ kind ~ ";\n"; 37 return s; 38 } 39 40 private static string makeCallbacks(Kinds...)() { 41 string s; 42 foreach (k; Kinds) 43 s ~= makeCallback(k.stringof); 44 return s; 45 } 46 47 // note that it requires a member variable called restrict 48 private static string makeVisitor(string node_t, string callback_member) { 49 string s = "override void visit(scope const " ~ node_t ~ " v) @trusted {\n"; 50 s ~= "auto callbacks = " ~ callback_member ~ ";"; 51 s ~= q{ 52 if (!v.cursor.location.path.startsWith(restrict.toString)) 53 return; 54 foreach (c; callbacks) 55 c(v); 56 v.accept(this); 57 }; 58 59 s ~= "}\n"; 60 return s; 61 } 62 63 private static string makeVisitors(Kinds...)() { 64 string s; 65 foreach (k; Kinds) { 66 s ~= makeVisitor(k.stringof, "on" ~ k.stringof); 67 } 68 return s; 69 } 70 71 import std.meta : AliasSeq; 72 73 private alias callbackKinds = AliasSeq!(FunctionDecl, Constructor, 74 Destructor, CxxMethod, ConversionFunction, FunctionTemplate, ClassTemplate); 75 76 // debugging 77 //pragma(msg, makeCallbacks!callbackKinds); 78 //pragma(msg, makeVisitors!callbackKinds); 79 80 mixin(makeCallbacks!(callbackKinds)); 81 mixin(makeVisitors!(callbackKinds)); 82 83 private AbsolutePath restrict; 84 85 /** 86 * Params: 87 * restrict = only analyze files starting with this path 88 */ 89 this(AbsolutePath restrict) nothrow { 90 this.restrict = restrict; 91 } 92 93 override void visit(scope const(TranslationUnit) v) { 94 v.accept(this); 95 } 96 97 override void visit(scope const(Attribute) v) { 98 v.accept(this); 99 } 100 101 override void visit(scope const(Declaration) v) { 102 v.accept(this); 103 } 104 105 override void visit(scope const(Directive) v) { 106 v.accept(this); 107 } 108 109 override void visit(scope const(Expression) v) { 110 v.accept(this); 111 } 112 113 override void visit(scope const(Preprocessor) v) { 114 v.accept(this); 115 } 116 117 override void visit(scope const(Reference) v) { 118 v.accept(this); 119 } 120 121 override void visit(scope const(Statement) v) { 122 v.accept(this); 123 } 124 }