/* * Copyright (c) 2013-2017 Chris Newland. * Licensed under https://github.com/AdoptOpenJDK/jitwatch/blob/master/LICENSE-BSD * Instructions: https://github.com/AdoptOpenJDK/jitwatch/wiki */ package org.adoptopenjdk.jitwatch.report.suggestion; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.BRANCH_TAKEN_ALWAYS; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.BRANCH_TAKEN_MIN; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.BRANCH_TAKEN_MAX; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.BRANCH_TAKEN_NEVER; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_BCI; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_BRANCH_COUNT; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_BRANCH_PROB; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_BYTES; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_ID; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_IICOUNT; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_METHOD; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_NAME; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_REASON; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_UNLOADED; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.C_NEWLINE; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.S_EMPTY; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.S_PARSE_HIR; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_BC; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_BRANCH; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_CALL; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_CAST_UP; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_DEPENDENCY; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_DIRECT_CALL; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_INLINE_FAIL; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_INLINE_SUCCESS; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_INTRINSIC; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_KLASS; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_METHOD; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_OBSERVE; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_PARSE; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_PARSE_DONE; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_PHASE; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_PHASE_DONE; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_PREDICTED_CALL; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_TYPE; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_UNCOMMON_TRAP; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_VIRTUAL_CALL; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_ASSERT_NULL; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.TAG_HOT_THROW; import static org.adoptopenjdk.jitwatch.core.JITWatchConstants.ATTR_PREALLOCATED; import java.util.HashMap; import java.util.List; import java.util.Map; import org.adoptopenjdk.jitwatch.compilation.CompilationUtil; import org.adoptopenjdk.jitwatch.model.CodeCacheEvent; import org.adoptopenjdk.jitwatch.model.Compilation; import org.adoptopenjdk.jitwatch.model.IMetaMember; import org.adoptopenjdk.jitwatch.model.IParseDictionary; import org.adoptopenjdk.jitwatch.model.IReadOnlyJITDataModel; import org.adoptopenjdk.jitwatch.model.LogParseException; import org.adoptopenjdk.jitwatch.model.Tag; import org.adoptopenjdk.jitwatch.report.AbstractReportBuilder; import org.adoptopenjdk.jitwatch.report.Report; import org.adoptopenjdk.jitwatch.report.ReportType; import org.adoptopenjdk.jitwatch.util.ParseUtil; import org.adoptopenjdk.jitwatch.util.StringUtil; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class SuggestionWalker extends AbstractReportBuilder { private IMetaMember compilationRootMember; private int compilationIndex; private static final Map<String, Double> scoreMap = new HashMap<>(); private static final Map<String, String> explanationMap = new HashMap<>(); // see // https://wikis.oracle.com/display/HotSpotInternals/Server+Compiler+Inlining+Messages private static final String REASON_HOT_METHOD_TOO_BIG = "hot method too big"; private static final String REASON_TOO_BIG = "too big"; private static final String REASON_ALREADY_COMPILED_INTO_A_BIG_METHOD = "already compiled into a big method"; private static final String REASON_ALREADY_COMPILED_INTO_A_MEDIUM_METHOD = "already compiled into a medium method"; private static final String REASON_NEVER_EXECUTED = "never executed"; private static final String REASON_EXEC_LESS_MIN_INLINING_THRESHOLD = "executed < MinInliningThreshold times"; private static final String REASON_CALL_SITE_NOT_REACHED = "call site not reached"; private static final String REASON_UNCERTAIN_BRANCH = "Uncertain branch"; private static final String REASON_NATIVE_METHOD = "native method"; private static final String REASON_CALLEE_IS_TOO_LARGE = "callee is too large"; private static final String REASON_NO_STATIC_BINDING = "no static binding"; private static final String REASON_NOT_INLINEABLE = "not inlineable"; private static final String REASON_NOT_AN_ACCESSOR = "not an accessor"; private static final String REASON_RECURSIVE_INLINING_TOO_DEEP = "recursive inlining too deep"; // 7 private static final String REASON_RECURSIVE_INLINING_IS_TOO_DEEP = "recursive inlining is too deep"; // 8 private static final String REASON_RECURSIVELY_INLINING_TOO_DEEP = "recursively inlining too deep"; // 7 private static final String REASON_INLINING_IS_TOO_DEEP = "inlining too deep"; private static final String REASON_INLINING_PROHIBITED_BY_POLICY = "inlining prohibited by policy"; private static final String REASON_SIZE_ABOVE_DESIRED_METHOD_LIMIT = "size > DesiredMethodLimit"; private static final String REASON_NODE_COUNT_INLINING_CUTOFF = "NodeCountInliningCutoff"; private static final String REASON_UNLOADED_SIGNATURE_CLASSES = "unloaded signature classes"; // don't inline throwable methods unless the inlining tree is rooted in a // throwable class // src/share/vm/c1/c1_GraphBuilder.cpp private static final String REASON_DONT_THROW_INLINEABLE_CONSTRUCTORS = "don't inline Throwable constructors"; private static final String CODE_CACHE_FULL = "Code cache full, no further JIT compilation is possible"; private static final String HOT_THROW_NOT_PREALLOCATED = "Hot throw was not preallocated"; static { scoreMap.put(CODE_CACHE_FULL, 1.0); scoreMap.put(REASON_HOT_METHOD_TOO_BIG, 1.0); scoreMap.put(REASON_INLINING_IS_TOO_DEEP, 0.8); scoreMap.put(REASON_CALLEE_IS_TOO_LARGE, 0.5); scoreMap.put(REASON_UNCERTAIN_BRANCH, 0.5); scoreMap.put(REASON_TOO_BIG, 0.5); scoreMap.put(REASON_ALREADY_COMPILED_INTO_A_BIG_METHOD, 0.4); scoreMap.put(REASON_ALREADY_COMPILED_INTO_A_MEDIUM_METHOD, 0.4); scoreMap.put(REASON_NOT_INLINEABLE, 0.4); scoreMap.put(REASON_RECURSIVE_INLINING_TOO_DEEP, 0.4); scoreMap.put(REASON_RECURSIVE_INLINING_IS_TOO_DEEP, 0.4); scoreMap.put(REASON_RECURSIVELY_INLINING_TOO_DEEP, 0.4); scoreMap.put(REASON_SIZE_ABOVE_DESIRED_METHOD_LIMIT, 0.4); scoreMap.put(HOT_THROW_NOT_PREALLOCATED, 0.3); scoreMap.put(REASON_INLINING_PROHIBITED_BY_POLICY, 0.3); scoreMap.put(REASON_EXEC_LESS_MIN_INLINING_THRESHOLD, 0.2); scoreMap.put(REASON_NO_STATIC_BINDING, 0.2); scoreMap.put(REASON_NODE_COUNT_INLINING_CUTOFF, 0.2); scoreMap.put(REASON_UNLOADED_SIGNATURE_CLASSES, 0.1); scoreMap.put(REASON_NOT_AN_ACCESSOR, 0.1); scoreMap.put(REASON_NEVER_EXECUTED, 0.0); scoreMap.put(REASON_NATIVE_METHOD, 0.0); scoreMap.put(REASON_CALL_SITE_NOT_REACHED, 0.0); scoreMap.put(REASON_DONT_THROW_INLINEABLE_CONSTRUCTORS, 0.0); explanationMap.put(REASON_HOT_METHOD_TOO_BIG, "The callee method is 'hot' but is too big to be inlined into the caller.\nYou may want to consider refactoring the callee into smaller methods."); explanationMap.put(REASON_TOO_BIG, "The callee method is not 'hot' but is too big to be inlined into the caller method."); explanationMap.put(REASON_ALREADY_COMPILED_INTO_A_BIG_METHOD, "The callee method has already been compiled into a 'big' method somewhere else"); explanationMap.put(REASON_ALREADY_COMPILED_INTO_A_MEDIUM_METHOD, "The callee method has already been compiled into a 'medium' method somewhere else"); explanationMap.put(REASON_EXEC_LESS_MIN_INLINING_THRESHOLD, "The callee method was not called enough times to be inlined."); explanationMap.put(REASON_CALLEE_IS_TOO_LARGE, "The callee method is greater than the max inlining size at the C1 compiler level."); explanationMap.put(REASON_NO_STATIC_BINDING, "The callee is known but there is no static binding so could not be inlined."); explanationMap.put(REASON_NOT_AN_ACCESSOR, "The callee method is not an accessor."); explanationMap.put(REASON_DONT_THROW_INLINEABLE_CONSTRUCTORS, "A Throwable method won't be inlined unless the inlining tree is rooted in a throwable class."); final String explanationInliningTooDeep = "Inlining could not continue as the inlining depth exceeded the limit."; explanationMap.put(REASON_RECURSIVE_INLINING_TOO_DEEP, explanationInliningTooDeep); explanationMap.put(REASON_RECURSIVE_INLINING_IS_TOO_DEEP, explanationInliningTooDeep); explanationMap.put(REASON_RECURSIVELY_INLINING_TOO_DEEP, explanationInliningTooDeep); explanationMap.put(REASON_SIZE_ABOVE_DESIRED_METHOD_LIMIT, S_EMPTY); explanationMap.put(REASON_NODE_COUNT_INLINING_CUTOFF, S_EMPTY); explanationMap.put(REASON_UNLOADED_SIGNATURE_CLASSES, S_EMPTY); } private static final int MIN_BRANCH_INVOCATIONS = 1000; private static final int MIN_INLINING_INVOCATIONS = 1000; private static final Logger logger = LoggerFactory.getLogger(SuggestionWalker.class); public SuggestionWalker(IReadOnlyJITDataModel model) { super(model); ignoreTags.add(TAG_KLASS); ignoreTags.add(TAG_TYPE); ignoreTags.add(TAG_DEPENDENCY); ignoreTags.add(TAG_PARSE_DONE); ignoreTags.add(TAG_DIRECT_CALL); ignoreTags.add(TAG_PHASE_DONE); ignoreTags.add(TAG_INLINE_SUCCESS); ignoreTags.add(TAG_UNCOMMON_TRAP); ignoreTags.add(TAG_INTRINSIC); ignoreTags.add(TAG_PREDICTED_CALL); ignoreTags.add(TAG_VIRTUAL_CALL); ignoreTags.add(TAG_CAST_UP); ignoreTags.add(TAG_OBSERVE); ignoreTags.add(TAG_ASSERT_NULL); } @Override protected void findNonMemberReports() { checkIfCodeCacheFull(); } private void checkIfCodeCacheFull() { List<CodeCacheEvent> codeCacheEvents = model.getCodeCacheEvents(); for (CodeCacheEvent event : codeCacheEvents) { switch (event.getEventType()) { case CACHE_FULL: handleCodeCacheFull(event); break; default: break; } } } @Override public void visit(IMetaMember metaMember) { if (metaMember != null && metaMember.isCompiled()) { this.compilationRootMember = metaMember; try { for (Compilation compilation : metaMember.getCompilations()) { this.compilationIndex = compilation.getIndex(); CompilationUtil.visitParseTagsOfCompilation(compilation, this); } } catch (LogParseException e) { logger.error("Error building suggestions", e); } } } private void processParseTag(Tag parseTag, IMetaMember caller, IParseDictionary parseDictionary) { String methodID = null; int currentBytecode = -1; for (Tag child : parseTag.getChildren()) { String tagName = child.getName(); Map<String, String> attrs = child.getAttributes(); switch (tagName) { case TAG_METHOD: { methodID = attrs.get(ATTR_ID); break; } case TAG_BC: { String bci = attrs.get(ATTR_BCI); currentBytecode = Integer.parseInt(bci); break; } case TAG_BRANCH: { handleBranchTag(attrs, currentBytecode, caller); break; } case TAG_CALL: { methodID = attrs.get(ATTR_METHOD); break; } case TAG_INLINE_FAIL: { handleInlineFailTag(attrs, methodID, caller, currentBytecode, parseDictionary); break; } case TAG_PARSE: { String callerID = attrs.get(ATTR_METHOD); IMetaMember nestedCaller = ParseUtil.lookupMember(callerID, parseDictionary, model); if (nestedCaller != null) { processParseTag(child, nestedCaller, parseDictionary); } break; } case TAG_PHASE: { String phaseName = attrs.get(ATTR_NAME); if (S_PARSE_HIR.equals(phaseName)) { processParseTag(child, caller, parseDictionary); } else { logger.warn("Don't know how to handle phase {}", phaseName); } break; } case TAG_HOT_THROW: { String preallocated = attrs.get(ATTR_PREALLOCATED); if (!"1".equals(preallocated)) { handleHotThrowNotPreallocated(attrs, currentBytecode, caller); } break; } default: handleOther(child); break; } } } private void handleInlineFailTag(Map<String, String> attrs, String methodID, IMetaMember caller, int currentBytecode, IParseDictionary parseDictionary) { IMetaMember callee = ParseUtil.lookupMember(methodID, parseDictionary, model); if (callee != null) { Tag methodTag = parseDictionary.getMethod(methodID); Map<String, String> methodTagAttributes = methodTag.getAttributes(); String methodBytecodes = methodTagAttributes.get(ATTR_BYTES); String invocations = methodTagAttributes.get(ATTR_IICOUNT); if (invocations != null) { int invocationCount = Integer.parseInt(invocations); if (invocationCount >= MIN_INLINING_INVOCATIONS) { String reason = attrs.get(ATTR_REASON); reason = StringUtil.replaceXMLEntities(reason); double score = 0; if (scoreMap.containsKey(reason)) { score = scoreMap.get(reason); } else { logger.warn("No score is set for reason: {}", reason); } StringBuilder reasonBuilder = new StringBuilder(); reasonBuilder.append("The call at bytecode ").append(currentBytecode).append(" to\n"); reasonBuilder.append("Class: ").append(callee.getMetaClass().getFullyQualifiedName()).append(C_NEWLINE); reasonBuilder.append("Member: ").append(callee.toStringUnqualifiedMethodName(false, false)).append(C_NEWLINE); reasonBuilder.append("was not inlined for reason: '").append(reason).append("'\n"); if (explanationMap.containsKey(reason)) { reasonBuilder.append(explanationMap.get(reason)).append(C_NEWLINE); } reasonBuilder.append("Invocations: ").append(invocationCount).append(C_NEWLINE); reasonBuilder.append("Size of callee bytecode: ").append(methodBytecodes); score *= invocationCount; if (score > 0) { Report suggestion = new Report(caller, compilationIndex, currentBytecode, reasonBuilder.toString(), ReportType.INLINE_FAILURE, (int) Math.ceil(score)); if (!reportList.contains(suggestion)) { reportList.add(suggestion); } } } } else if ("1".equals(methodTagAttributes.get(ATTR_UNLOADED))) { } else { logger.warn("Invocation count missing for methodID: {}", methodID); logger.warn("{}", methodTag); } } } private void handleCodeCacheFull(CodeCacheEvent event) { String reason = CODE_CACHE_FULL; double score = 0; if (scoreMap.containsKey(reason)) { score = scoreMap.get(reason); } else { logger.warn("No score is set for reason: {}", reason); } StringBuilder builder = new StringBuilder(); builder.append(reason).append(C_NEWLINE); builder.append("Occurred at ").append(event.getStamp() / 1000).append(" seconds").append(C_NEWLINE); builder.append( "The code cache is a memory region in the VM where JIT-compiled methods are stored. Once this becomes full no further JIT compilation is possible and uncompiled methods will run in the interpreter which may cause performance issues for your application.") .append(C_NEWLINE); builder.append("You can control the code cache size with -XX:ReservedCodeCacheSize=<size>m"); Report suggestion = new Report(null, -1, -1, builder.toString(), ReportType.CODE_CACHE, (int) Math.ceil(score)); if (!reportList.contains(suggestion)) { reportList.add(suggestion); } } private void handleHotThrowNotPreallocated(Map<String, String> attrs, int currentBytecode, IMetaMember caller) { double score = scoreMap.get(HOT_THROW_NOT_PREALLOCATED); if (score > 0) { StringBuilder reasonBuilder = new StringBuilder(); reasonBuilder.append("Method contains a hot throw at bytecode "); reasonBuilder.append(currentBytecode); reasonBuilder.append(" that was not pre-allocated."); Report suggestion = new Report(caller, compilationIndex, currentBytecode, reasonBuilder.toString(), ReportType.HOT_THROW, (int) Math.ceil(score)); if (!reportList.contains(suggestion)) { reportList.add(suggestion); } } } private void handleBranchTag(Map<String, String> attrs, int currentBytecode, IMetaMember caller) { String countStr = attrs.get(ATTR_BRANCH_COUNT); String probStr = attrs.get(ATTR_BRANCH_PROB); long count = 0; double probability = 0.0; if (countStr != null) { try { count = (long) ParseUtil.parseLocaleSafeDouble(countStr); } catch (NumberFormatException nfe) { logger.error("Couldn't parse branch tag attribute {}", countStr, nfe); } } if (probStr != null) { if (BRANCH_TAKEN_NEVER.equalsIgnoreCase(probStr) || BRANCH_TAKEN_MIN.equalsIgnoreCase(probStr)) { probability = 0; } else if (BRANCH_TAKEN_ALWAYS.equalsIgnoreCase(probStr) || BRANCH_TAKEN_MAX.equalsIgnoreCase(probStr)) { probability = 1; } else { try { probability = ParseUtil.parseLocaleSafeDouble(probStr); } catch (NumberFormatException nfe) { logger.error("Unrecognised branch probability: {}", probStr, nfe); } } } double score = 0; if (probability > 0.45 && probability < 0.55 && count >= MIN_BRANCH_INVOCATIONS) { score = scoreMap.get(REASON_UNCERTAIN_BRANCH); score *= count; } if (score > 0) { StringBuilder reasonBuilder = new StringBuilder(); reasonBuilder.append("Method contains an unpredictable branch at bytecode "); reasonBuilder.append(currentBytecode); reasonBuilder.append(" that was observed "); reasonBuilder.append(count); reasonBuilder.append(" times and is taken with probability "); reasonBuilder.append(probability); reasonBuilder.append( ".\nIt may be possbile to modify the branch (for example by sorting a Collection before iterating) to make it more predictable."); Report suggestion = new Report(caller, compilationIndex, currentBytecode, reasonBuilder.toString(), ReportType.BRANCH, (int) Math.ceil(score)); if (!reportList.contains(suggestion)) { reportList.add(suggestion); } } } @Override public void visitTag(Tag parseTag, IParseDictionary parseDictionary) throws LogParseException { processParseTag(parseTag, compilationRootMember, parseDictionary); } }