Update NDK prebuilts to build 4418563.
Taken from branch aosp-master.
Bug: http://b/68201917
Test: run test.py.
Change-Id: Ia742e18189fc88e904d7f0f09cbdf29ad4770da7
diff --git a/annotate.py b/annotate.py
index 56b83e1..740a17c 100644
--- a/annotate.py
+++ b/annotate.py
@@ -550,7 +550,7 @@
elif suffix_len == best_suffix_len:
best_path_count += 1
if best_path_count > 1:
- log_warning('multiple source for %s, select %s' % (file, result))
+ log_warning('multiple source for %s, select %s' % (file, best_path))
return best_path
diff --git a/app_profiler.py b/app_profiler.py
index c089bbe..3a8f0b5 100644
--- a/app_profiler.py
+++ b/app_profiler.py
@@ -163,8 +163,17 @@
self.config['launch_activity'] = '.MainActivity'
self.adb.check_run(['shell', 'am', 'force-stop', self.config['app_package_name']])
- while self._find_app_process():
+ count = 0
+ while True:
time.sleep(1)
+ pid = self._find_app_process()
+ if pid is None:
+ break
+ # When testing on Android N, `am force-stop` sometimes can't kill
+ # com.example.simpleperf.simpleperfexampleofkotlin. So use kill when this happens.
+ count += 1
+ if count >= 3:
+ self.run_in_app_dir(['kill', '-9', str(pid)], check_result=False, log_output=False)
if self.config['profile_from_launch']:
self._download_simpleperf()
@@ -257,7 +266,7 @@
def _download_native_libs(self):
- if not self.config['native_lib_dir']:
+ if not self.config['native_lib_dir'] or not self.config['app_package_name']:
return
filename_dict = dict()
for root, _, files in os.walk(self.config['native_lib_dir']):
@@ -417,8 +426,8 @@
parser.add_argument('--arch', help=
"""Select which arch the app is running on, possible values are:
arm, arm64, x86, x86_64. If not set, the script will try to detect it.""")
- parser.add_argument('-r', '--record_options', default="-e cpu-cycles:u -g --duration 10", help=
-"""Set options for `simpleperf record` command.""")
+ parser.add_argument('-r', '--record_options', default="-e cpu-cycles:u -g -f 1000 --duration 10", help=
+"""Set options for `simpleperf record` command. Default is '-e cpu-cycles:u -g -f 1000 --duration 10'.""")
parser.add_argument('-o', '--perf_data_path', default="perf.data", help=
"""The path to store profiling data.""")
parser.add_argument('-nb', '--skip_collect_binaries', action='store_true', help=
@@ -455,4 +464,4 @@
profiler.profile()
if __name__ == '__main__':
- main()
\ No newline at end of file
+ main()
diff --git a/bin/android/arm/simpleperf b/bin/android/arm/simpleperf
index c4d0fce..6f66509 100755
--- a/bin/android/arm/simpleperf
+++ b/bin/android/arm/simpleperf
Binary files differ
diff --git a/bin/android/arm64/simpleperf b/bin/android/arm64/simpleperf
index 9519f68..2c4b0b3 100755
--- a/bin/android/arm64/simpleperf
+++ b/bin/android/arm64/simpleperf
Binary files differ
diff --git a/bin/android/x86/simpleperf b/bin/android/x86/simpleperf
index b7c753a..650fc8d 100755
--- a/bin/android/x86/simpleperf
+++ b/bin/android/x86/simpleperf
Binary files differ
diff --git a/bin/android/x86_64/simpleperf b/bin/android/x86_64/simpleperf
index 069370b..a15fcb6 100755
--- a/bin/android/x86_64/simpleperf
+++ b/bin/android/x86_64/simpleperf
Binary files differ
diff --git a/bin/darwin/x86/libsimpleperf_report.dylib b/bin/darwin/x86/libsimpleperf_report.dylib
index 07ff964..bbeb281 100755
--- a/bin/darwin/x86/libsimpleperf_report.dylib
+++ b/bin/darwin/x86/libsimpleperf_report.dylib
Binary files differ
diff --git a/bin/darwin/x86/simpleperf b/bin/darwin/x86/simpleperf
index 61fd1fe..a20fd3c 100755
--- a/bin/darwin/x86/simpleperf
+++ b/bin/darwin/x86/simpleperf
Binary files differ
diff --git a/bin/darwin/x86_64/libsimpleperf_report.dylib b/bin/darwin/x86_64/libsimpleperf_report.dylib
index f70d7e7..d988a2d 100755
--- a/bin/darwin/x86_64/libsimpleperf_report.dylib
+++ b/bin/darwin/x86_64/libsimpleperf_report.dylib
Binary files differ
diff --git a/bin/darwin/x86_64/simpleperf b/bin/darwin/x86_64/simpleperf
index 1af4ecf..b72869f 100755
--- a/bin/darwin/x86_64/simpleperf
+++ b/bin/darwin/x86_64/simpleperf
Binary files differ
diff --git a/bin/linux/x86/libsimpleperf_report.so b/bin/linux/x86/libsimpleperf_report.so
index 8d3ea66..eb6dcfe 100755
--- a/bin/linux/x86/libsimpleperf_report.so
+++ b/bin/linux/x86/libsimpleperf_report.so
Binary files differ
diff --git a/bin/linux/x86/simpleperf b/bin/linux/x86/simpleperf
index fc76b5a..2d3aa57 100755
--- a/bin/linux/x86/simpleperf
+++ b/bin/linux/x86/simpleperf
Binary files differ
diff --git a/bin/linux/x86_64/libsimpleperf_report.so b/bin/linux/x86_64/libsimpleperf_report.so
index 9e57247..0f2748f 100755
--- a/bin/linux/x86_64/libsimpleperf_report.so
+++ b/bin/linux/x86_64/libsimpleperf_report.so
Binary files differ
diff --git a/bin/linux/x86_64/simpleperf b/bin/linux/x86_64/simpleperf
index 69b0cf9..1e5eee4 100755
--- a/bin/linux/x86_64/simpleperf
+++ b/bin/linux/x86_64/simpleperf
Binary files differ
diff --git a/bin/windows/x86/libsimpleperf_report.dll b/bin/windows/x86/libsimpleperf_report.dll
index 3a37be9..7c12e59 100755
--- a/bin/windows/x86/libsimpleperf_report.dll
+++ b/bin/windows/x86/libsimpleperf_report.dll
Binary files differ
diff --git a/bin/windows/x86/simpleperf.exe b/bin/windows/x86/simpleperf.exe
index cce47bf..b9eb90d 100755
--- a/bin/windows/x86/simpleperf.exe
+++ b/bin/windows/x86/simpleperf.exe
Binary files differ
diff --git a/bin/windows/x86_64/libsimpleperf_report.dll b/bin/windows/x86_64/libsimpleperf_report.dll
index bbe6de6..49cdf3f 100755
--- a/bin/windows/x86_64/libsimpleperf_report.dll
+++ b/bin/windows/x86_64/libsimpleperf_report.dll
Binary files differ
diff --git a/bin/windows/x86_64/simpleperf.exe b/bin/windows/x86_64/simpleperf.exe
index cce47bf..b9eb90d 100755
--- a/bin/windows/x86_64/simpleperf.exe
+++ b/bin/windows/x86_64/simpleperf.exe
Binary files differ
diff --git a/inferno.bat b/inferno.bat
index a2d18a7..5818f98 100644
--- a/inferno.bat
+++ b/inferno.bat
@@ -1 +1,2 @@
+set PYTHONPATH=%PYTHONPATH%;%~dp0
python -m inferno.inferno %*
diff --git a/inferno.sh b/inferno.sh
index 8ba5097..d30ee31 100755
--- a/inferno.sh
+++ b/inferno.sh
@@ -1,2 +1,4 @@
#!/bin/bash
+SCRIPTPATH=$(dirname "$0")
+export PYTHONPATH=$SCRIPTPATH:$PYTHONPATH
python -m inferno.inferno "$@"
diff --git a/inferno/data_types.py b/inferno/data_types.py
index 4f07ba7..dd5bf28 100644
--- a/inferno/data_types.py
+++ b/inferno/data_types.py
@@ -17,8 +17,7 @@
class CallSite:
- def __init__(self, ip, method, dso):
- self.ip = ip
+ def __init__(self, method, dso):
self.method = method
self.dso = dso
@@ -43,9 +42,9 @@
entry = callchain.entries[callchain.nr - j - 1]
if entry.ip == 0:
continue
- chain.append(CallSite(entry.ip, entry.symbol.symbol_name, entry.symbol.dso_name))
+ chain.append(CallSite(entry.symbol.symbol_name, entry.symbol.dso_name))
- chain.append(CallSite(sample.ip, symbol.symbol_name, symbol.dso_name))
+ chain.append(CallSite(symbol.symbol_name, symbol.dso_name))
self.flamegraph.add_callchain(chain, sample.period)
@@ -60,9 +59,10 @@
self.num_samples = 0
def get_thread(self, tid, pid):
- if tid not in self.threads.keys():
- self.threads[tid] = Thread(tid, pid)
- return self.threads[tid]
+ thread = self.threads.get(tid)
+ if thread is None:
+ thread = self.threads[tid] = Thread(tid, pid)
+ return thread
class FlameGraphCallSite:
@@ -74,6 +74,8 @@
return cls.callsite_counter
def __init__(self, method, dso, id):
+ # map from (dso, method) to FlameGraphCallSite. Used to speed up add_callchain().
+ self.child_dict = {}
self.children = []
self.method = method
self.dso = dso
@@ -92,15 +94,24 @@
current.event_count += event_count
def _get_child(self, callsite):
- for c in self.children:
- if c._equivalent(callsite.method, callsite.dso):
- return c
- new_child = FlameGraphCallSite(callsite.method, callsite.dso, self._get_next_callsite_id())
- self.children.append(new_child)
- return new_child
+ key = (callsite.dso, callsite.method)
+ child = self.child_dict.get(key)
+ if child is None:
+ child = self.child_dict[key] = FlameGraphCallSite(callsite.method, callsite.dso,
+ self._get_next_callsite_id())
+ return child
- def _equivalent(self, method, dso):
- return self.method == method and self.dso == dso
+ def trim_callchain(self, min_event_count):
+ """ Remove call sites with event_count < min_event_count in the subtree.
+ Remaining children are collected in a list.
+ """
+ for key in self.child_dict:
+ child = self.child_dict[key]
+ if child.event_count >= min_event_count:
+ child.trim_callchain(min_event_count)
+ self.children.append(child)
+ # Relese child_dict since it will not be used.
+ self.child_dict = None
def get_max_depth(self):
return max([c.get_max_depth() for c in self.children]) + 1 if self.children else 1
diff --git a/inferno/inferno.py b/inferno/inferno.py
index a7c60fa..834bbbb 100644
--- a/inferno/inferno.py
+++ b/inferno/inferno.py
@@ -34,21 +34,18 @@
import os
import subprocess
import sys
-import webbrowser
-try:
- from simpleperf_report_lib import ReportLib
- from utils import log_exit, log_info, AdbHelper
-except:
- print("Please go to the parent directory, and run inferno.sh or inferno.bat.")
- sys.exit(1)
+scripts_path = os.path.dirname(os.path.dirname(os.path.realpath(__file__)))
+sys.path.append(scripts_path)
+from simpleperf_report_lib import ReportLib
+from utils import log_exit, log_info, AdbHelper, open_report_in_browser
from data_types import *
from svg_renderer import *
def collect_data(args):
- app_profiler_args = [sys.executable, "app_profiler.py", "-nb"]
+ app_profiler_args = [sys.executable, os.path.join(scripts_path, "app_profiler.py"), "-nb"]
if args.app:
app_profiler_args += ["-p", args.app]
elif args.native_program:
@@ -122,6 +119,10 @@
process.name = main_threads[0].name
process.pid = main_threads[0].pid
+ for thread in process.threads.values():
+ min_event_count = thread.event_count * args.min_callchain_percentage * 0.01
+ thread.flamegraph.trim_callchain(min_event_count)
+
log_info("Parsed %s callchains." % process.num_samples)
@@ -141,15 +142,12 @@
:param process: Process object
:return: str, absolute path to the file
"""
- f = open('report.html', 'w')
+ f = open(args.report_path, 'w')
filepath = os.path.realpath(f.name)
- f.write("<html>")
- f.write("<head>")
- f.write("""<style type="text/css">""")
- f.write(get_local_asset_content(os.path.join("jqueryui", "jquery-ui.min.css")))
- f.write("</style>")
- f.write("</head>")
- f.write("<body style='font-family: Monospace;' onload='init()'>")
+ if not args.embedded_flamegraph:
+ f.write("<html><body>")
+ f.write("<div id='flamegraph_id' style='font-family: Monospace; %s'>" % (
+ "display: none;" if args.embedded_flamegraph else ""))
f.write("""<style type="text/css"> .s { stroke:black; stroke-width:0.5; cursor:pointer;}
</style>""")
f.write('<style type="text/css"> .t:hover { cursor:pointer; } </style>')
@@ -184,27 +182,26 @@
f.write("</div>")
f.write("""<br/><br/>
<div>Navigate with WASD, zoom in with SPACE, zoom out with BACKSPACE.</div>""")
- f.write("<script>")
- f.write(get_local_asset_content(os.path.join("jqueryui", "jquery-3.2.1.min.js")))
- f.write(get_local_asset_content(os.path.join("jqueryui", "jquery-ui.min.js")))
- f.write("</script>")
f.write("<script>%s</script>" % get_local_asset_content("script.js"))
+ if not args.embedded_flamegraph:
+ f.write("<script>document.addEventListener('DOMContentLoaded', flamegraphInit);</script>")
# Output tid == pid Thread first.
main_thread = [x for x in process.threads.values() if x.tid == process.pid]
for thread in main_thread:
f.write("<br/><br/><b>Main Thread %d (%s) (%d samples):</b><br/>\n\n\n\n" % (
thread.tid, thread.name, thread.num_samples))
- renderSVG(thread.flamegraph, f, args.color, args.svg_width)
+ renderSVG(thread.flamegraph, f, args.color)
other_threads = [x for x in process.threads.values() if x.tid != process.pid]
for thread in other_threads:
f.write("<br/><br/><b>Thread %d (%s) (%d samples):</b><br/>\n\n\n\n" % (
thread.tid, thread.name, thread.num_samples))
- renderSVG(thread.flamegraph, f, args.color, args.svg_width)
+ renderSVG(thread.flamegraph, f, args.color)
- f.write("</body>")
- f.write("</html>")
+ f.write("</div>")
+ if not args.embedded_flamegraph:
+ f.write("</body></html")
f.close()
return "file://" + filepath
@@ -222,20 +219,6 @@
process.props['ro.product.manufacturer'] = adb.get_property('ro.product.manufacturer')
-def open_report_in_browser(report_path):
- try:
- # Try to open the report with Chrome
- browser_key = ""
- for key, value in webbrowser._browsers.items():
- if key.find("chrome") != -1:
- browser_key = key
- browser = webbrowser.get(browser_key)
- browser.open(report_path, new=0, autoraise=True)
- except:
- # webbrowser.get() doesn't work well on darwin/windows.
- webbrowser.open_new_tab(report_path)
-
-
def main():
parser = argparse.ArgumentParser(description='Report samples in perf.data.')
@@ -260,7 +243,6 @@
instructions to profile java code. It takes some time. You can skip it
if the code has been compiled or you don't need to profile java code.""")
parser.add_argument('-f', '--sample_frequency', type=int, default=6000, help='Sample frequency')
- parser.add_argument('-w', '--svg_width', type=int, default=1124)
parser.add_argument(
'-du',
'--dwarf_unwinding',
@@ -274,6 +256,15 @@
default="")
parser.add_argument('--disable_adb_root', action='store_true', help="""Force adb to run in non
root mode.""")
+ parser.add_argument('-o', '--report_path', default='report.html', help="Set report path.")
+ parser.add_argument('--embedded_flamegraph', action='store_true', help="""
+ Generate embedded flamegraph.""")
+ parser.add_argument('--min_callchain_percentage', default=0.01, type=float, help="""
+ Set min percentage of callchains shown in the report.
+ It is used to limit nodes shown in the flamegraph. For example,
+ when set to 0.01, only callchains taking >= 0.01%% of the event count of
+ the owner thread are collected in the report.""")
+ parser.add_argument('--no_browser', action='store_true', help="Don't open report in browser.")
args = parser.parse_args()
process = Process("", 0)
@@ -295,9 +286,10 @@
parse_samples(process, args)
generate_threads_offsets(process)
report_path = output_report(process, args)
- open_report_in_browser(report_path)
+ if not args.no_browser:
+ open_report_in_browser(report_path)
- log_info("Report generated at '%s'." % report_path)
+ log_info("Flamegraph generated at '%s'." % report_path)
if __name__ == "__main__":
main()
diff --git a/inferno/jqueryui/LICENSE.txt b/inferno/jqueryui/LICENSE.txt
deleted file mode 100644
index 4819e54..0000000
--- a/inferno/jqueryui/LICENSE.txt
+++ /dev/null
@@ -1,43 +0,0 @@
-Copyright jQuery Foundation and other contributors, https://jquery.org/
-
-This software consists of voluntary contributions made by many
-individuals. For exact contribution history, see the revision history
-available at https://github.com/jquery/jquery-ui
-
-The following license applies to all parts of this software except as
-documented below:
-
-====
-
-Permission is hereby granted, free of charge, to any person obtaining
-a copy of this software and associated documentation files (the
-"Software"), to deal in the Software without restriction, including
-without limitation the rights to use, copy, modify, merge, publish,
-distribute, sublicense, and/or sell copies of the Software, and to
-permit persons to whom the Software is furnished to do so, subject to
-the following conditions:
-
-The above copyright notice and this permission notice shall be
-included in all copies or substantial portions of the Software.
-
-THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
-EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
-MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
-NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
-LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
-OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
-WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
-
-====
-
-Copyright and related rights for sample code are waived via CC0. Sample
-code is defined as all source code contained within the demos directory.
-
-CC0: http://creativecommons.org/publicdomain/zero/1.0/
-
-====
-
-All files located in the node_modules and external directories are
-externally maintained libraries used by this software which have their
-own licenses; we recommend you read them, as their terms may differ from
-the terms above.
diff --git a/inferno/jqueryui/jquery-3.2.1.min.js b/inferno/jqueryui/jquery-3.2.1.min.js
deleted file mode 100644
index 644d35e..0000000
--- a/inferno/jqueryui/jquery-3.2.1.min.js
+++ /dev/null
@@ -1,4 +0,0 @@
-/*! jQuery v3.2.1 | (c) JS Foundation and other contributors | jquery.org/license */
-!function(a,b){"use strict";"object"==typeof module&&"object"==typeof module.exports?module.exports=a.document?b(a,!0):function(a){if(!a.document)throw new Error("jQuery requires a window with a document");return b(a)}:b(a)}("undefined"!=typeof window?window:this,function(a,b){"use strict";var c=[],d=a.document,e=Object.getPrototypeOf,f=c.slice,g=c.concat,h=c.push,i=c.indexOf,j={},k=j.toString,l=j.hasOwnProperty,m=l.toString,n=m.call(Object),o={};function p(a,b){b=b||d;var c=b.createElement("script");c.text=a,b.head.appendChild(c).parentNode.removeChild(c)}var q="3.2.1",r=function(a,b){return new r.fn.init(a,b)},s=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,t=/^-ms-/,u=/-([a-z])/g,v=function(a,b){return b.toUpperCase()};r.fn=r.prototype={jquery:q,constructor:r,length:0,toArray:function(){return f.call(this)},get:function(a){return null==a?f.call(this):a<0?this[a+this.length]:this[a]},pushStack:function(a){var b=r.merge(this.constructor(),a);return b.prevObject=this,b},each:function(a){return r.each(this,a)},map:function(a){return this.pushStack(r.map(this,function(b,c){return a.call(b,c,b)}))},slice:function(){return this.pushStack(f.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(a){var b=this.length,c=+a+(a<0?b:0);return this.pushStack(c>=0&&c<b?[this[c]]:[])},end:function(){return this.prevObject||this.constructor()},push:h,sort:c.sort,splice:c.splice},r.extend=r.fn.extend=function(){var a,b,c,d,e,f,g=arguments[0]||{},h=1,i=arguments.length,j=!1;for("boolean"==typeof g&&(j=g,g=arguments[h]||{},h++),"object"==typeof g||r.isFunction(g)||(g={}),h===i&&(g=this,h--);h<i;h++)if(null!=(a=arguments[h]))for(b in a)c=g[b],d=a[b],g!==d&&(j&&d&&(r.isPlainObject(d)||(e=Array.isArray(d)))?(e?(e=!1,f=c&&Array.isArray(c)?c:[]):f=c&&r.isPlainObject(c)?c:{},g[b]=r.extend(j,f,d)):void 0!==d&&(g[b]=d));return g},r.extend({expando:"jQuery"+(q+Math.random()).replace(/\D/g,""),isReady:!0,error:function(a){throw new Error(a)},noop:function(){},isFunction:function(a){return"function"===r.type(a)},isWindow:function(a){return null!=a&&a===a.window},isNumeric:function(a){var b=r.type(a);return("number"===b||"string"===b)&&!isNaN(a-parseFloat(a))},isPlainObject:function(a){var b,c;return!(!a||"[object Object]"!==k.call(a))&&(!(b=e(a))||(c=l.call(b,"constructor")&&b.constructor,"function"==typeof c&&m.call(c)===n))},isEmptyObject:function(a){var b;for(b in a)return!1;return!0},type:function(a){return null==a?a+"":"object"==typeof a||"function"==typeof a?j[k.call(a)]||"object":typeof a},globalEval:function(a){p(a)},camelCase:function(a){return a.replace(t,"ms-").replace(u,v)},each:function(a,b){var c,d=0;if(w(a)){for(c=a.length;d<c;d++)if(b.call(a[d],d,a[d])===!1)break}else for(d in a)if(b.call(a[d],d,a[d])===!1)break;return a},trim:function(a){return null==a?"":(a+"").replace(s,"")},makeArray:function(a,b){var c=b||[];return null!=a&&(w(Object(a))?r.merge(c,"string"==typeof a?[a]:a):h.call(c,a)),c},inArray:function(a,b,c){return null==b?-1:i.call(b,a,c)},merge:function(a,b){for(var c=+b.length,d=0,e=a.length;d<c;d++)a[e++]=b[d];return a.length=e,a},grep:function(a,b,c){for(var d,e=[],f=0,g=a.length,h=!c;f<g;f++)d=!b(a[f],f),d!==h&&e.push(a[f]);return e},map:function(a,b,c){var d,e,f=0,h=[];if(w(a))for(d=a.length;f<d;f++)e=b(a[f],f,c),null!=e&&h.push(e);else for(f in a)e=b(a[f],f,c),null!=e&&h.push(e);return g.apply([],h)},guid:1,proxy:function(a,b){var c,d,e;if("string"==typeof b&&(c=a[b],b=a,a=c),r.isFunction(a))return d=f.call(arguments,2),e=function(){return a.apply(b||this,d.concat(f.call(arguments)))},e.guid=a.guid=a.guid||r.guid++,e},now:Date.now,support:o}),"function"==typeof Symbol&&(r.fn[Symbol.iterator]=c[Symbol.iterator]),r.each("Boolean Number String Function Array Date RegExp Object Error Symbol".split(" "),function(a,b){j["[object "+b+"]"]=b.toLowerCase()});function w(a){var b=!!a&&"length"in a&&a.length,c=r.type(a);return"function"!==c&&!r.isWindow(a)&&("array"===c||0===b||"number"==typeof b&&b>0&&b-1 in a)}var x=function(a){var b,c,d,e,f,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u="sizzle"+1*new Date,v=a.document,w=0,x=0,y=ha(),z=ha(),A=ha(),B=function(a,b){return a===b&&(l=!0),0},C={}.hasOwnProperty,D=[],E=D.pop,F=D.push,G=D.push,H=D.slice,I=function(a,b){for(var c=0,d=a.length;c<d;c++)if(a[c]===b)return c;return-1},J="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",K="[\\x20\\t\\r\\n\\f]",L="(?:\\\\.|[\\w-]|[^\0-\\xa0])+",M="\\["+K+"*("+L+")(?:"+K+"*([*^$|!~]?=)"+K+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+L+"))|)"+K+"*\\]",N=":("+L+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+M+")*)|.*)\\)|)",O=new RegExp(K+"+","g"),P=new RegExp("^"+K+"+|((?:^|[^\\\\])(?:\\\\.)*)"+K+"+$","g"),Q=new RegExp("^"+K+"*,"+K+"*"),R=new RegExp("^"+K+"*([>+~]|"+K+")"+K+"*"),S=new RegExp("="+K+"*([^\\]'\"]*?)"+K+"*\\]","g"),T=new RegExp(N),U=new RegExp("^"+L+"$"),V={ID:new RegExp("^#("+L+")"),CLASS:new RegExp("^\\.("+L+")"),TAG:new RegExp("^("+L+"|[*])"),ATTR:new RegExp("^"+M),PSEUDO:new RegExp("^"+N),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+K+"*(even|odd|(([+-]|)(\\d*)n|)"+K+"*(?:([+-]|)"+K+"*(\\d+)|))"+K+"*\\)|)","i"),bool:new RegExp("^(?:"+J+")$","i"),needsContext:new RegExp("^"+K+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+K+"*((?:-\\d)?\\d*)"+K+"*\\)|)(?=[^-]|$)","i")},W=/^(?:input|select|textarea|button)$/i,X=/^h\d$/i,Y=/^[^{]+\{\s*\[native \w/,Z=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,$=/[+~]/,_=new RegExp("\\\\([\\da-f]{1,6}"+K+"?|("+K+")|.)","ig"),aa=function(a,b,c){var d="0x"+b-65536;return d!==d||c?b:d<0?String.fromCharCode(d+65536):String.fromCharCode(d>>10|55296,1023&d|56320)},ba=/([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,ca=function(a,b){return b?"\0"===a?"\ufffd":a.slice(0,-1)+"\\"+a.charCodeAt(a.length-1).toString(16)+" ":"\\"+a},da=function(){m()},ea=ta(function(a){return a.disabled===!0&&("form"in a||"label"in a)},{dir:"parentNode",next:"legend"});try{G.apply(D=H.call(v.childNodes),v.childNodes),D[v.childNodes.length].nodeType}catch(fa){G={apply:D.length?function(a,b){F.apply(a,H.call(b))}:function(a,b){var c=a.length,d=0;while(a[c++]=b[d++]);a.length=c-1}}}function ga(a,b,d,e){var f,h,j,k,l,o,r,s=b&&b.ownerDocument,w=b?b.nodeType:9;if(d=d||[],"string"!=typeof a||!a||1!==w&&9!==w&&11!==w)return d;if(!e&&((b?b.ownerDocument||b:v)!==n&&m(b),b=b||n,p)){if(11!==w&&(l=Z.exec(a)))if(f=l[1]){if(9===w){if(!(j=b.getElementById(f)))return d;if(j.id===f)return d.push(j),d}else if(s&&(j=s.getElementById(f))&&t(b,j)&&j.id===f)return d.push(j),d}else{if(l[2])return G.apply(d,b.getElementsByTagName(a)),d;if((f=l[3])&&c.getElementsByClassName&&b.getElementsByClassName)return G.apply(d,b.getElementsByClassName(f)),d}if(c.qsa&&!A[a+" "]&&(!q||!q.test(a))){if(1!==w)s=b,r=a;else if("object"!==b.nodeName.toLowerCase()){(k=b.getAttribute("id"))?k=k.replace(ba,ca):b.setAttribute("id",k=u),o=g(a),h=o.length;while(h--)o[h]="#"+k+" "+sa(o[h]);r=o.join(","),s=$.test(a)&&qa(b.parentNode)||b}if(r)try{return G.apply(d,s.querySelectorAll(r)),d}catch(x){}finally{k===u&&b.removeAttribute("id")}}}return i(a.replace(P,"$1"),b,d,e)}function ha(){var a=[];function b(c,e){return a.push(c+" ")>d.cacheLength&&delete b[a.shift()],b[c+" "]=e}return b}function ia(a){return a[u]=!0,a}function ja(a){var b=n.createElement("fieldset");try{return!!a(b)}catch(c){return!1}finally{b.parentNode&&b.parentNode.removeChild(b),b=null}}function ka(a,b){var c=a.split("|"),e=c.length;while(e--)d.attrHandle[c[e]]=b}function la(a,b){var c=b&&a,d=c&&1===a.nodeType&&1===b.nodeType&&a.sourceIndex-b.sourceIndex;if(d)return d;if(c)while(c=c.nextSibling)if(c===b)return-1;return a?1:-1}function ma(a){return function(b){var c=b.nodeName.toLowerCase();return"input"===c&&b.type===a}}function na(a){return function(b){var c=b.nodeName.toLowerCase();return("input"===c||"button"===c)&&b.type===a}}function oa(a){return function(b){return"form"in b?b.parentNode&&b.disabled===!1?"label"in b?"label"in b.parentNode?b.parentNode.disabled===a:b.disabled===a:b.isDisabled===a||b.isDisabled!==!a&&ea(b)===a:b.disabled===a:"label"in b&&b.disabled===a}}function pa(a){return ia(function(b){return b=+b,ia(function(c,d){var e,f=a([],c.length,b),g=f.length;while(g--)c[e=f[g]]&&(c[e]=!(d[e]=c[e]))})})}function qa(a){return a&&"undefined"!=typeof a.getElementsByTagName&&a}c=ga.support={},f=ga.isXML=function(a){var b=a&&(a.ownerDocument||a).documentElement;return!!b&&"HTML"!==b.nodeName},m=ga.setDocument=function(a){var b,e,g=a?a.ownerDocument||a:v;return g!==n&&9===g.nodeType&&g.documentElement?(n=g,o=n.documentElement,p=!f(n),v!==n&&(e=n.defaultView)&&e.top!==e&&(e.addEventListener?e.addEventListener("unload",da,!1):e.attachEvent&&e.attachEvent("onunload",da)),c.attributes=ja(function(a){return a.className="i",!a.getAttribute("className")}),c.getElementsByTagName=ja(function(a){return a.appendChild(n.createComment("")),!a.getElementsByTagName("*").length}),c.getElementsByClassName=Y.test(n.getElementsByClassName),c.getById=ja(function(a){return o.appendChild(a).id=u,!n.getElementsByName||!n.getElementsByName(u).length}),c.getById?(d.filter.ID=function(a){var b=a.replace(_,aa);return function(a){return a.getAttribute("id")===b}},d.find.ID=function(a,b){if("undefined"!=typeof b.getElementById&&p){var c=b.getElementById(a);return c?[c]:[]}}):(d.filter.ID=function(a){var b=a.replace(_,aa);return function(a){var c="undefined"!=typeof a.getAttributeNode&&a.getAttributeNode("id");return c&&c.value===b}},d.find.ID=function(a,b){if("undefined"!=typeof b.getElementById&&p){var c,d,e,f=b.getElementById(a);if(f){if(c=f.getAttributeNode("id"),c&&c.value===a)return[f];e=b.getElementsByName(a),d=0;while(f=e[d++])if(c=f.getAttributeNode("id"),c&&c.value===a)return[f]}return[]}}),d.find.TAG=c.getElementsByTagName?function(a,b){return"undefined"!=typeof b.getElementsByTagName?b.getElementsByTagName(a):c.qsa?b.querySelectorAll(a):void 0}:function(a,b){var c,d=[],e=0,f=b.getElementsByTagName(a);if("*"===a){while(c=f[e++])1===c.nodeType&&d.push(c);return d}return f},d.find.CLASS=c.getElementsByClassName&&function(a,b){if("undefined"!=typeof b.getElementsByClassName&&p)return b.getElementsByClassName(a)},r=[],q=[],(c.qsa=Y.test(n.querySelectorAll))&&(ja(function(a){o.appendChild(a).innerHTML="<a id='"+u+"'></a><select id='"+u+"-\r\\' msallowcapture=''><option selected=''></option></select>",a.querySelectorAll("[msallowcapture^='']").length&&q.push("[*^$]="+K+"*(?:''|\"\")"),a.querySelectorAll("[selected]").length||q.push("\\["+K+"*(?:value|"+J+")"),a.querySelectorAll("[id~="+u+"-]").length||q.push("~="),a.querySelectorAll(":checked").length||q.push(":checked"),a.querySelectorAll("a#"+u+"+*").length||q.push(".#.+[+~]")}),ja(function(a){a.innerHTML="<a href='' disabled='disabled'></a><select disabled='disabled'><option/></select>";var b=n.createElement("input");b.setAttribute("type","hidden"),a.appendChild(b).setAttribute("name","D"),a.querySelectorAll("[name=d]").length&&q.push("name"+K+"*[*^$|!~]?="),2!==a.querySelectorAll(":enabled").length&&q.push(":enabled",":disabled"),o.appendChild(a).disabled=!0,2!==a.querySelectorAll(":disabled").length&&q.push(":enabled",":disabled"),a.querySelectorAll("*,:x"),q.push(",.*:")})),(c.matchesSelector=Y.test(s=o.matches||o.webkitMatchesSelector||o.mozMatchesSelector||o.oMatchesSelector||o.msMatchesSelector))&&ja(function(a){c.disconnectedMatch=s.call(a,"*"),s.call(a,"[s!='']:x"),r.push("!=",N)}),q=q.length&&new RegExp(q.join("|")),r=r.length&&new RegExp(r.join("|")),b=Y.test(o.compareDocumentPosition),t=b||Y.test(o.contains)?function(a,b){var c=9===a.nodeType?a.documentElement:a,d=b&&b.parentNode;return a===d||!(!d||1!==d.nodeType||!(c.contains?c.contains(d):a.compareDocumentPosition&&16&a.compareDocumentPosition(d)))}:function(a,b){if(b)while(b=b.parentNode)if(b===a)return!0;return!1},B=b?function(a,b){if(a===b)return l=!0,0;var d=!a.compareDocumentPosition-!b.compareDocumentPosition;return d?d:(d=(a.ownerDocument||a)===(b.ownerDocument||b)?a.compareDocumentPosition(b):1,1&d||!c.sortDetached&&b.compareDocumentPosition(a)===d?a===n||a.ownerDocument===v&&t(v,a)?-1:b===n||b.ownerDocument===v&&t(v,b)?1:k?I(k,a)-I(k,b):0:4&d?-1:1)}:function(a,b){if(a===b)return l=!0,0;var c,d=0,e=a.parentNode,f=b.parentNode,g=[a],h=[b];if(!e||!f)return a===n?-1:b===n?1:e?-1:f?1:k?I(k,a)-I(k,b):0;if(e===f)return la(a,b);c=a;while(c=c.parentNode)g.unshift(c);c=b;while(c=c.parentNode)h.unshift(c);while(g[d]===h[d])d++;return d?la(g[d],h[d]):g[d]===v?-1:h[d]===v?1:0},n):n},ga.matches=function(a,b){return ga(a,null,null,b)},ga.matchesSelector=function(a,b){if((a.ownerDocument||a)!==n&&m(a),b=b.replace(S,"='$1']"),c.matchesSelector&&p&&!A[b+" "]&&(!r||!r.test(b))&&(!q||!q.test(b)))try{var d=s.call(a,b);if(d||c.disconnectedMatch||a.document&&11!==a.document.nodeType)return d}catch(e){}return ga(b,n,null,[a]).length>0},ga.contains=function(a,b){return(a.ownerDocument||a)!==n&&m(a),t(a,b)},ga.attr=function(a,b){(a.ownerDocument||a)!==n&&m(a);var e=d.attrHandle[b.toLowerCase()],f=e&&C.call(d.attrHandle,b.toLowerCase())?e(a,b,!p):void 0;return void 0!==f?f:c.attributes||!p?a.getAttribute(b):(f=a.getAttributeNode(b))&&f.specified?f.value:null},ga.escape=function(a){return(a+"").replace(ba,ca)},ga.error=function(a){throw new Error("Syntax error, unrecognized expression: "+a)},ga.uniqueSort=function(a){var b,d=[],e=0,f=0;if(l=!c.detectDuplicates,k=!c.sortStable&&a.slice(0),a.sort(B),l){while(b=a[f++])b===a[f]&&(e=d.push(f));while(e--)a.splice(d[e],1)}return k=null,a},e=ga.getText=function(a){var b,c="",d=0,f=a.nodeType;if(f){if(1===f||9===f||11===f){if("string"==typeof a.textContent)return a.textContent;for(a=a.firstChild;a;a=a.nextSibling)c+=e(a)}else if(3===f||4===f)return a.nodeValue}else while(b=a[d++])c+=e(b);return c},d=ga.selectors={cacheLength:50,createPseudo:ia,match:V,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(a){return a[1]=a[1].replace(_,aa),a[3]=(a[3]||a[4]||a[5]||"").replace(_,aa),"~="===a[2]&&(a[3]=" "+a[3]+" "),a.slice(0,4)},CHILD:function(a){return a[1]=a[1].toLowerCase(),"nth"===a[1].slice(0,3)?(a[3]||ga.error(a[0]),a[4]=+(a[4]?a[5]+(a[6]||1):2*("even"===a[3]||"odd"===a[3])),a[5]=+(a[7]+a[8]||"odd"===a[3])):a[3]&&ga.error(a[0]),a},PSEUDO:function(a){var b,c=!a[6]&&a[2];return V.CHILD.test(a[0])?null:(a[3]?a[2]=a[4]||a[5]||"":c&&T.test(c)&&(b=g(c,!0))&&(b=c.indexOf(")",c.length-b)-c.length)&&(a[0]=a[0].slice(0,b),a[2]=c.slice(0,b)),a.slice(0,3))}},filter:{TAG:function(a){var b=a.replace(_,aa).toLowerCase();return"*"===a?function(){return!0}:function(a){return a.nodeName&&a.nodeName.toLowerCase()===b}},CLASS:function(a){var b=y[a+" "];return b||(b=new RegExp("(^|"+K+")"+a+"("+K+"|$)"))&&y(a,function(a){return b.test("string"==typeof a.className&&a.className||"undefined"!=typeof a.getAttribute&&a.getAttribute("class")||"")})},ATTR:function(a,b,c){return function(d){var e=ga.attr(d,a);return null==e?"!="===b:!b||(e+="","="===b?e===c:"!="===b?e!==c:"^="===b?c&&0===e.indexOf(c):"*="===b?c&&e.indexOf(c)>-1:"$="===b?c&&e.slice(-c.length)===c:"~="===b?(" "+e.replace(O," ")+" ").indexOf(c)>-1:"|="===b&&(e===c||e.slice(0,c.length+1)===c+"-"))}},CHILD:function(a,b,c,d,e){var f="nth"!==a.slice(0,3),g="last"!==a.slice(-4),h="of-type"===b;return 1===d&&0===e?function(a){return!!a.parentNode}:function(b,c,i){var j,k,l,m,n,o,p=f!==g?"nextSibling":"previousSibling",q=b.parentNode,r=h&&b.nodeName.toLowerCase(),s=!i&&!h,t=!1;if(q){if(f){while(p){m=b;while(m=m[p])if(h?m.nodeName.toLowerCase()===r:1===m.nodeType)return!1;o=p="only"===a&&!o&&"nextSibling"}return!0}if(o=[g?q.firstChild:q.lastChild],g&&s){m=q,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n&&j[2],m=n&&q.childNodes[n];while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if(1===m.nodeType&&++t&&m===b){k[a]=[w,n,t];break}}else if(s&&(m=b,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n),t===!1)while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if((h?m.nodeName.toLowerCase()===r:1===m.nodeType)&&++t&&(s&&(l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),k[a]=[w,t]),m===b))break;return t-=e,t===d||t%d===0&&t/d>=0}}},PSEUDO:function(a,b){var c,e=d.pseudos[a]||d.setFilters[a.toLowerCase()]||ga.error("unsupported pseudo: "+a);return e[u]?e(b):e.length>1?(c=[a,a,"",b],d.setFilters.hasOwnProperty(a.toLowerCase())?ia(function(a,c){var d,f=e(a,b),g=f.length;while(g--)d=I(a,f[g]),a[d]=!(c[d]=f[g])}):function(a){return e(a,0,c)}):e}},pseudos:{not:ia(function(a){var b=[],c=[],d=h(a.replace(P,"$1"));return d[u]?ia(function(a,b,c,e){var f,g=d(a,null,e,[]),h=a.length;while(h--)(f=g[h])&&(a[h]=!(b[h]=f))}):function(a,e,f){return b[0]=a,d(b,null,f,c),b[0]=null,!c.pop()}}),has:ia(function(a){return function(b){return ga(a,b).length>0}}),contains:ia(function(a){return a=a.replace(_,aa),function(b){return(b.textContent||b.innerText||e(b)).indexOf(a)>-1}}),lang:ia(function(a){return U.test(a||"")||ga.error("unsupported lang: "+a),a=a.replace(_,aa).toLowerCase(),function(b){var c;do if(c=p?b.lang:b.getAttribute("xml:lang")||b.getAttribute("lang"))return c=c.toLowerCase(),c===a||0===c.indexOf(a+"-");while((b=b.parentNode)&&1===b.nodeType);return!1}}),target:function(b){var c=a.location&&a.location.hash;return c&&c.slice(1)===b.id},root:function(a){return a===o},focus:function(a){return a===n.activeElement&&(!n.hasFocus||n.hasFocus())&&!!(a.type||a.href||~a.tabIndex)},enabled:oa(!1),disabled:oa(!0),checked:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&!!a.checked||"option"===b&&!!a.selected},selected:function(a){return a.parentNode&&a.parentNode.selectedIndex,a.selected===!0},empty:function(a){for(a=a.firstChild;a;a=a.nextSibling)if(a.nodeType<6)return!1;return!0},parent:function(a){return!d.pseudos.empty(a)},header:function(a){return X.test(a.nodeName)},input:function(a){return W.test(a.nodeName)},button:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&"button"===a.type||"button"===b},text:function(a){var b;return"input"===a.nodeName.toLowerCase()&&"text"===a.type&&(null==(b=a.getAttribute("type"))||"text"===b.toLowerCase())},first:pa(function(){return[0]}),last:pa(function(a,b){return[b-1]}),eq:pa(function(a,b,c){return[c<0?c+b:c]}),even:pa(function(a,b){for(var c=0;c<b;c+=2)a.push(c);return a}),odd:pa(function(a,b){for(var c=1;c<b;c+=2)a.push(c);return a}),lt:pa(function(a,b,c){for(var d=c<0?c+b:c;--d>=0;)a.push(d);return a}),gt:pa(function(a,b,c){for(var d=c<0?c+b:c;++d<b;)a.push(d);return a})}},d.pseudos.nth=d.pseudos.eq;for(b in{radio:!0,checkbox:!0,file:!0,password:!0,image:!0})d.pseudos[b]=ma(b);for(b in{submit:!0,reset:!0})d.pseudos[b]=na(b);function ra(){}ra.prototype=d.filters=d.pseudos,d.setFilters=new ra,g=ga.tokenize=function(a,b){var c,e,f,g,h,i,j,k=z[a+" "];if(k)return b?0:k.slice(0);h=a,i=[],j=d.preFilter;while(h){c&&!(e=Q.exec(h))||(e&&(h=h.slice(e[0].length)||h),i.push(f=[])),c=!1,(e=R.exec(h))&&(c=e.shift(),f.push({value:c,type:e[0].replace(P," ")}),h=h.slice(c.length));for(g in d.filter)!(e=V[g].exec(h))||j[g]&&!(e=j[g](e))||(c=e.shift(),f.push({value:c,type:g,matches:e}),h=h.slice(c.length));if(!c)break}return b?h.length:h?ga.error(a):z(a,i).slice(0)};function sa(a){for(var b=0,c=a.length,d="";b<c;b++)d+=a[b].value;return d}function ta(a,b,c){var d=b.dir,e=b.next,f=e||d,g=c&&"parentNode"===f,h=x++;return b.first?function(b,c,e){while(b=b[d])if(1===b.nodeType||g)return a(b,c,e);return!1}:function(b,c,i){var j,k,l,m=[w,h];if(i){while(b=b[d])if((1===b.nodeType||g)&&a(b,c,i))return!0}else while(b=b[d])if(1===b.nodeType||g)if(l=b[u]||(b[u]={}),k=l[b.uniqueID]||(l[b.uniqueID]={}),e&&e===b.nodeName.toLowerCase())b=b[d]||b;else{if((j=k[f])&&j[0]===w&&j[1]===h)return m[2]=j[2];if(k[f]=m,m[2]=a(b,c,i))return!0}return!1}}function ua(a){return a.length>1?function(b,c,d){var e=a.length;while(e--)if(!a[e](b,c,d))return!1;return!0}:a[0]}function va(a,b,c){for(var d=0,e=b.length;d<e;d++)ga(a,b[d],c);return c}function wa(a,b,c,d,e){for(var f,g=[],h=0,i=a.length,j=null!=b;h<i;h++)(f=a[h])&&(c&&!c(f,d,e)||(g.push(f),j&&b.push(h)));return g}function xa(a,b,c,d,e,f){return d&&!d[u]&&(d=xa(d)),e&&!e[u]&&(e=xa(e,f)),ia(function(f,g,h,i){var j,k,l,m=[],n=[],o=g.length,p=f||va(b||"*",h.nodeType?[h]:h,[]),q=!a||!f&&b?p:wa(p,m,a,h,i),r=c?e||(f?a:o||d)?[]:g:q;if(c&&c(q,r,h,i),d){j=wa(r,n),d(j,[],h,i),k=j.length;while(k--)(l=j[k])&&(r[n[k]]=!(q[n[k]]=l))}if(f){if(e||a){if(e){j=[],k=r.length;while(k--)(l=r[k])&&j.push(q[k]=l);e(null,r=[],j,i)}k=r.length;while(k--)(l=r[k])&&(j=e?I(f,l):m[k])>-1&&(f[j]=!(g[j]=l))}}else r=wa(r===g?r.splice(o,r.length):r),e?e(null,g,r,i):G.apply(g,r)})}function ya(a){for(var b,c,e,f=a.length,g=d.relative[a[0].type],h=g||d.relative[" "],i=g?1:0,k=ta(function(a){return a===b},h,!0),l=ta(function(a){return I(b,a)>-1},h,!0),m=[function(a,c,d){var e=!g&&(d||c!==j)||((b=c).nodeType?k(a,c,d):l(a,c,d));return b=null,e}];i<f;i++)if(c=d.relative[a[i].type])m=[ta(ua(m),c)];else{if(c=d.filter[a[i].type].apply(null,a[i].matches),c[u]){for(e=++i;e<f;e++)if(d.relative[a[e].type])break;return xa(i>1&&ua(m),i>1&&sa(a.slice(0,i-1).concat({value:" "===a[i-2].type?"*":""})).replace(P,"$1"),c,i<e&&ya(a.slice(i,e)),e<f&&ya(a=a.slice(e)),e<f&&sa(a))}m.push(c)}return ua(m)}function za(a,b){var c=b.length>0,e=a.length>0,f=function(f,g,h,i,k){var l,o,q,r=0,s="0",t=f&&[],u=[],v=j,x=f||e&&d.find.TAG("*",k),y=w+=null==v?1:Math.random()||.1,z=x.length;for(k&&(j=g===n||g||k);s!==z&&null!=(l=x[s]);s++){if(e&&l){o=0,g||l.ownerDocument===n||(m(l),h=!p);while(q=a[o++])if(q(l,g||n,h)){i.push(l);break}k&&(w=y)}c&&((l=!q&&l)&&r--,f&&t.push(l))}if(r+=s,c&&s!==r){o=0;while(q=b[o++])q(t,u,g,h);if(f){if(r>0)while(s--)t[s]||u[s]||(u[s]=E.call(i));u=wa(u)}G.apply(i,u),k&&!f&&u.length>0&&r+b.length>1&&ga.uniqueSort(i)}return k&&(w=y,j=v),t};return c?ia(f):f}return h=ga.compile=function(a,b){var c,d=[],e=[],f=A[a+" "];if(!f){b||(b=g(a)),c=b.length;while(c--)f=ya(b[c]),f[u]?d.push(f):e.push(f);f=A(a,za(e,d)),f.selector=a}return f},i=ga.select=function(a,b,c,e){var f,i,j,k,l,m="function"==typeof a&&a,n=!e&&g(a=m.selector||a);if(c=c||[],1===n.length){if(i=n[0]=n[0].slice(0),i.length>2&&"ID"===(j=i[0]).type&&9===b.nodeType&&p&&d.relative[i[1].type]){if(b=(d.find.ID(j.matches[0].replace(_,aa),b)||[])[0],!b)return c;m&&(b=b.parentNode),a=a.slice(i.shift().value.length)}f=V.needsContext.test(a)?0:i.length;while(f--){if(j=i[f],d.relative[k=j.type])break;if((l=d.find[k])&&(e=l(j.matches[0].replace(_,aa),$.test(i[0].type)&&qa(b.parentNode)||b))){if(i.splice(f,1),a=e.length&&sa(i),!a)return G.apply(c,e),c;break}}}return(m||h(a,n))(e,b,!p,c,!b||$.test(a)&&qa(b.parentNode)||b),c},c.sortStable=u.split("").sort(B).join("")===u,c.detectDuplicates=!!l,m(),c.sortDetached=ja(function(a){return 1&a.compareDocumentPosition(n.createElement("fieldset"))}),ja(function(a){return a.innerHTML="<a href='#'></a>","#"===a.firstChild.getAttribute("href")})||ka("type|href|height|width",function(a,b,c){if(!c)return a.getAttribute(b,"type"===b.toLowerCase()?1:2)}),c.attributes&&ja(function(a){return a.innerHTML="<input/>",a.firstChild.setAttribute("value",""),""===a.firstChild.getAttribute("value")})||ka("value",function(a,b,c){if(!c&&"input"===a.nodeName.toLowerCase())return a.defaultValue}),ja(function(a){return null==a.getAttribute("disabled")})||ka(J,function(a,b,c){var d;if(!c)return a[b]===!0?b.toLowerCase():(d=a.getAttributeNode(b))&&d.specified?d.value:null}),ga}(a);r.find=x,r.expr=x.selectors,r.expr[":"]=r.expr.pseudos,r.uniqueSort=r.unique=x.uniqueSort,r.text=x.getText,r.isXMLDoc=x.isXML,r.contains=x.contains,r.escapeSelector=x.escape;var y=function(a,b,c){var d=[],e=void 0!==c;while((a=a[b])&&9!==a.nodeType)if(1===a.nodeType){if(e&&r(a).is(c))break;d.push(a)}return d},z=function(a,b){for(var c=[];a;a=a.nextSibling)1===a.nodeType&&a!==b&&c.push(a);return c},A=r.expr.match.needsContext;function B(a,b){return a.nodeName&&a.nodeName.toLowerCase()===b.toLowerCase()}var C=/^<([a-z][^\/\0>:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i,D=/^.[^:#\[\.,]*$/;function E(a,b,c){return r.isFunction(b)?r.grep(a,function(a,d){return!!b.call(a,d,a)!==c}):b.nodeType?r.grep(a,function(a){return a===b!==c}):"string"!=typeof b?r.grep(a,function(a){return i.call(b,a)>-1!==c}):D.test(b)?r.filter(b,a,c):(b=r.filter(b,a),r.grep(a,function(a){return i.call(b,a)>-1!==c&&1===a.nodeType}))}r.filter=function(a,b,c){var d=b[0];return c&&(a=":not("+a+")"),1===b.length&&1===d.nodeType?r.find.matchesSelector(d,a)?[d]:[]:r.find.matches(a,r.grep(b,function(a){return 1===a.nodeType}))},r.fn.extend({find:function(a){var b,c,d=this.length,e=this;if("string"!=typeof a)return this.pushStack(r(a).filter(function(){for(b=0;b<d;b++)if(r.contains(e[b],this))return!0}));for(c=this.pushStack([]),b=0;b<d;b++)r.find(a,e[b],c);return d>1?r.uniqueSort(c):c},filter:function(a){return this.pushStack(E(this,a||[],!1))},not:function(a){return this.pushStack(E(this,a||[],!0))},is:function(a){return!!E(this,"string"==typeof a&&A.test(a)?r(a):a||[],!1).length}});var F,G=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]+))$/,H=r.fn.init=function(a,b,c){var e,f;if(!a)return this;if(c=c||F,"string"==typeof a){if(e="<"===a[0]&&">"===a[a.length-1]&&a.length>=3?[null,a,null]:G.exec(a),!e||!e[1]&&b)return!b||b.jquery?(b||c).find(a):this.constructor(b).find(a);if(e[1]){if(b=b instanceof r?b[0]:b,r.merge(this,r.parseHTML(e[1],b&&b.nodeType?b.ownerDocument||b:d,!0)),C.test(e[1])&&r.isPlainObject(b))for(e in b)r.isFunction(this[e])?this[e](b[e]):this.attr(e,b[e]);return this}return f=d.getElementById(e[2]),f&&(this[0]=f,this.length=1),this}return a.nodeType?(this[0]=a,this.length=1,this):r.isFunction(a)?void 0!==c.ready?c.ready(a):a(r):r.makeArray(a,this)};H.prototype=r.fn,F=r(d);var I=/^(?:parents|prev(?:Until|All))/,J={children:!0,contents:!0,next:!0,prev:!0};r.fn.extend({has:function(a){var b=r(a,this),c=b.length;return this.filter(function(){for(var a=0;a<c;a++)if(r.contains(this,b[a]))return!0})},closest:function(a,b){var c,d=0,e=this.length,f=[],g="string"!=typeof a&&r(a);if(!A.test(a))for(;d<e;d++)for(c=this[d];c&&c!==b;c=c.parentNode)if(c.nodeType<11&&(g?g.index(c)>-1:1===c.nodeType&&r.find.matchesSelector(c,a))){f.push(c);break}return this.pushStack(f.length>1?r.uniqueSort(f):f)},index:function(a){return a?"string"==typeof a?i.call(r(a),this[0]):i.call(this,a.jquery?a[0]:a):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(a,b){return this.pushStack(r.uniqueSort(r.merge(this.get(),r(a,b))))},addBack:function(a){return this.add(null==a?this.prevObject:this.prevObject.filter(a))}});function K(a,b){while((a=a[b])&&1!==a.nodeType);return a}r.each({parent:function(a){var b=a.parentNode;return b&&11!==b.nodeType?b:null},parents:function(a){return y(a,"parentNode")},parentsUntil:function(a,b,c){return y(a,"parentNode",c)},next:function(a){return K(a,"nextSibling")},prev:function(a){return K(a,"previousSibling")},nextAll:function(a){return y(a,"nextSibling")},prevAll:function(a){return y(a,"previousSibling")},nextUntil:function(a,b,c){return y(a,"nextSibling",c)},prevUntil:function(a,b,c){return y(a,"previousSibling",c)},siblings:function(a){return z((a.parentNode||{}).firstChild,a)},children:function(a){return z(a.firstChild)},contents:function(a){return B(a,"iframe")?a.contentDocument:(B(a,"template")&&(a=a.content||a),r.merge([],a.childNodes))}},function(a,b){r.fn[a]=function(c,d){var e=r.map(this,b,c);return"Until"!==a.slice(-5)&&(d=c),d&&"string"==typeof d&&(e=r.filter(d,e)),this.length>1&&(J[a]||r.uniqueSort(e),I.test(a)&&e.reverse()),this.pushStack(e)}});var L=/[^\x20\t\r\n\f]+/g;function M(a){var b={};return r.each(a.match(L)||[],function(a,c){b[c]=!0}),b}r.Callbacks=function(a){a="string"==typeof a?M(a):r.extend({},a);var b,c,d,e,f=[],g=[],h=-1,i=function(){for(e=e||a.once,d=b=!0;g.length;h=-1){c=g.shift();while(++h<f.length)f[h].apply(c[0],c[1])===!1&&a.stopOnFalse&&(h=f.length,c=!1)}a.memory||(c=!1),b=!1,e&&(f=c?[]:"")},j={add:function(){return f&&(c&&!b&&(h=f.length-1,g.push(c)),function d(b){r.each(b,function(b,c){r.isFunction(c)?a.unique&&j.has(c)||f.push(c):c&&c.length&&"string"!==r.type(c)&&d(c)})}(arguments),c&&!b&&i()),this},remove:function(){return r.each(arguments,function(a,b){var c;while((c=r.inArray(b,f,c))>-1)f.splice(c,1),c<=h&&h--}),this},has:function(a){return a?r.inArray(a,f)>-1:f.length>0},empty:function(){return f&&(f=[]),this},disable:function(){return e=g=[],f=c="",this},disabled:function(){return!f},lock:function(){return e=g=[],c||b||(f=c=""),this},locked:function(){return!!e},fireWith:function(a,c){return e||(c=c||[],c=[a,c.slice?c.slice():c],g.push(c),b||i()),this},fire:function(){return j.fireWith(this,arguments),this},fired:function(){return!!d}};return j};function N(a){return a}function O(a){throw a}function P(a,b,c,d){var e;try{a&&r.isFunction(e=a.promise)?e.call(a).done(b).fail(c):a&&r.isFunction(e=a.then)?e.call(a,b,c):b.apply(void 0,[a].slice(d))}catch(a){c.apply(void 0,[a])}}r.extend({Deferred:function(b){var c=[["notify","progress",r.Callbacks("memory"),r.Callbacks("memory"),2],["resolve","done",r.Callbacks("once memory"),r.Callbacks("once memory"),0,"resolved"],["reject","fail",r.Callbacks("once memory"),r.Callbacks("once memory"),1,"rejected"]],d="pending",e={state:function(){return d},always:function(){return f.done(arguments).fail(arguments),this},"catch":function(a){return e.then(null,a)},pipe:function(){var a=arguments;return r.Deferred(function(b){r.each(c,function(c,d){var e=r.isFunction(a[d[4]])&&a[d[4]];f[d[1]](function(){var a=e&&e.apply(this,arguments);a&&r.isFunction(a.promise)?a.promise().progress(b.notify).done(b.resolve).fail(b.reject):b[d[0]+"With"](this,e?[a]:arguments)})}),a=null}).promise()},then:function(b,d,e){var f=0;function g(b,c,d,e){return function(){var h=this,i=arguments,j=function(){var a,j;if(!(b<f)){if(a=d.apply(h,i),a===c.promise())throw new TypeError("Thenable self-resolution");j=a&&("object"==typeof a||"function"==typeof a)&&a.then,r.isFunction(j)?e?j.call(a,g(f,c,N,e),g(f,c,O,e)):(f++,j.call(a,g(f,c,N,e),g(f,c,O,e),g(f,c,N,c.notifyWith))):(d!==N&&(h=void 0,i=[a]),(e||c.resolveWith)(h,i))}},k=e?j:function(){try{j()}catch(a){r.Deferred.exceptionHook&&r.Deferred.exceptionHook(a,k.stackTrace),b+1>=f&&(d!==O&&(h=void 0,i=[a]),c.rejectWith(h,i))}};b?k():(r.Deferred.getStackHook&&(k.stackTrace=r.Deferred.getStackHook()),a.setTimeout(k))}}return r.Deferred(function(a){c[0][3].add(g(0,a,r.isFunction(e)?e:N,a.notifyWith)),c[1][3].add(g(0,a,r.isFunction(b)?b:N)),c[2][3].add(g(0,a,r.isFunction(d)?d:O))}).promise()},promise:function(a){return null!=a?r.extend(a,e):e}},f={};return r.each(c,function(a,b){var g=b[2],h=b[5];e[b[1]]=g.add,h&&g.add(function(){d=h},c[3-a][2].disable,c[0][2].lock),g.add(b[3].fire),f[b[0]]=function(){return f[b[0]+"With"](this===f?void 0:this,arguments),this},f[b[0]+"With"]=g.fireWith}),e.promise(f),b&&b.call(f,f),f},when:function(a){var b=arguments.length,c=b,d=Array(c),e=f.call(arguments),g=r.Deferred(),h=function(a){return function(c){d[a]=this,e[a]=arguments.length>1?f.call(arguments):c,--b||g.resolveWith(d,e)}};if(b<=1&&(P(a,g.done(h(c)).resolve,g.reject,!b),"pending"===g.state()||r.isFunction(e[c]&&e[c].then)))return g.then();while(c--)P(e[c],h(c),g.reject);return g.promise()}});var Q=/^(Eval|Internal|Range|Reference|Syntax|Type|URI)Error$/;r.Deferred.exceptionHook=function(b,c){a.console&&a.console.warn&&b&&Q.test(b.name)&&a.console.warn("jQuery.Deferred exception: "+b.message,b.stack,c)},r.readyException=function(b){a.setTimeout(function(){throw b})};var R=r.Deferred();r.fn.ready=function(a){return R.then(a)["catch"](function(a){r.readyException(a)}),this},r.extend({isReady:!1,readyWait:1,ready:function(a){(a===!0?--r.readyWait:r.isReady)||(r.isReady=!0,a!==!0&&--r.readyWait>0||R.resolveWith(d,[r]))}}),r.ready.then=R.then;function S(){d.removeEventListener("DOMContentLoaded",S),
-a.removeEventListener("load",S),r.ready()}"complete"===d.readyState||"loading"!==d.readyState&&!d.documentElement.doScroll?a.setTimeout(r.ready):(d.addEventListener("DOMContentLoaded",S),a.addEventListener("load",S));var T=function(a,b,c,d,e,f,g){var h=0,i=a.length,j=null==c;if("object"===r.type(c)){e=!0;for(h in c)T(a,b,h,c[h],!0,f,g)}else if(void 0!==d&&(e=!0,r.isFunction(d)||(g=!0),j&&(g?(b.call(a,d),b=null):(j=b,b=function(a,b,c){return j.call(r(a),c)})),b))for(;h<i;h++)b(a[h],c,g?d:d.call(a[h],h,b(a[h],c)));return e?a:j?b.call(a):i?b(a[0],c):f},U=function(a){return 1===a.nodeType||9===a.nodeType||!+a.nodeType};function V(){this.expando=r.expando+V.uid++}V.uid=1,V.prototype={cache:function(a){var b=a[this.expando];return b||(b={},U(a)&&(a.nodeType?a[this.expando]=b:Object.defineProperty(a,this.expando,{value:b,configurable:!0}))),b},set:function(a,b,c){var d,e=this.cache(a);if("string"==typeof b)e[r.camelCase(b)]=c;else for(d in b)e[r.camelCase(d)]=b[d];return e},get:function(a,b){return void 0===b?this.cache(a):a[this.expando]&&a[this.expando][r.camelCase(b)]},access:function(a,b,c){return void 0===b||b&&"string"==typeof b&&void 0===c?this.get(a,b):(this.set(a,b,c),void 0!==c?c:b)},remove:function(a,b){var c,d=a[this.expando];if(void 0!==d){if(void 0!==b){Array.isArray(b)?b=b.map(r.camelCase):(b=r.camelCase(b),b=b in d?[b]:b.match(L)||[]),c=b.length;while(c--)delete d[b[c]]}(void 0===b||r.isEmptyObject(d))&&(a.nodeType?a[this.expando]=void 0:delete a[this.expando])}},hasData:function(a){var b=a[this.expando];return void 0!==b&&!r.isEmptyObject(b)}};var W=new V,X=new V,Y=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,Z=/[A-Z]/g;function $(a){return"true"===a||"false"!==a&&("null"===a?null:a===+a+""?+a:Y.test(a)?JSON.parse(a):a)}function _(a,b,c){var d;if(void 0===c&&1===a.nodeType)if(d="data-"+b.replace(Z,"-$&").toLowerCase(),c=a.getAttribute(d),"string"==typeof c){try{c=$(c)}catch(e){}X.set(a,b,c)}else c=void 0;return c}r.extend({hasData:function(a){return X.hasData(a)||W.hasData(a)},data:function(a,b,c){return X.access(a,b,c)},removeData:function(a,b){X.remove(a,b)},_data:function(a,b,c){return W.access(a,b,c)},_removeData:function(a,b){W.remove(a,b)}}),r.fn.extend({data:function(a,b){var c,d,e,f=this[0],g=f&&f.attributes;if(void 0===a){if(this.length&&(e=X.get(f),1===f.nodeType&&!W.get(f,"hasDataAttrs"))){c=g.length;while(c--)g[c]&&(d=g[c].name,0===d.indexOf("data-")&&(d=r.camelCase(d.slice(5)),_(f,d,e[d])));W.set(f,"hasDataAttrs",!0)}return e}return"object"==typeof a?this.each(function(){X.set(this,a)}):T(this,function(b){var c;if(f&&void 0===b){if(c=X.get(f,a),void 0!==c)return c;if(c=_(f,a),void 0!==c)return c}else this.each(function(){X.set(this,a,b)})},null,b,arguments.length>1,null,!0)},removeData:function(a){return this.each(function(){X.remove(this,a)})}}),r.extend({queue:function(a,b,c){var d;if(a)return b=(b||"fx")+"queue",d=W.get(a,b),c&&(!d||Array.isArray(c)?d=W.access(a,b,r.makeArray(c)):d.push(c)),d||[]},dequeue:function(a,b){b=b||"fx";var c=r.queue(a,b),d=c.length,e=c.shift(),f=r._queueHooks(a,b),g=function(){r.dequeue(a,b)};"inprogress"===e&&(e=c.shift(),d--),e&&("fx"===b&&c.unshift("inprogress"),delete f.stop,e.call(a,g,f)),!d&&f&&f.empty.fire()},_queueHooks:function(a,b){var c=b+"queueHooks";return W.get(a,c)||W.access(a,c,{empty:r.Callbacks("once memory").add(function(){W.remove(a,[b+"queue",c])})})}}),r.fn.extend({queue:function(a,b){var c=2;return"string"!=typeof a&&(b=a,a="fx",c--),arguments.length<c?r.queue(this[0],a):void 0===b?this:this.each(function(){var c=r.queue(this,a,b);r._queueHooks(this,a),"fx"===a&&"inprogress"!==c[0]&&r.dequeue(this,a)})},dequeue:function(a){return this.each(function(){r.dequeue(this,a)})},clearQueue:function(a){return this.queue(a||"fx",[])},promise:function(a,b){var c,d=1,e=r.Deferred(),f=this,g=this.length,h=function(){--d||e.resolveWith(f,[f])};"string"!=typeof a&&(b=a,a=void 0),a=a||"fx";while(g--)c=W.get(f[g],a+"queueHooks"),c&&c.empty&&(d++,c.empty.add(h));return h(),e.promise(b)}});var aa=/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/.source,ba=new RegExp("^(?:([+-])=|)("+aa+")([a-z%]*)$","i"),ca=["Top","Right","Bottom","Left"],da=function(a,b){return a=b||a,"none"===a.style.display||""===a.style.display&&r.contains(a.ownerDocument,a)&&"none"===r.css(a,"display")},ea=function(a,b,c,d){var e,f,g={};for(f in b)g[f]=a.style[f],a.style[f]=b[f];e=c.apply(a,d||[]);for(f in b)a.style[f]=g[f];return e};function fa(a,b,c,d){var e,f=1,g=20,h=d?function(){return d.cur()}:function(){return r.css(a,b,"")},i=h(),j=c&&c[3]||(r.cssNumber[b]?"":"px"),k=(r.cssNumber[b]||"px"!==j&&+i)&&ba.exec(r.css(a,b));if(k&&k[3]!==j){j=j||k[3],c=c||[],k=+i||1;do f=f||".5",k/=f,r.style(a,b,k+j);while(f!==(f=h()/i)&&1!==f&&--g)}return c&&(k=+k||+i||0,e=c[1]?k+(c[1]+1)*c[2]:+c[2],d&&(d.unit=j,d.start=k,d.end=e)),e}var ga={};function ha(a){var b,c=a.ownerDocument,d=a.nodeName,e=ga[d];return e?e:(b=c.body.appendChild(c.createElement(d)),e=r.css(b,"display"),b.parentNode.removeChild(b),"none"===e&&(e="block"),ga[d]=e,e)}function ia(a,b){for(var c,d,e=[],f=0,g=a.length;f<g;f++)d=a[f],d.style&&(c=d.style.display,b?("none"===c&&(e[f]=W.get(d,"display")||null,e[f]||(d.style.display="")),""===d.style.display&&da(d)&&(e[f]=ha(d))):"none"!==c&&(e[f]="none",W.set(d,"display",c)));for(f=0;f<g;f++)null!=e[f]&&(a[f].style.display=e[f]);return a}r.fn.extend({show:function(){return ia(this,!0)},hide:function(){return ia(this)},toggle:function(a){return"boolean"==typeof a?a?this.show():this.hide():this.each(function(){da(this)?r(this).show():r(this).hide()})}});var ja=/^(?:checkbox|radio)$/i,ka=/<([a-z][^\/\0>\x20\t\r\n\f]+)/i,la=/^$|\/(?:java|ecma)script/i,ma={option:[1,"<select multiple='multiple'>","</select>"],thead:[1,"<table>","</table>"],col:[2,"<table><colgroup>","</colgroup></table>"],tr:[2,"<table><tbody>","</tbody></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],_default:[0,"",""]};ma.optgroup=ma.option,ma.tbody=ma.tfoot=ma.colgroup=ma.caption=ma.thead,ma.th=ma.td;function na(a,b){var c;return c="undefined"!=typeof a.getElementsByTagName?a.getElementsByTagName(b||"*"):"undefined"!=typeof a.querySelectorAll?a.querySelectorAll(b||"*"):[],void 0===b||b&&B(a,b)?r.merge([a],c):c}function oa(a,b){for(var c=0,d=a.length;c<d;c++)W.set(a[c],"globalEval",!b||W.get(b[c],"globalEval"))}var pa=/<|&#?\w+;/;function qa(a,b,c,d,e){for(var f,g,h,i,j,k,l=b.createDocumentFragment(),m=[],n=0,o=a.length;n<o;n++)if(f=a[n],f||0===f)if("object"===r.type(f))r.merge(m,f.nodeType?[f]:f);else if(pa.test(f)){g=g||l.appendChild(b.createElement("div")),h=(ka.exec(f)||["",""])[1].toLowerCase(),i=ma[h]||ma._default,g.innerHTML=i[1]+r.htmlPrefilter(f)+i[2],k=i[0];while(k--)g=g.lastChild;r.merge(m,g.childNodes),g=l.firstChild,g.textContent=""}else m.push(b.createTextNode(f));l.textContent="",n=0;while(f=m[n++])if(d&&r.inArray(f,d)>-1)e&&e.push(f);else if(j=r.contains(f.ownerDocument,f),g=na(l.appendChild(f),"script"),j&&oa(g),c){k=0;while(f=g[k++])la.test(f.type||"")&&c.push(f)}return l}!function(){var a=d.createDocumentFragment(),b=a.appendChild(d.createElement("div")),c=d.createElement("input");c.setAttribute("type","radio"),c.setAttribute("checked","checked"),c.setAttribute("name","t"),b.appendChild(c),o.checkClone=b.cloneNode(!0).cloneNode(!0).lastChild.checked,b.innerHTML="<textarea>x</textarea>",o.noCloneChecked=!!b.cloneNode(!0).lastChild.defaultValue}();var ra=d.documentElement,sa=/^key/,ta=/^(?:mouse|pointer|contextmenu|drag|drop)|click/,ua=/^([^.]*)(?:\.(.+)|)/;function va(){return!0}function wa(){return!1}function xa(){try{return d.activeElement}catch(a){}}function ya(a,b,c,d,e,f){var g,h;if("object"==typeof b){"string"!=typeof c&&(d=d||c,c=void 0);for(h in b)ya(a,h,c,d,b[h],f);return a}if(null==d&&null==e?(e=c,d=c=void 0):null==e&&("string"==typeof c?(e=d,d=void 0):(e=d,d=c,c=void 0)),e===!1)e=wa;else if(!e)return a;return 1===f&&(g=e,e=function(a){return r().off(a),g.apply(this,arguments)},e.guid=g.guid||(g.guid=r.guid++)),a.each(function(){r.event.add(this,b,e,d,c)})}r.event={global:{},add:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,n,o,p,q=W.get(a);if(q){c.handler&&(f=c,c=f.handler,e=f.selector),e&&r.find.matchesSelector(ra,e),c.guid||(c.guid=r.guid++),(i=q.events)||(i=q.events={}),(g=q.handle)||(g=q.handle=function(b){return"undefined"!=typeof r&&r.event.triggered!==b.type?r.event.dispatch.apply(a,arguments):void 0}),b=(b||"").match(L)||[""],j=b.length;while(j--)h=ua.exec(b[j])||[],n=p=h[1],o=(h[2]||"").split(".").sort(),n&&(l=r.event.special[n]||{},n=(e?l.delegateType:l.bindType)||n,l=r.event.special[n]||{},k=r.extend({type:n,origType:p,data:d,handler:c,guid:c.guid,selector:e,needsContext:e&&r.expr.match.needsContext.test(e),namespace:o.join(".")},f),(m=i[n])||(m=i[n]=[],m.delegateCount=0,l.setup&&l.setup.call(a,d,o,g)!==!1||a.addEventListener&&a.addEventListener(n,g)),l.add&&(l.add.call(a,k),k.handler.guid||(k.handler.guid=c.guid)),e?m.splice(m.delegateCount++,0,k):m.push(k),r.event.global[n]=!0)}},remove:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,n,o,p,q=W.hasData(a)&&W.get(a);if(q&&(i=q.events)){b=(b||"").match(L)||[""],j=b.length;while(j--)if(h=ua.exec(b[j])||[],n=p=h[1],o=(h[2]||"").split(".").sort(),n){l=r.event.special[n]||{},n=(d?l.delegateType:l.bindType)||n,m=i[n]||[],h=h[2]&&new RegExp("(^|\\.)"+o.join("\\.(?:.*\\.|)")+"(\\.|$)"),g=f=m.length;while(f--)k=m[f],!e&&p!==k.origType||c&&c.guid!==k.guid||h&&!h.test(k.namespace)||d&&d!==k.selector&&("**"!==d||!k.selector)||(m.splice(f,1),k.selector&&m.delegateCount--,l.remove&&l.remove.call(a,k));g&&!m.length&&(l.teardown&&l.teardown.call(a,o,q.handle)!==!1||r.removeEvent(a,n,q.handle),delete i[n])}else for(n in i)r.event.remove(a,n+b[j],c,d,!0);r.isEmptyObject(i)&&W.remove(a,"handle events")}},dispatch:function(a){var b=r.event.fix(a),c,d,e,f,g,h,i=new Array(arguments.length),j=(W.get(this,"events")||{})[b.type]||[],k=r.event.special[b.type]||{};for(i[0]=b,c=1;c<arguments.length;c++)i[c]=arguments[c];if(b.delegateTarget=this,!k.preDispatch||k.preDispatch.call(this,b)!==!1){h=r.event.handlers.call(this,b,j),c=0;while((f=h[c++])&&!b.isPropagationStopped()){b.currentTarget=f.elem,d=0;while((g=f.handlers[d++])&&!b.isImmediatePropagationStopped())b.rnamespace&&!b.rnamespace.test(g.namespace)||(b.handleObj=g,b.data=g.data,e=((r.event.special[g.origType]||{}).handle||g.handler).apply(f.elem,i),void 0!==e&&(b.result=e)===!1&&(b.preventDefault(),b.stopPropagation()))}return k.postDispatch&&k.postDispatch.call(this,b),b.result}},handlers:function(a,b){var c,d,e,f,g,h=[],i=b.delegateCount,j=a.target;if(i&&j.nodeType&&!("click"===a.type&&a.button>=1))for(;j!==this;j=j.parentNode||this)if(1===j.nodeType&&("click"!==a.type||j.disabled!==!0)){for(f=[],g={},c=0;c<i;c++)d=b[c],e=d.selector+" ",void 0===g[e]&&(g[e]=d.needsContext?r(e,this).index(j)>-1:r.find(e,this,null,[j]).length),g[e]&&f.push(d);f.length&&h.push({elem:j,handlers:f})}return j=this,i<b.length&&h.push({elem:j,handlers:b.slice(i)}),h},addProp:function(a,b){Object.defineProperty(r.Event.prototype,a,{enumerable:!0,configurable:!0,get:r.isFunction(b)?function(){if(this.originalEvent)return b(this.originalEvent)}:function(){if(this.originalEvent)return this.originalEvent[a]},set:function(b){Object.defineProperty(this,a,{enumerable:!0,configurable:!0,writable:!0,value:b})}})},fix:function(a){return a[r.expando]?a:new r.Event(a)},special:{load:{noBubble:!0},focus:{trigger:function(){if(this!==xa()&&this.focus)return this.focus(),!1},delegateType:"focusin"},blur:{trigger:function(){if(this===xa()&&this.blur)return this.blur(),!1},delegateType:"focusout"},click:{trigger:function(){if("checkbox"===this.type&&this.click&&B(this,"input"))return this.click(),!1},_default:function(a){return B(a.target,"a")}},beforeunload:{postDispatch:function(a){void 0!==a.result&&a.originalEvent&&(a.originalEvent.returnValue=a.result)}}}},r.removeEvent=function(a,b,c){a.removeEventListener&&a.removeEventListener(b,c)},r.Event=function(a,b){return this instanceof r.Event?(a&&a.type?(this.originalEvent=a,this.type=a.type,this.isDefaultPrevented=a.defaultPrevented||void 0===a.defaultPrevented&&a.returnValue===!1?va:wa,this.target=a.target&&3===a.target.nodeType?a.target.parentNode:a.target,this.currentTarget=a.currentTarget,this.relatedTarget=a.relatedTarget):this.type=a,b&&r.extend(this,b),this.timeStamp=a&&a.timeStamp||r.now(),void(this[r.expando]=!0)):new r.Event(a,b)},r.Event.prototype={constructor:r.Event,isDefaultPrevented:wa,isPropagationStopped:wa,isImmediatePropagationStopped:wa,isSimulated:!1,preventDefault:function(){var a=this.originalEvent;this.isDefaultPrevented=va,a&&!this.isSimulated&&a.preventDefault()},stopPropagation:function(){var a=this.originalEvent;this.isPropagationStopped=va,a&&!this.isSimulated&&a.stopPropagation()},stopImmediatePropagation:function(){var a=this.originalEvent;this.isImmediatePropagationStopped=va,a&&!this.isSimulated&&a.stopImmediatePropagation(),this.stopPropagation()}},r.each({altKey:!0,bubbles:!0,cancelable:!0,changedTouches:!0,ctrlKey:!0,detail:!0,eventPhase:!0,metaKey:!0,pageX:!0,pageY:!0,shiftKey:!0,view:!0,"char":!0,charCode:!0,key:!0,keyCode:!0,button:!0,buttons:!0,clientX:!0,clientY:!0,offsetX:!0,offsetY:!0,pointerId:!0,pointerType:!0,screenX:!0,screenY:!0,targetTouches:!0,toElement:!0,touches:!0,which:function(a){var b=a.button;return null==a.which&&sa.test(a.type)?null!=a.charCode?a.charCode:a.keyCode:!a.which&&void 0!==b&&ta.test(a.type)?1&b?1:2&b?3:4&b?2:0:a.which}},r.event.addProp),r.each({mouseenter:"mouseover",mouseleave:"mouseout",pointerenter:"pointerover",pointerleave:"pointerout"},function(a,b){r.event.special[a]={delegateType:b,bindType:b,handle:function(a){var c,d=this,e=a.relatedTarget,f=a.handleObj;return e&&(e===d||r.contains(d,e))||(a.type=f.origType,c=f.handler.apply(this,arguments),a.type=b),c}}}),r.fn.extend({on:function(a,b,c,d){return ya(this,a,b,c,d)},one:function(a,b,c,d){return ya(this,a,b,c,d,1)},off:function(a,b,c){var d,e;if(a&&a.preventDefault&&a.handleObj)return d=a.handleObj,r(a.delegateTarget).off(d.namespace?d.origType+"."+d.namespace:d.origType,d.selector,d.handler),this;if("object"==typeof a){for(e in a)this.off(e,b,a[e]);return this}return b!==!1&&"function"!=typeof b||(c=b,b=void 0),c===!1&&(c=wa),this.each(function(){r.event.remove(this,a,c,b)})}});var za=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([a-z][^\/\0>\x20\t\r\n\f]*)[^>]*)\/>/gi,Aa=/<script|<style|<link/i,Ba=/checked\s*(?:[^=]|=\s*.checked.)/i,Ca=/^true\/(.*)/,Da=/^\s*<!(?:\[CDATA\[|--)|(?:\]\]|--)>\s*$/g;function Ea(a,b){return B(a,"table")&&B(11!==b.nodeType?b:b.firstChild,"tr")?r(">tbody",a)[0]||a:a}function Fa(a){return a.type=(null!==a.getAttribute("type"))+"/"+a.type,a}function Ga(a){var b=Ca.exec(a.type);return b?a.type=b[1]:a.removeAttribute("type"),a}function Ha(a,b){var c,d,e,f,g,h,i,j;if(1===b.nodeType){if(W.hasData(a)&&(f=W.access(a),g=W.set(b,f),j=f.events)){delete g.handle,g.events={};for(e in j)for(c=0,d=j[e].length;c<d;c++)r.event.add(b,e,j[e][c])}X.hasData(a)&&(h=X.access(a),i=r.extend({},h),X.set(b,i))}}function Ia(a,b){var c=b.nodeName.toLowerCase();"input"===c&&ja.test(a.type)?b.checked=a.checked:"input"!==c&&"textarea"!==c||(b.defaultValue=a.defaultValue)}function Ja(a,b,c,d){b=g.apply([],b);var e,f,h,i,j,k,l=0,m=a.length,n=m-1,q=b[0],s=r.isFunction(q);if(s||m>1&&"string"==typeof q&&!o.checkClone&&Ba.test(q))return a.each(function(e){var f=a.eq(e);s&&(b[0]=q.call(this,e,f.html())),Ja(f,b,c,d)});if(m&&(e=qa(b,a[0].ownerDocument,!1,a,d),f=e.firstChild,1===e.childNodes.length&&(e=f),f||d)){for(h=r.map(na(e,"script"),Fa),i=h.length;l<m;l++)j=e,l!==n&&(j=r.clone(j,!0,!0),i&&r.merge(h,na(j,"script"))),c.call(a[l],j,l);if(i)for(k=h[h.length-1].ownerDocument,r.map(h,Ga),l=0;l<i;l++)j=h[l],la.test(j.type||"")&&!W.access(j,"globalEval")&&r.contains(k,j)&&(j.src?r._evalUrl&&r._evalUrl(j.src):p(j.textContent.replace(Da,""),k))}return a}function Ka(a,b,c){for(var d,e=b?r.filter(b,a):a,f=0;null!=(d=e[f]);f++)c||1!==d.nodeType||r.cleanData(na(d)),d.parentNode&&(c&&r.contains(d.ownerDocument,d)&&oa(na(d,"script")),d.parentNode.removeChild(d));return a}r.extend({htmlPrefilter:function(a){return a.replace(za,"<$1></$2>")},clone:function(a,b,c){var d,e,f,g,h=a.cloneNode(!0),i=r.contains(a.ownerDocument,a);if(!(o.noCloneChecked||1!==a.nodeType&&11!==a.nodeType||r.isXMLDoc(a)))for(g=na(h),f=na(a),d=0,e=f.length;d<e;d++)Ia(f[d],g[d]);if(b)if(c)for(f=f||na(a),g=g||na(h),d=0,e=f.length;d<e;d++)Ha(f[d],g[d]);else Ha(a,h);return g=na(h,"script"),g.length>0&&oa(g,!i&&na(a,"script")),h},cleanData:function(a){for(var b,c,d,e=r.event.special,f=0;void 0!==(c=a[f]);f++)if(U(c)){if(b=c[W.expando]){if(b.events)for(d in b.events)e[d]?r.event.remove(c,d):r.removeEvent(c,d,b.handle);c[W.expando]=void 0}c[X.expando]&&(c[X.expando]=void 0)}}}),r.fn.extend({detach:function(a){return Ka(this,a,!0)},remove:function(a){return Ka(this,a)},text:function(a){return T(this,function(a){return void 0===a?r.text(this):this.empty().each(function(){1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||(this.textContent=a)})},null,a,arguments.length)},append:function(){return Ja(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=Ea(this,a);b.appendChild(a)}})},prepend:function(){return Ja(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=Ea(this,a);b.insertBefore(a,b.firstChild)}})},before:function(){return Ja(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this)})},after:function(){return Ja(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this.nextSibling)})},empty:function(){for(var a,b=0;null!=(a=this[b]);b++)1===a.nodeType&&(r.cleanData(na(a,!1)),a.textContent="");return this},clone:function(a,b){return a=null!=a&&a,b=null==b?a:b,this.map(function(){return r.clone(this,a,b)})},html:function(a){return T(this,function(a){var b=this[0]||{},c=0,d=this.length;if(void 0===a&&1===b.nodeType)return b.innerHTML;if("string"==typeof a&&!Aa.test(a)&&!ma[(ka.exec(a)||["",""])[1].toLowerCase()]){a=r.htmlPrefilter(a);try{for(;c<d;c++)b=this[c]||{},1===b.nodeType&&(r.cleanData(na(b,!1)),b.innerHTML=a);b=0}catch(e){}}b&&this.empty().append(a)},null,a,arguments.length)},replaceWith:function(){var a=[];return Ja(this,arguments,function(b){var c=this.parentNode;r.inArray(this,a)<0&&(r.cleanData(na(this)),c&&c.replaceChild(b,this))},a)}}),r.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){r.fn[a]=function(a){for(var c,d=[],e=r(a),f=e.length-1,g=0;g<=f;g++)c=g===f?this:this.clone(!0),r(e[g])[b](c),h.apply(d,c.get());return this.pushStack(d)}});var La=/^margin/,Ma=new RegExp("^("+aa+")(?!px)[a-z%]+$","i"),Na=function(b){var c=b.ownerDocument.defaultView;return c&&c.opener||(c=a),c.getComputedStyle(b)};!function(){function b(){if(i){i.style.cssText="box-sizing:border-box;position:relative;display:block;margin:auto;border:1px;padding:1px;top:1%;width:50%",i.innerHTML="",ra.appendChild(h);var b=a.getComputedStyle(i);c="1%"!==b.top,g="2px"===b.marginLeft,e="4px"===b.width,i.style.marginRight="50%",f="4px"===b.marginRight,ra.removeChild(h),i=null}}var c,e,f,g,h=d.createElement("div"),i=d.createElement("div");i.style&&(i.style.backgroundClip="content-box",i.cloneNode(!0).style.backgroundClip="",o.clearCloneStyle="content-box"===i.style.backgroundClip,h.style.cssText="border:0;width:8px;height:0;top:0;left:-9999px;padding:0;margin-top:1px;position:absolute",h.appendChild(i),r.extend(o,{pixelPosition:function(){return b(),c},boxSizingReliable:function(){return b(),e},pixelMarginRight:function(){return b(),f},reliableMarginLeft:function(){return b(),g}}))}();function Oa(a,b,c){var d,e,f,g,h=a.style;return c=c||Na(a),c&&(g=c.getPropertyValue(b)||c[b],""!==g||r.contains(a.ownerDocument,a)||(g=r.style(a,b)),!o.pixelMarginRight()&&Ma.test(g)&&La.test(b)&&(d=h.width,e=h.minWidth,f=h.maxWidth,h.minWidth=h.maxWidth=h.width=g,g=c.width,h.width=d,h.minWidth=e,h.maxWidth=f)),void 0!==g?g+"":g}function Pa(a,b){return{get:function(){return a()?void delete this.get:(this.get=b).apply(this,arguments)}}}var Qa=/^(none|table(?!-c[ea]).+)/,Ra=/^--/,Sa={position:"absolute",visibility:"hidden",display:"block"},Ta={letterSpacing:"0",fontWeight:"400"},Ua=["Webkit","Moz","ms"],Va=d.createElement("div").style;function Wa(a){if(a in Va)return a;var b=a[0].toUpperCase()+a.slice(1),c=Ua.length;while(c--)if(a=Ua[c]+b,a in Va)return a}function Xa(a){var b=r.cssProps[a];return b||(b=r.cssProps[a]=Wa(a)||a),b}function Ya(a,b,c){var d=ba.exec(b);return d?Math.max(0,d[2]-(c||0))+(d[3]||"px"):b}function Za(a,b,c,d,e){var f,g=0;for(f=c===(d?"border":"content")?4:"width"===b?1:0;f<4;f+=2)"margin"===c&&(g+=r.css(a,c+ca[f],!0,e)),d?("content"===c&&(g-=r.css(a,"padding"+ca[f],!0,e)),"margin"!==c&&(g-=r.css(a,"border"+ca[f]+"Width",!0,e))):(g+=r.css(a,"padding"+ca[f],!0,e),"padding"!==c&&(g+=r.css(a,"border"+ca[f]+"Width",!0,e)));return g}function $a(a,b,c){var d,e=Na(a),f=Oa(a,b,e),g="border-box"===r.css(a,"boxSizing",!1,e);return Ma.test(f)?f:(d=g&&(o.boxSizingReliable()||f===a.style[b]),"auto"===f&&(f=a["offset"+b[0].toUpperCase()+b.slice(1)]),f=parseFloat(f)||0,f+Za(a,b,c||(g?"border":"content"),d,e)+"px")}r.extend({cssHooks:{opacity:{get:function(a,b){if(b){var c=Oa(a,"opacity");return""===c?"1":c}}}},cssNumber:{animationIterationCount:!0,columnCount:!0,fillOpacity:!0,flexGrow:!0,flexShrink:!0,fontWeight:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":"cssFloat"},style:function(a,b,c,d){if(a&&3!==a.nodeType&&8!==a.nodeType&&a.style){var e,f,g,h=r.camelCase(b),i=Ra.test(b),j=a.style;return i||(b=Xa(h)),g=r.cssHooks[b]||r.cssHooks[h],void 0===c?g&&"get"in g&&void 0!==(e=g.get(a,!1,d))?e:j[b]:(f=typeof c,"string"===f&&(e=ba.exec(c))&&e[1]&&(c=fa(a,b,e),f="number"),null!=c&&c===c&&("number"===f&&(c+=e&&e[3]||(r.cssNumber[h]?"":"px")),o.clearCloneStyle||""!==c||0!==b.indexOf("background")||(j[b]="inherit"),g&&"set"in g&&void 0===(c=g.set(a,c,d))||(i?j.setProperty(b,c):j[b]=c)),void 0)}},css:function(a,b,c,d){var e,f,g,h=r.camelCase(b),i=Ra.test(b);return i||(b=Xa(h)),g=r.cssHooks[b]||r.cssHooks[h],g&&"get"in g&&(e=g.get(a,!0,c)),void 0===e&&(e=Oa(a,b,d)),"normal"===e&&b in Ta&&(e=Ta[b]),""===c||c?(f=parseFloat(e),c===!0||isFinite(f)?f||0:e):e}}),r.each(["height","width"],function(a,b){r.cssHooks[b]={get:function(a,c,d){if(c)return!Qa.test(r.css(a,"display"))||a.getClientRects().length&&a.getBoundingClientRect().width?$a(a,b,d):ea(a,Sa,function(){return $a(a,b,d)})},set:function(a,c,d){var e,f=d&&Na(a),g=d&&Za(a,b,d,"border-box"===r.css(a,"boxSizing",!1,f),f);return g&&(e=ba.exec(c))&&"px"!==(e[3]||"px")&&(a.style[b]=c,c=r.css(a,b)),Ya(a,c,g)}}}),r.cssHooks.marginLeft=Pa(o.reliableMarginLeft,function(a,b){if(b)return(parseFloat(Oa(a,"marginLeft"))||a.getBoundingClientRect().left-ea(a,{marginLeft:0},function(){return a.getBoundingClientRect().left}))+"px"}),r.each({margin:"",padding:"",border:"Width"},function(a,b){r.cssHooks[a+b]={expand:function(c){for(var d=0,e={},f="string"==typeof c?c.split(" "):[c];d<4;d++)e[a+ca[d]+b]=f[d]||f[d-2]||f[0];return e}},La.test(a)||(r.cssHooks[a+b].set=Ya)}),r.fn.extend({css:function(a,b){return T(this,function(a,b,c){var d,e,f={},g=0;if(Array.isArray(b)){for(d=Na(a),e=b.length;g<e;g++)f[b[g]]=r.css(a,b[g],!1,d);return f}return void 0!==c?r.style(a,b,c):r.css(a,b)},a,b,arguments.length>1)}});function _a(a,b,c,d,e){return new _a.prototype.init(a,b,c,d,e)}r.Tween=_a,_a.prototype={constructor:_a,init:function(a,b,c,d,e,f){this.elem=a,this.prop=c,this.easing=e||r.easing._default,this.options=b,this.start=this.now=this.cur(),this.end=d,this.unit=f||(r.cssNumber[c]?"":"px")},cur:function(){var a=_a.propHooks[this.prop];return a&&a.get?a.get(this):_a.propHooks._default.get(this)},run:function(a){var b,c=_a.propHooks[this.prop];return this.options.duration?this.pos=b=r.easing[this.easing](a,this.options.duration*a,0,1,this.options.duration):this.pos=b=a,this.now=(this.end-this.start)*b+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),c&&c.set?c.set(this):_a.propHooks._default.set(this),this}},_a.prototype.init.prototype=_a.prototype,_a.propHooks={_default:{get:function(a){var b;return 1!==a.elem.nodeType||null!=a.elem[a.prop]&&null==a.elem.style[a.prop]?a.elem[a.prop]:(b=r.css(a.elem,a.prop,""),b&&"auto"!==b?b:0)},set:function(a){r.fx.step[a.prop]?r.fx.step[a.prop](a):1!==a.elem.nodeType||null==a.elem.style[r.cssProps[a.prop]]&&!r.cssHooks[a.prop]?a.elem[a.prop]=a.now:r.style(a.elem,a.prop,a.now+a.unit)}}},_a.propHooks.scrollTop=_a.propHooks.scrollLeft={set:function(a){a.elem.nodeType&&a.elem.parentNode&&(a.elem[a.prop]=a.now)}},r.easing={linear:function(a){return a},swing:function(a){return.5-Math.cos(a*Math.PI)/2},_default:"swing"},r.fx=_a.prototype.init,r.fx.step={};var ab,bb,cb=/^(?:toggle|show|hide)$/,db=/queueHooks$/;function eb(){bb&&(d.hidden===!1&&a.requestAnimationFrame?a.requestAnimationFrame(eb):a.setTimeout(eb,r.fx.interval),r.fx.tick())}function fb(){return a.setTimeout(function(){ab=void 0}),ab=r.now()}function gb(a,b){var c,d=0,e={height:a};for(b=b?1:0;d<4;d+=2-b)c=ca[d],e["margin"+c]=e["padding"+c]=a;return b&&(e.opacity=e.width=a),e}function hb(a,b,c){for(var d,e=(kb.tweeners[b]||[]).concat(kb.tweeners["*"]),f=0,g=e.length;f<g;f++)if(d=e[f].call(c,b,a))return d}function ib(a,b,c){var d,e,f,g,h,i,j,k,l="width"in b||"height"in b,m=this,n={},o=a.style,p=a.nodeType&&da(a),q=W.get(a,"fxshow");c.queue||(g=r._queueHooks(a,"fx"),null==g.unqueued&&(g.unqueued=0,h=g.empty.fire,g.empty.fire=function(){g.unqueued||h()}),g.unqueued++,m.always(function(){m.always(function(){g.unqueued--,r.queue(a,"fx").length||g.empty.fire()})}));for(d in b)if(e=b[d],cb.test(e)){if(delete b[d],f=f||"toggle"===e,e===(p?"hide":"show")){if("show"!==e||!q||void 0===q[d])continue;p=!0}n[d]=q&&q[d]||r.style(a,d)}if(i=!r.isEmptyObject(b),i||!r.isEmptyObject(n)){l&&1===a.nodeType&&(c.overflow=[o.overflow,o.overflowX,o.overflowY],j=q&&q.display,null==j&&(j=W.get(a,"display")),k=r.css(a,"display"),"none"===k&&(j?k=j:(ia([a],!0),j=a.style.display||j,k=r.css(a,"display"),ia([a]))),("inline"===k||"inline-block"===k&&null!=j)&&"none"===r.css(a,"float")&&(i||(m.done(function(){o.display=j}),null==j&&(k=o.display,j="none"===k?"":k)),o.display="inline-block")),c.overflow&&(o.overflow="hidden",m.always(function(){o.overflow=c.overflow[0],o.overflowX=c.overflow[1],o.overflowY=c.overflow[2]})),i=!1;for(d in n)i||(q?"hidden"in q&&(p=q.hidden):q=W.access(a,"fxshow",{display:j}),f&&(q.hidden=!p),p&&ia([a],!0),m.done(function(){p||ia([a]),W.remove(a,"fxshow");for(d in n)r.style(a,d,n[d])})),i=hb(p?q[d]:0,d,m),d in q||(q[d]=i.start,p&&(i.end=i.start,i.start=0))}}function jb(a,b){var c,d,e,f,g;for(c in a)if(d=r.camelCase(c),e=b[d],f=a[c],Array.isArray(f)&&(e=f[1],f=a[c]=f[0]),c!==d&&(a[d]=f,delete a[c]),g=r.cssHooks[d],g&&"expand"in g){f=g.expand(f),delete a[d];for(c in f)c in a||(a[c]=f[c],b[c]=e)}else b[d]=e}function kb(a,b,c){var d,e,f=0,g=kb.prefilters.length,h=r.Deferred().always(function(){delete i.elem}),i=function(){if(e)return!1;for(var b=ab||fb(),c=Math.max(0,j.startTime+j.duration-b),d=c/j.duration||0,f=1-d,g=0,i=j.tweens.length;g<i;g++)j.tweens[g].run(f);return h.notifyWith(a,[j,f,c]),f<1&&i?c:(i||h.notifyWith(a,[j,1,0]),h.resolveWith(a,[j]),!1)},j=h.promise({elem:a,props:r.extend({},b),opts:r.extend(!0,{specialEasing:{},easing:r.easing._default},c),originalProperties:b,originalOptions:c,startTime:ab||fb(),duration:c.duration,tweens:[],createTween:function(b,c){var d=r.Tween(a,j.opts,b,c,j.opts.specialEasing[b]||j.opts.easing);return j.tweens.push(d),d},stop:function(b){var c=0,d=b?j.tweens.length:0;if(e)return this;for(e=!0;c<d;c++)j.tweens[c].run(1);return b?(h.notifyWith(a,[j,1,0]),h.resolveWith(a,[j,b])):h.rejectWith(a,[j,b]),this}}),k=j.props;for(jb(k,j.opts.specialEasing);f<g;f++)if(d=kb.prefilters[f].call(j,a,k,j.opts))return r.isFunction(d.stop)&&(r._queueHooks(j.elem,j.opts.queue).stop=r.proxy(d.stop,d)),d;return r.map(k,hb,j),r.isFunction(j.opts.start)&&j.opts.start.call(a,j),j.progress(j.opts.progress).done(j.opts.done,j.opts.complete).fail(j.opts.fail).always(j.opts.always),r.fx.timer(r.extend(i,{elem:a,anim:j,queue:j.opts.queue})),j}r.Animation=r.extend(kb,{tweeners:{"*":[function(a,b){var c=this.createTween(a,b);return fa(c.elem,a,ba.exec(b),c),c}]},tweener:function(a,b){r.isFunction(a)?(b=a,a=["*"]):a=a.match(L);for(var c,d=0,e=a.length;d<e;d++)c=a[d],kb.tweeners[c]=kb.tweeners[c]||[],kb.tweeners[c].unshift(b)},prefilters:[ib],prefilter:function(a,b){b?kb.prefilters.unshift(a):kb.prefilters.push(a)}}),r.speed=function(a,b,c){var d=a&&"object"==typeof a?r.extend({},a):{complete:c||!c&&b||r.isFunction(a)&&a,duration:a,easing:c&&b||b&&!r.isFunction(b)&&b};return r.fx.off?d.duration=0:"number"!=typeof d.duration&&(d.duration in r.fx.speeds?d.duration=r.fx.speeds[d.duration]:d.duration=r.fx.speeds._default),null!=d.queue&&d.queue!==!0||(d.queue="fx"),d.old=d.complete,d.complete=function(){r.isFunction(d.old)&&d.old.call(this),d.queue&&r.dequeue(this,d.queue)},d},r.fn.extend({fadeTo:function(a,b,c,d){return this.filter(da).css("opacity",0).show().end().animate({opacity:b},a,c,d)},animate:function(a,b,c,d){var e=r.isEmptyObject(a),f=r.speed(b,c,d),g=function(){var b=kb(this,r.extend({},a),f);(e||W.get(this,"finish"))&&b.stop(!0)};return g.finish=g,e||f.queue===!1?this.each(g):this.queue(f.queue,g)},stop:function(a,b,c){var d=function(a){var b=a.stop;delete a.stop,b(c)};return"string"!=typeof a&&(c=b,b=a,a=void 0),b&&a!==!1&&this.queue(a||"fx",[]),this.each(function(){var b=!0,e=null!=a&&a+"queueHooks",f=r.timers,g=W.get(this);if(e)g[e]&&g[e].stop&&d(g[e]);else for(e in g)g[e]&&g[e].stop&&db.test(e)&&d(g[e]);for(e=f.length;e--;)f[e].elem!==this||null!=a&&f[e].queue!==a||(f[e].anim.stop(c),b=!1,f.splice(e,1));!b&&c||r.dequeue(this,a)})},finish:function(a){return a!==!1&&(a=a||"fx"),this.each(function(){var b,c=W.get(this),d=c[a+"queue"],e=c[a+"queueHooks"],f=r.timers,g=d?d.length:0;for(c.finish=!0,r.queue(this,a,[]),e&&e.stop&&e.stop.call(this,!0),b=f.length;b--;)f[b].elem===this&&f[b].queue===a&&(f[b].anim.stop(!0),f.splice(b,1));for(b=0;b<g;b++)d[b]&&d[b].finish&&d[b].finish.call(this);delete c.finish})}}),r.each(["toggle","show","hide"],function(a,b){var c=r.fn[b];r.fn[b]=function(a,d,e){return null==a||"boolean"==typeof a?c.apply(this,arguments):this.animate(gb(b,!0),a,d,e)}}),r.each({slideDown:gb("show"),slideUp:gb("hide"),slideToggle:gb("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(a,b){r.fn[a]=function(a,c,d){return this.animate(b,a,c,d)}}),r.timers=[],r.fx.tick=function(){var a,b=0,c=r.timers;for(ab=r.now();b<c.length;b++)a=c[b],a()||c[b]!==a||c.splice(b--,1);c.length||r.fx.stop(),ab=void 0},r.fx.timer=function(a){r.timers.push(a),r.fx.start()},r.fx.interval=13,r.fx.start=function(){bb||(bb=!0,eb())},r.fx.stop=function(){bb=null},r.fx.speeds={slow:600,fast:200,_default:400},r.fn.delay=function(b,c){return b=r.fx?r.fx.speeds[b]||b:b,c=c||"fx",this.queue(c,function(c,d){var e=a.setTimeout(c,b);d.stop=function(){a.clearTimeout(e)}})},function(){var a=d.createElement("input"),b=d.createElement("select"),c=b.appendChild(d.createElement("option"));a.type="checkbox",o.checkOn=""!==a.value,o.optSelected=c.selected,a=d.createElement("input"),a.value="t",a.type="radio",o.radioValue="t"===a.value}();var lb,mb=r.expr.attrHandle;r.fn.extend({attr:function(a,b){return T(this,r.attr,a,b,arguments.length>1)},removeAttr:function(a){return this.each(function(){r.removeAttr(this,a)})}}),r.extend({attr:function(a,b,c){var d,e,f=a.nodeType;if(3!==f&&8!==f&&2!==f)return"undefined"==typeof a.getAttribute?r.prop(a,b,c):(1===f&&r.isXMLDoc(a)||(e=r.attrHooks[b.toLowerCase()]||(r.expr.match.bool.test(b)?lb:void 0)),void 0!==c?null===c?void r.removeAttr(a,b):e&&"set"in e&&void 0!==(d=e.set(a,c,b))?d:(a.setAttribute(b,c+""),c):e&&"get"in e&&null!==(d=e.get(a,b))?d:(d=r.find.attr(a,b),
-null==d?void 0:d))},attrHooks:{type:{set:function(a,b){if(!o.radioValue&&"radio"===b&&B(a,"input")){var c=a.value;return a.setAttribute("type",b),c&&(a.value=c),b}}}},removeAttr:function(a,b){var c,d=0,e=b&&b.match(L);if(e&&1===a.nodeType)while(c=e[d++])a.removeAttribute(c)}}),lb={set:function(a,b,c){return b===!1?r.removeAttr(a,c):a.setAttribute(c,c),c}},r.each(r.expr.match.bool.source.match(/\w+/g),function(a,b){var c=mb[b]||r.find.attr;mb[b]=function(a,b,d){var e,f,g=b.toLowerCase();return d||(f=mb[g],mb[g]=e,e=null!=c(a,b,d)?g:null,mb[g]=f),e}});var nb=/^(?:input|select|textarea|button)$/i,ob=/^(?:a|area)$/i;r.fn.extend({prop:function(a,b){return T(this,r.prop,a,b,arguments.length>1)},removeProp:function(a){return this.each(function(){delete this[r.propFix[a]||a]})}}),r.extend({prop:function(a,b,c){var d,e,f=a.nodeType;if(3!==f&&8!==f&&2!==f)return 1===f&&r.isXMLDoc(a)||(b=r.propFix[b]||b,e=r.propHooks[b]),void 0!==c?e&&"set"in e&&void 0!==(d=e.set(a,c,b))?d:a[b]=c:e&&"get"in e&&null!==(d=e.get(a,b))?d:a[b]},propHooks:{tabIndex:{get:function(a){var b=r.find.attr(a,"tabindex");return b?parseInt(b,10):nb.test(a.nodeName)||ob.test(a.nodeName)&&a.href?0:-1}}},propFix:{"for":"htmlFor","class":"className"}}),o.optSelected||(r.propHooks.selected={get:function(a){var b=a.parentNode;return b&&b.parentNode&&b.parentNode.selectedIndex,null},set:function(a){var b=a.parentNode;b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex)}}),r.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){r.propFix[this.toLowerCase()]=this});function pb(a){var b=a.match(L)||[];return b.join(" ")}function qb(a){return a.getAttribute&&a.getAttribute("class")||""}r.fn.extend({addClass:function(a){var b,c,d,e,f,g,h,i=0;if(r.isFunction(a))return this.each(function(b){r(this).addClass(a.call(this,b,qb(this)))});if("string"==typeof a&&a){b=a.match(L)||[];while(c=this[i++])if(e=qb(c),d=1===c.nodeType&&" "+pb(e)+" "){g=0;while(f=b[g++])d.indexOf(" "+f+" ")<0&&(d+=f+" ");h=pb(d),e!==h&&c.setAttribute("class",h)}}return this},removeClass:function(a){var b,c,d,e,f,g,h,i=0;if(r.isFunction(a))return this.each(function(b){r(this).removeClass(a.call(this,b,qb(this)))});if(!arguments.length)return this.attr("class","");if("string"==typeof a&&a){b=a.match(L)||[];while(c=this[i++])if(e=qb(c),d=1===c.nodeType&&" "+pb(e)+" "){g=0;while(f=b[g++])while(d.indexOf(" "+f+" ")>-1)d=d.replace(" "+f+" "," ");h=pb(d),e!==h&&c.setAttribute("class",h)}}return this},toggleClass:function(a,b){var c=typeof a;return"boolean"==typeof b&&"string"===c?b?this.addClass(a):this.removeClass(a):r.isFunction(a)?this.each(function(c){r(this).toggleClass(a.call(this,c,qb(this),b),b)}):this.each(function(){var b,d,e,f;if("string"===c){d=0,e=r(this),f=a.match(L)||[];while(b=f[d++])e.hasClass(b)?e.removeClass(b):e.addClass(b)}else void 0!==a&&"boolean"!==c||(b=qb(this),b&&W.set(this,"__className__",b),this.setAttribute&&this.setAttribute("class",b||a===!1?"":W.get(this,"__className__")||""))})},hasClass:function(a){var b,c,d=0;b=" "+a+" ";while(c=this[d++])if(1===c.nodeType&&(" "+pb(qb(c))+" ").indexOf(b)>-1)return!0;return!1}});var rb=/\r/g;r.fn.extend({val:function(a){var b,c,d,e=this[0];{if(arguments.length)return d=r.isFunction(a),this.each(function(c){var e;1===this.nodeType&&(e=d?a.call(this,c,r(this).val()):a,null==e?e="":"number"==typeof e?e+="":Array.isArray(e)&&(e=r.map(e,function(a){return null==a?"":a+""})),b=r.valHooks[this.type]||r.valHooks[this.nodeName.toLowerCase()],b&&"set"in b&&void 0!==b.set(this,e,"value")||(this.value=e))});if(e)return b=r.valHooks[e.type]||r.valHooks[e.nodeName.toLowerCase()],b&&"get"in b&&void 0!==(c=b.get(e,"value"))?c:(c=e.value,"string"==typeof c?c.replace(rb,""):null==c?"":c)}}}),r.extend({valHooks:{option:{get:function(a){var b=r.find.attr(a,"value");return null!=b?b:pb(r.text(a))}},select:{get:function(a){var b,c,d,e=a.options,f=a.selectedIndex,g="select-one"===a.type,h=g?null:[],i=g?f+1:e.length;for(d=f<0?i:g?f:0;d<i;d++)if(c=e[d],(c.selected||d===f)&&!c.disabled&&(!c.parentNode.disabled||!B(c.parentNode,"optgroup"))){if(b=r(c).val(),g)return b;h.push(b)}return h},set:function(a,b){var c,d,e=a.options,f=r.makeArray(b),g=e.length;while(g--)d=e[g],(d.selected=r.inArray(r.valHooks.option.get(d),f)>-1)&&(c=!0);return c||(a.selectedIndex=-1),f}}}}),r.each(["radio","checkbox"],function(){r.valHooks[this]={set:function(a,b){if(Array.isArray(b))return a.checked=r.inArray(r(a).val(),b)>-1}},o.checkOn||(r.valHooks[this].get=function(a){return null===a.getAttribute("value")?"on":a.value})});var sb=/^(?:focusinfocus|focusoutblur)$/;r.extend(r.event,{trigger:function(b,c,e,f){var g,h,i,j,k,m,n,o=[e||d],p=l.call(b,"type")?b.type:b,q=l.call(b,"namespace")?b.namespace.split("."):[];if(h=i=e=e||d,3!==e.nodeType&&8!==e.nodeType&&!sb.test(p+r.event.triggered)&&(p.indexOf(".")>-1&&(q=p.split("."),p=q.shift(),q.sort()),k=p.indexOf(":")<0&&"on"+p,b=b[r.expando]?b:new r.Event(p,"object"==typeof b&&b),b.isTrigger=f?2:3,b.namespace=q.join("."),b.rnamespace=b.namespace?new RegExp("(^|\\.)"+q.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,b.result=void 0,b.target||(b.target=e),c=null==c?[b]:r.makeArray(c,[b]),n=r.event.special[p]||{},f||!n.trigger||n.trigger.apply(e,c)!==!1)){if(!f&&!n.noBubble&&!r.isWindow(e)){for(j=n.delegateType||p,sb.test(j+p)||(h=h.parentNode);h;h=h.parentNode)o.push(h),i=h;i===(e.ownerDocument||d)&&o.push(i.defaultView||i.parentWindow||a)}g=0;while((h=o[g++])&&!b.isPropagationStopped())b.type=g>1?j:n.bindType||p,m=(W.get(h,"events")||{})[b.type]&&W.get(h,"handle"),m&&m.apply(h,c),m=k&&h[k],m&&m.apply&&U(h)&&(b.result=m.apply(h,c),b.result===!1&&b.preventDefault());return b.type=p,f||b.isDefaultPrevented()||n._default&&n._default.apply(o.pop(),c)!==!1||!U(e)||k&&r.isFunction(e[p])&&!r.isWindow(e)&&(i=e[k],i&&(e[k]=null),r.event.triggered=p,e[p](),r.event.triggered=void 0,i&&(e[k]=i)),b.result}},simulate:function(a,b,c){var d=r.extend(new r.Event,c,{type:a,isSimulated:!0});r.event.trigger(d,null,b)}}),r.fn.extend({trigger:function(a,b){return this.each(function(){r.event.trigger(a,b,this)})},triggerHandler:function(a,b){var c=this[0];if(c)return r.event.trigger(a,b,c,!0)}}),r.each("blur focus focusin focusout resize scroll click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup contextmenu".split(" "),function(a,b){r.fn[b]=function(a,c){return arguments.length>0?this.on(b,null,a,c):this.trigger(b)}}),r.fn.extend({hover:function(a,b){return this.mouseenter(a).mouseleave(b||a)}}),o.focusin="onfocusin"in a,o.focusin||r.each({focus:"focusin",blur:"focusout"},function(a,b){var c=function(a){r.event.simulate(b,a.target,r.event.fix(a))};r.event.special[b]={setup:function(){var d=this.ownerDocument||this,e=W.access(d,b);e||d.addEventListener(a,c,!0),W.access(d,b,(e||0)+1)},teardown:function(){var d=this.ownerDocument||this,e=W.access(d,b)-1;e?W.access(d,b,e):(d.removeEventListener(a,c,!0),W.remove(d,b))}}});var tb=a.location,ub=r.now(),vb=/\?/;r.parseXML=function(b){var c;if(!b||"string"!=typeof b)return null;try{c=(new a.DOMParser).parseFromString(b,"text/xml")}catch(d){c=void 0}return c&&!c.getElementsByTagName("parsererror").length||r.error("Invalid XML: "+b),c};var wb=/\[\]$/,xb=/\r?\n/g,yb=/^(?:submit|button|image|reset|file)$/i,zb=/^(?:input|select|textarea|keygen)/i;function Ab(a,b,c,d){var e;if(Array.isArray(b))r.each(b,function(b,e){c||wb.test(a)?d(a,e):Ab(a+"["+("object"==typeof e&&null!=e?b:"")+"]",e,c,d)});else if(c||"object"!==r.type(b))d(a,b);else for(e in b)Ab(a+"["+e+"]",b[e],c,d)}r.param=function(a,b){var c,d=[],e=function(a,b){var c=r.isFunction(b)?b():b;d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(null==c?"":c)};if(Array.isArray(a)||a.jquery&&!r.isPlainObject(a))r.each(a,function(){e(this.name,this.value)});else for(c in a)Ab(c,a[c],b,e);return d.join("&")},r.fn.extend({serialize:function(){return r.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var a=r.prop(this,"elements");return a?r.makeArray(a):this}).filter(function(){var a=this.type;return this.name&&!r(this).is(":disabled")&&zb.test(this.nodeName)&&!yb.test(a)&&(this.checked||!ja.test(a))}).map(function(a,b){var c=r(this).val();return null==c?null:Array.isArray(c)?r.map(c,function(a){return{name:b.name,value:a.replace(xb,"\r\n")}}):{name:b.name,value:c.replace(xb,"\r\n")}}).get()}});var Bb=/%20/g,Cb=/#.*$/,Db=/([?&])_=[^&]*/,Eb=/^(.*?):[ \t]*([^\r\n]*)$/gm,Fb=/^(?:about|app|app-storage|.+-extension|file|res|widget):$/,Gb=/^(?:GET|HEAD)$/,Hb=/^\/\//,Ib={},Jb={},Kb="*/".concat("*"),Lb=d.createElement("a");Lb.href=tb.href;function Mb(a){return function(b,c){"string"!=typeof b&&(c=b,b="*");var d,e=0,f=b.toLowerCase().match(L)||[];if(r.isFunction(c))while(d=f[e++])"+"===d[0]?(d=d.slice(1)||"*",(a[d]=a[d]||[]).unshift(c)):(a[d]=a[d]||[]).push(c)}}function Nb(a,b,c,d){var e={},f=a===Jb;function g(h){var i;return e[h]=!0,r.each(a[h]||[],function(a,h){var j=h(b,c,d);return"string"!=typeof j||f||e[j]?f?!(i=j):void 0:(b.dataTypes.unshift(j),g(j),!1)}),i}return g(b.dataTypes[0])||!e["*"]&&g("*")}function Ob(a,b){var c,d,e=r.ajaxSettings.flatOptions||{};for(c in b)void 0!==b[c]&&((e[c]?a:d||(d={}))[c]=b[c]);return d&&r.extend(!0,a,d),a}function Pb(a,b,c){var d,e,f,g,h=a.contents,i=a.dataTypes;while("*"===i[0])i.shift(),void 0===d&&(d=a.mimeType||b.getResponseHeader("Content-Type"));if(d)for(e in h)if(h[e]&&h[e].test(d)){i.unshift(e);break}if(i[0]in c)f=i[0];else{for(e in c){if(!i[0]||a.converters[e+" "+i[0]]){f=e;break}g||(g=e)}f=f||g}if(f)return f!==i[0]&&i.unshift(f),c[f]}function Qb(a,b,c,d){var e,f,g,h,i,j={},k=a.dataTypes.slice();if(k[1])for(g in a.converters)j[g.toLowerCase()]=a.converters[g];f=k.shift();while(f)if(a.responseFields[f]&&(c[a.responseFields[f]]=b),!i&&d&&a.dataFilter&&(b=a.dataFilter(b,a.dataType)),i=f,f=k.shift())if("*"===f)f=i;else if("*"!==i&&i!==f){if(g=j[i+" "+f]||j["* "+f],!g)for(e in j)if(h=e.split(" "),h[1]===f&&(g=j[i+" "+h[0]]||j["* "+h[0]])){g===!0?g=j[e]:j[e]!==!0&&(f=h[0],k.unshift(h[1]));break}if(g!==!0)if(g&&a["throws"])b=g(b);else try{b=g(b)}catch(l){return{state:"parsererror",error:g?l:"No conversion from "+i+" to "+f}}}return{state:"success",data:b}}r.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:tb.href,type:"GET",isLocal:Fb.test(tb.protocol),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":Kb,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/\bxml\b/,html:/\bhtml/,json:/\bjson\b/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":JSON.parse,"text xml":r.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(a,b){return b?Ob(Ob(a,r.ajaxSettings),b):Ob(r.ajaxSettings,a)},ajaxPrefilter:Mb(Ib),ajaxTransport:Mb(Jb),ajax:function(b,c){"object"==typeof b&&(c=b,b=void 0),c=c||{};var e,f,g,h,i,j,k,l,m,n,o=r.ajaxSetup({},c),p=o.context||o,q=o.context&&(p.nodeType||p.jquery)?r(p):r.event,s=r.Deferred(),t=r.Callbacks("once memory"),u=o.statusCode||{},v={},w={},x="canceled",y={readyState:0,getResponseHeader:function(a){var b;if(k){if(!h){h={};while(b=Eb.exec(g))h[b[1].toLowerCase()]=b[2]}b=h[a.toLowerCase()]}return null==b?null:b},getAllResponseHeaders:function(){return k?g:null},setRequestHeader:function(a,b){return null==k&&(a=w[a.toLowerCase()]=w[a.toLowerCase()]||a,v[a]=b),this},overrideMimeType:function(a){return null==k&&(o.mimeType=a),this},statusCode:function(a){var b;if(a)if(k)y.always(a[y.status]);else for(b in a)u[b]=[u[b],a[b]];return this},abort:function(a){var b=a||x;return e&&e.abort(b),A(0,b),this}};if(s.promise(y),o.url=((b||o.url||tb.href)+"").replace(Hb,tb.protocol+"//"),o.type=c.method||c.type||o.method||o.type,o.dataTypes=(o.dataType||"*").toLowerCase().match(L)||[""],null==o.crossDomain){j=d.createElement("a");try{j.href=o.url,j.href=j.href,o.crossDomain=Lb.protocol+"//"+Lb.host!=j.protocol+"//"+j.host}catch(z){o.crossDomain=!0}}if(o.data&&o.processData&&"string"!=typeof o.data&&(o.data=r.param(o.data,o.traditional)),Nb(Ib,o,c,y),k)return y;l=r.event&&o.global,l&&0===r.active++&&r.event.trigger("ajaxStart"),o.type=o.type.toUpperCase(),o.hasContent=!Gb.test(o.type),f=o.url.replace(Cb,""),o.hasContent?o.data&&o.processData&&0===(o.contentType||"").indexOf("application/x-www-form-urlencoded")&&(o.data=o.data.replace(Bb,"+")):(n=o.url.slice(f.length),o.data&&(f+=(vb.test(f)?"&":"?")+o.data,delete o.data),o.cache===!1&&(f=f.replace(Db,"$1"),n=(vb.test(f)?"&":"?")+"_="+ub++ +n),o.url=f+n),o.ifModified&&(r.lastModified[f]&&y.setRequestHeader("If-Modified-Since",r.lastModified[f]),r.etag[f]&&y.setRequestHeader("If-None-Match",r.etag[f])),(o.data&&o.hasContent&&o.contentType!==!1||c.contentType)&&y.setRequestHeader("Content-Type",o.contentType),y.setRequestHeader("Accept",o.dataTypes[0]&&o.accepts[o.dataTypes[0]]?o.accepts[o.dataTypes[0]]+("*"!==o.dataTypes[0]?", "+Kb+"; q=0.01":""):o.accepts["*"]);for(m in o.headers)y.setRequestHeader(m,o.headers[m]);if(o.beforeSend&&(o.beforeSend.call(p,y,o)===!1||k))return y.abort();if(x="abort",t.add(o.complete),y.done(o.success),y.fail(o.error),e=Nb(Jb,o,c,y)){if(y.readyState=1,l&&q.trigger("ajaxSend",[y,o]),k)return y;o.async&&o.timeout>0&&(i=a.setTimeout(function(){y.abort("timeout")},o.timeout));try{k=!1,e.send(v,A)}catch(z){if(k)throw z;A(-1,z)}}else A(-1,"No Transport");function A(b,c,d,h){var j,m,n,v,w,x=c;k||(k=!0,i&&a.clearTimeout(i),e=void 0,g=h||"",y.readyState=b>0?4:0,j=b>=200&&b<300||304===b,d&&(v=Pb(o,y,d)),v=Qb(o,v,y,j),j?(o.ifModified&&(w=y.getResponseHeader("Last-Modified"),w&&(r.lastModified[f]=w),w=y.getResponseHeader("etag"),w&&(r.etag[f]=w)),204===b||"HEAD"===o.type?x="nocontent":304===b?x="notmodified":(x=v.state,m=v.data,n=v.error,j=!n)):(n=x,!b&&x||(x="error",b<0&&(b=0))),y.status=b,y.statusText=(c||x)+"",j?s.resolveWith(p,[m,x,y]):s.rejectWith(p,[y,x,n]),y.statusCode(u),u=void 0,l&&q.trigger(j?"ajaxSuccess":"ajaxError",[y,o,j?m:n]),t.fireWith(p,[y,x]),l&&(q.trigger("ajaxComplete",[y,o]),--r.active||r.event.trigger("ajaxStop")))}return y},getJSON:function(a,b,c){return r.get(a,b,c,"json")},getScript:function(a,b){return r.get(a,void 0,b,"script")}}),r.each(["get","post"],function(a,b){r[b]=function(a,c,d,e){return r.isFunction(c)&&(e=e||d,d=c,c=void 0),r.ajax(r.extend({url:a,type:b,dataType:e,data:c,success:d},r.isPlainObject(a)&&a))}}),r._evalUrl=function(a){return r.ajax({url:a,type:"GET",dataType:"script",cache:!0,async:!1,global:!1,"throws":!0})},r.fn.extend({wrapAll:function(a){var b;return this[0]&&(r.isFunction(a)&&(a=a.call(this[0])),b=r(a,this[0].ownerDocument).eq(0).clone(!0),this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstElementChild)a=a.firstElementChild;return a}).append(this)),this},wrapInner:function(a){return r.isFunction(a)?this.each(function(b){r(this).wrapInner(a.call(this,b))}):this.each(function(){var b=r(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){var b=r.isFunction(a);return this.each(function(c){r(this).wrapAll(b?a.call(this,c):a)})},unwrap:function(a){return this.parent(a).not("body").each(function(){r(this).replaceWith(this.childNodes)}),this}}),r.expr.pseudos.hidden=function(a){return!r.expr.pseudos.visible(a)},r.expr.pseudos.visible=function(a){return!!(a.offsetWidth||a.offsetHeight||a.getClientRects().length)},r.ajaxSettings.xhr=function(){try{return new a.XMLHttpRequest}catch(b){}};var Rb={0:200,1223:204},Sb=r.ajaxSettings.xhr();o.cors=!!Sb&&"withCredentials"in Sb,o.ajax=Sb=!!Sb,r.ajaxTransport(function(b){var c,d;if(o.cors||Sb&&!b.crossDomain)return{send:function(e,f){var g,h=b.xhr();if(h.open(b.type,b.url,b.async,b.username,b.password),b.xhrFields)for(g in b.xhrFields)h[g]=b.xhrFields[g];b.mimeType&&h.overrideMimeType&&h.overrideMimeType(b.mimeType),b.crossDomain||e["X-Requested-With"]||(e["X-Requested-With"]="XMLHttpRequest");for(g in e)h.setRequestHeader(g,e[g]);c=function(a){return function(){c&&(c=d=h.onload=h.onerror=h.onabort=h.onreadystatechange=null,"abort"===a?h.abort():"error"===a?"number"!=typeof h.status?f(0,"error"):f(h.status,h.statusText):f(Rb[h.status]||h.status,h.statusText,"text"!==(h.responseType||"text")||"string"!=typeof h.responseText?{binary:h.response}:{text:h.responseText},h.getAllResponseHeaders()))}},h.onload=c(),d=h.onerror=c("error"),void 0!==h.onabort?h.onabort=d:h.onreadystatechange=function(){4===h.readyState&&a.setTimeout(function(){c&&d()})},c=c("abort");try{h.send(b.hasContent&&b.data||null)}catch(i){if(c)throw i}},abort:function(){c&&c()}}}),r.ajaxPrefilter(function(a){a.crossDomain&&(a.contents.script=!1)}),r.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/\b(?:java|ecma)script\b/},converters:{"text script":function(a){return r.globalEval(a),a}}}),r.ajaxPrefilter("script",function(a){void 0===a.cache&&(a.cache=!1),a.crossDomain&&(a.type="GET")}),r.ajaxTransport("script",function(a){if(a.crossDomain){var b,c;return{send:function(e,f){b=r("<script>").prop({charset:a.scriptCharset,src:a.url}).on("load error",c=function(a){b.remove(),c=null,a&&f("error"===a.type?404:200,a.type)}),d.head.appendChild(b[0])},abort:function(){c&&c()}}}});var Tb=[],Ub=/(=)\?(?=&|$)|\?\?/;r.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var a=Tb.pop()||r.expando+"_"+ub++;return this[a]=!0,a}}),r.ajaxPrefilter("json jsonp",function(b,c,d){var e,f,g,h=b.jsonp!==!1&&(Ub.test(b.url)?"url":"string"==typeof b.data&&0===(b.contentType||"").indexOf("application/x-www-form-urlencoded")&&Ub.test(b.data)&&"data");if(h||"jsonp"===b.dataTypes[0])return e=b.jsonpCallback=r.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,h?b[h]=b[h].replace(Ub,"$1"+e):b.jsonp!==!1&&(b.url+=(vb.test(b.url)?"&":"?")+b.jsonp+"="+e),b.converters["script json"]=function(){return g||r.error(e+" was not called"),g[0]},b.dataTypes[0]="json",f=a[e],a[e]=function(){g=arguments},d.always(function(){void 0===f?r(a).removeProp(e):a[e]=f,b[e]&&(b.jsonpCallback=c.jsonpCallback,Tb.push(e)),g&&r.isFunction(f)&&f(g[0]),g=f=void 0}),"script"}),o.createHTMLDocument=function(){var a=d.implementation.createHTMLDocument("").body;return a.innerHTML="<form></form><form></form>",2===a.childNodes.length}(),r.parseHTML=function(a,b,c){if("string"!=typeof a)return[];"boolean"==typeof b&&(c=b,b=!1);var e,f,g;return b||(o.createHTMLDocument?(b=d.implementation.createHTMLDocument(""),e=b.createElement("base"),e.href=d.location.href,b.head.appendChild(e)):b=d),f=C.exec(a),g=!c&&[],f?[b.createElement(f[1])]:(f=qa([a],b,g),g&&g.length&&r(g).remove(),r.merge([],f.childNodes))},r.fn.load=function(a,b,c){var d,e,f,g=this,h=a.indexOf(" ");return h>-1&&(d=pb(a.slice(h)),a=a.slice(0,h)),r.isFunction(b)?(c=b,b=void 0):b&&"object"==typeof b&&(e="POST"),g.length>0&&r.ajax({url:a,type:e||"GET",dataType:"html",data:b}).done(function(a){f=arguments,g.html(d?r("<div>").append(r.parseHTML(a)).find(d):a)}).always(c&&function(a,b){g.each(function(){c.apply(this,f||[a.responseText,b,a])})}),this},r.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(a,b){r.fn[b]=function(a){return this.on(b,a)}}),r.expr.pseudos.animated=function(a){return r.grep(r.timers,function(b){return a===b.elem}).length},r.offset={setOffset:function(a,b,c){var d,e,f,g,h,i,j,k=r.css(a,"position"),l=r(a),m={};"static"===k&&(a.style.position="relative"),h=l.offset(),f=r.css(a,"top"),i=r.css(a,"left"),j=("absolute"===k||"fixed"===k)&&(f+i).indexOf("auto")>-1,j?(d=l.position(),g=d.top,e=d.left):(g=parseFloat(f)||0,e=parseFloat(i)||0),r.isFunction(b)&&(b=b.call(a,c,r.extend({},h))),null!=b.top&&(m.top=b.top-h.top+g),null!=b.left&&(m.left=b.left-h.left+e),"using"in b?b.using.call(a,m):l.css(m)}},r.fn.extend({offset:function(a){if(arguments.length)return void 0===a?this:this.each(function(b){r.offset.setOffset(this,a,b)});var b,c,d,e,f=this[0];if(f)return f.getClientRects().length?(d=f.getBoundingClientRect(),b=f.ownerDocument,c=b.documentElement,e=b.defaultView,{top:d.top+e.pageYOffset-c.clientTop,left:d.left+e.pageXOffset-c.clientLeft}):{top:0,left:0}},position:function(){if(this[0]){var a,b,c=this[0],d={top:0,left:0};return"fixed"===r.css(c,"position")?b=c.getBoundingClientRect():(a=this.offsetParent(),b=this.offset(),B(a[0],"html")||(d=a.offset()),d={top:d.top+r.css(a[0],"borderTopWidth",!0),left:d.left+r.css(a[0],"borderLeftWidth",!0)}),{top:b.top-d.top-r.css(c,"marginTop",!0),left:b.left-d.left-r.css(c,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var a=this.offsetParent;while(a&&"static"===r.css(a,"position"))a=a.offsetParent;return a||ra})}}),r.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(a,b){var c="pageYOffset"===b;r.fn[a]=function(d){return T(this,function(a,d,e){var f;return r.isWindow(a)?f=a:9===a.nodeType&&(f=a.defaultView),void 0===e?f?f[b]:a[d]:void(f?f.scrollTo(c?f.pageXOffset:e,c?e:f.pageYOffset):a[d]=e)},a,d,arguments.length)}}),r.each(["top","left"],function(a,b){r.cssHooks[b]=Pa(o.pixelPosition,function(a,c){if(c)return c=Oa(a,b),Ma.test(c)?r(a).position()[b]+"px":c})}),r.each({Height:"height",Width:"width"},function(a,b){r.each({padding:"inner"+a,content:b,"":"outer"+a},function(c,d){r.fn[d]=function(e,f){var g=arguments.length&&(c||"boolean"!=typeof e),h=c||(e===!0||f===!0?"margin":"border");return T(this,function(b,c,e){var f;return r.isWindow(b)?0===d.indexOf("outer")?b["inner"+a]:b.document.documentElement["client"+a]:9===b.nodeType?(f=b.documentElement,Math.max(b.body["scroll"+a],f["scroll"+a],b.body["offset"+a],f["offset"+a],f["client"+a])):void 0===e?r.css(b,c,h):r.style(b,c,e,h)},b,g?e:void 0,g)}})}),r.fn.extend({bind:function(a,b,c){return this.on(a,null,b,c)},unbind:function(a,b){return this.off(a,null,b)},delegate:function(a,b,c,d){return this.on(b,a,c,d)},undelegate:function(a,b,c){return 1===arguments.length?this.off(a,"**"):this.off(b,a||"**",c)}}),r.holdReady=function(a){a?r.readyWait++:r.ready(!0)},r.isArray=Array.isArray,r.parseJSON=JSON.parse,r.nodeName=B,"function"==typeof define&&define.amd&&define("jquery",[],function(){return r});var Vb=a.jQuery,Wb=a.$;return r.noConflict=function(b){return a.$===r&&(a.$=Wb),b&&a.jQuery===r&&(a.jQuery=Vb),r},b||(a.jQuery=a.$=r),r});
diff --git a/inferno/jqueryui/jquery-ui.min.css b/inferno/jqueryui/jquery-ui.min.css
deleted file mode 100644
index 22ca1d3..0000000
--- a/inferno/jqueryui/jquery-ui.min.css
+++ /dev/null
@@ -1,7 +0,0 @@
-/*! jQuery UI - v1.12.1 - 2017-08-18
-* http://jqueryui.com
-* Includes: core.css, resizable.css, theme.css
-* To view and modify this theme, visit http://jqueryui.com/themeroller/?scope=&folderName=base&cornerRadiusShadow=8px&offsetLeftShadow=0px&offsetTopShadow=0px&thicknessShadow=5px&opacityShadow=30&bgImgOpacityShadow=0&bgTextureShadow=flat&bgColorShadow=666666&opacityOverlay=30&bgImgOpacityOverlay=0&bgTextureOverlay=flat&bgColorOverlay=aaaaaa&iconColorError=cc0000&fcError=5f3f3f&borderColorError=f1a899&bgTextureError=flat&bgColorError=fddfdf&iconColorHighlight=777620&fcHighlight=777620&borderColorHighlight=dad55e&bgTextureHighlight=flat&bgColorHighlight=fffa90&iconColorActive=ffffff&fcActive=ffffff&borderColorActive=003eff&bgTextureActive=flat&bgColorActive=007fff&iconColorHover=555555&fcHover=2b2b2b&borderColorHover=cccccc&bgTextureHover=flat&bgColorHover=ededed&iconColorDefault=777777&fcDefault=454545&borderColorDefault=c5c5c5&bgTextureDefault=flat&bgColorDefault=f6f6f6&iconColorContent=444444&fcContent=333333&borderColorContent=dddddd&bgTextureContent=flat&bgColorContent=ffffff&iconColorHeader=444444&fcHeader=333333&borderColorHeader=dddddd&bgTextureHeader=flat&bgColorHeader=e9e9e9&cornerRadius=3px&fwDefault=normal&fsDefault=1em&ffDefault=Arial%2CHelvetica%2Csans-serif
-* Copyright jQuery Foundation and other contributors; Licensed MIT */
-
-.ui-helper-hidden{display:none}.ui-helper-hidden-accessible{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px}.ui-helper-reset{margin:0;padding:0;border:0;outline:0;line-height:1.3;text-decoration:none;font-size:100%;list-style:none}.ui-helper-clearfix:before,.ui-helper-clearfix:after{content:"";display:table;border-collapse:collapse}.ui-helper-clearfix:after{clear:both}.ui-helper-zfix{width:100%;height:100%;top:0;left:0;position:absolute;opacity:0;filter:Alpha(Opacity=0)}.ui-front{z-index:100}.ui-state-disabled{cursor:default!important;pointer-events:none}.ui-icon{display:inline-block;vertical-align:middle;margin-top:-.25em;position:relative;text-indent:-99999px;overflow:hidden;background-repeat:no-repeat}.ui-widget-icon-block{left:50%;margin-left:-8px;display:block}.ui-widget-overlay{position:fixed;top:0;left:0;width:100%;height:100%}.ui-resizable{position:relative}.ui-resizable-handle{position:absolute;font-size:0.1px;display:block;-ms-touch-action:none;touch-action:none}.ui-resizable-disabled .ui-resizable-handle,.ui-resizable-autohide .ui-resizable-handle{display:none}.ui-resizable-n{cursor:n-resize;height:7px;width:100%;top:-5px;left:0}.ui-resizable-s{cursor:s-resize;height:7px;width:100%;bottom:-5px;left:0}.ui-resizable-e{cursor:e-resize;width:7px;right:-5px;top:0;height:100%}.ui-resizable-w{cursor:w-resize;width:7px;left:-5px;top:0;height:100%}.ui-resizable-se{cursor:se-resize;width:12px;height:12px;right:1px;bottom:1px}.ui-resizable-sw{cursor:sw-resize;width:9px;height:9px;left:-5px;bottom:-5px}.ui-resizable-nw{cursor:nw-resize;width:9px;height:9px;left:-5px;top:-5px}.ui-resizable-ne{cursor:ne-resize;width:9px;height:9px;right:-5px;top:-5px}.ui-widget{font-family:Arial,Helvetica,sans-serif;font-size:1em}.ui-widget .ui-widget{font-size:1em}.ui-widget input,.ui-widget select,.ui-widget textarea,.ui-widget button{font-family:Arial,Helvetica,sans-serif;font-size:1em}.ui-widget.ui-widget-content{border:1px solid #c5c5c5}.ui-widget-content{border:1px solid #ddd;background:#fff;color:#333}.ui-widget-content a{color:#333}.ui-widget-header{border:1px solid #ddd;background:#e9e9e9;color:#333;font-weight:bold}.ui-widget-header a{color:#333}.ui-state-default,.ui-widget-content .ui-state-default,.ui-widget-header .ui-state-default,.ui-button,html .ui-button.ui-state-disabled:hover,html .ui-button.ui-state-disabled:active{border:1px solid #c5c5c5;background:#f6f6f6;font-weight:normal;color:#454545}.ui-state-default a,.ui-state-default a:link,.ui-state-default a:visited,a.ui-button,a:link.ui-button,a:visited.ui-button,.ui-button{color:#454545;text-decoration:none}.ui-state-hover,.ui-widget-content .ui-state-hover,.ui-widget-header .ui-state-hover,.ui-state-focus,.ui-widget-content .ui-state-focus,.ui-widget-header .ui-state-focus,.ui-button:hover,.ui-button:focus{border:1px solid #ccc;background:#ededed;font-weight:normal;color:#2b2b2b}.ui-state-hover a,.ui-state-hover a:hover,.ui-state-hover a:link,.ui-state-hover a:visited,.ui-state-focus a,.ui-state-focus a:hover,.ui-state-focus a:link,.ui-state-focus a:visited,a.ui-button:hover,a.ui-button:focus{color:#2b2b2b;text-decoration:none}.ui-visual-focus{box-shadow:0 0 3px 1px rgb(94,158,214)}.ui-state-active,.ui-widget-content .ui-state-active,.ui-widget-header .ui-state-active,a.ui-button:active,.ui-button:active,.ui-button.ui-state-active:hover{border:1px solid #003eff;background:#007fff;font-weight:normal;color:#fff}.ui-icon-background,.ui-state-active .ui-icon-background{border:#003eff;background-color:#fff}.ui-state-active a,.ui-state-active a:link,.ui-state-active a:visited{color:#fff;text-decoration:none}.ui-state-highlight,.ui-widget-content .ui-state-highlight,.ui-widget-header .ui-state-highlight{border:1px solid #dad55e;background:#fffa90;color:#777620}.ui-state-checked{border:1px solid #dad55e;background:#fffa90}.ui-state-highlight a,.ui-widget-content .ui-state-highlight a,.ui-widget-header .ui-state-highlight a{color:#777620}.ui-state-error,.ui-widget-content .ui-state-error,.ui-widget-header .ui-state-error{border:1px solid #f1a899;background:#fddfdf;color:#5f3f3f}.ui-state-error a,.ui-widget-content .ui-state-error a,.ui-widget-header .ui-state-error a{color:#5f3f3f}.ui-state-error-text,.ui-widget-content .ui-state-error-text,.ui-widget-header .ui-state-error-text{color:#5f3f3f}.ui-priority-primary,.ui-widget-content .ui-priority-primary,.ui-widget-header .ui-priority-primary{font-weight:bold}.ui-priority-secondary,.ui-widget-content .ui-priority-secondary,.ui-widget-header .ui-priority-secondary{opacity:.7;filter:Alpha(Opacity=70);font-weight:normal}.ui-state-disabled,.ui-widget-content .ui-state-disabled,.ui-widget-header .ui-state-disabled{opacity:.35;filter:Alpha(Opacity=35);background-image:none}.ui-state-disabled .ui-icon{filter:Alpha(Opacity=35)}.ui-icon{width:16px;height:16px}.ui-icon,.ui-widget-content .ui-icon{background-image:url("images/ui-icons_444444_256x240.png")}.ui-widget-header .ui-icon{background-image:url("images/ui-icons_444444_256x240.png")}.ui-state-hover .ui-icon,.ui-state-focus .ui-icon,.ui-button:hover .ui-icon,.ui-button:focus .ui-icon{background-image:url("images/ui-icons_555555_256x240.png")}.ui-state-active .ui-icon,.ui-button:active .ui-icon{background-image:url("images/ui-icons_ffffff_256x240.png")}.ui-state-highlight .ui-icon,.ui-button .ui-state-highlight.ui-icon{background-image:url("images/ui-icons_777620_256x240.png")}.ui-state-error .ui-icon,.ui-state-error-text .ui-icon{background-image:url("images/ui-icons_cc0000_256x240.png")}.ui-button .ui-icon{background-image:url("images/ui-icons_777777_256x240.png")}.ui-icon-blank{background-position:16px 16px}.ui-icon-caret-1-n{background-position:0 0}.ui-icon-caret-1-ne{background-position:-16px 0}.ui-icon-caret-1-e{background-position:-32px 0}.ui-icon-caret-1-se{background-position:-48px 0}.ui-icon-caret-1-s{background-position:-65px 0}.ui-icon-caret-1-sw{background-position:-80px 0}.ui-icon-caret-1-w{background-position:-96px 0}.ui-icon-caret-1-nw{background-position:-112px 0}.ui-icon-caret-2-n-s{background-position:-128px 0}.ui-icon-caret-2-e-w{background-position:-144px 0}.ui-icon-triangle-1-n{background-position:0 -16px}.ui-icon-triangle-1-ne{background-position:-16px -16px}.ui-icon-triangle-1-e{background-position:-32px -16px}.ui-icon-triangle-1-se{background-position:-48px -16px}.ui-icon-triangle-1-s{background-position:-65px -16px}.ui-icon-triangle-1-sw{background-position:-80px -16px}.ui-icon-triangle-1-w{background-position:-96px -16px}.ui-icon-triangle-1-nw{background-position:-112px -16px}.ui-icon-triangle-2-n-s{background-position:-128px -16px}.ui-icon-triangle-2-e-w{background-position:-144px -16px}.ui-icon-arrow-1-n{background-position:0 -32px}.ui-icon-arrow-1-ne{background-position:-16px -32px}.ui-icon-arrow-1-e{background-position:-32px -32px}.ui-icon-arrow-1-se{background-position:-48px -32px}.ui-icon-arrow-1-s{background-position:-65px -32px}.ui-icon-arrow-1-sw{background-position:-80px -32px}.ui-icon-arrow-1-w{background-position:-96px -32px}.ui-icon-arrow-1-nw{background-position:-112px -32px}.ui-icon-arrow-2-n-s{background-position:-128px -32px}.ui-icon-arrow-2-ne-sw{background-position:-144px -32px}.ui-icon-arrow-2-e-w{background-position:-160px -32px}.ui-icon-arrow-2-se-nw{background-position:-176px -32px}.ui-icon-arrowstop-1-n{background-position:-192px -32px}.ui-icon-arrowstop-1-e{background-position:-208px -32px}.ui-icon-arrowstop-1-s{background-position:-224px -32px}.ui-icon-arrowstop-1-w{background-position:-240px -32px}.ui-icon-arrowthick-1-n{background-position:1px -48px}.ui-icon-arrowthick-1-ne{background-position:-16px -48px}.ui-icon-arrowthick-1-e{background-position:-32px -48px}.ui-icon-arrowthick-1-se{background-position:-48px -48px}.ui-icon-arrowthick-1-s{background-position:-64px -48px}.ui-icon-arrowthick-1-sw{background-position:-80px -48px}.ui-icon-arrowthick-1-w{background-position:-96px -48px}.ui-icon-arrowthick-1-nw{background-position:-112px -48px}.ui-icon-arrowthick-2-n-s{background-position:-128px -48px}.ui-icon-arrowthick-2-ne-sw{background-position:-144px -48px}.ui-icon-arrowthick-2-e-w{background-position:-160px -48px}.ui-icon-arrowthick-2-se-nw{background-position:-176px -48px}.ui-icon-arrowthickstop-1-n{background-position:-192px -48px}.ui-icon-arrowthickstop-1-e{background-position:-208px -48px}.ui-icon-arrowthickstop-1-s{background-position:-224px -48px}.ui-icon-arrowthickstop-1-w{background-position:-240px -48px}.ui-icon-arrowreturnthick-1-w{background-position:0 -64px}.ui-icon-arrowreturnthick-1-n{background-position:-16px -64px}.ui-icon-arrowreturnthick-1-e{background-position:-32px -64px}.ui-icon-arrowreturnthick-1-s{background-position:-48px -64px}.ui-icon-arrowreturn-1-w{background-position:-64px -64px}.ui-icon-arrowreturn-1-n{background-position:-80px -64px}.ui-icon-arrowreturn-1-e{background-position:-96px -64px}.ui-icon-arrowreturn-1-s{background-position:-112px -64px}.ui-icon-arrowrefresh-1-w{background-position:-128px -64px}.ui-icon-arrowrefresh-1-n{background-position:-144px -64px}.ui-icon-arrowrefresh-1-e{background-position:-160px -64px}.ui-icon-arrowrefresh-1-s{background-position:-176px -64px}.ui-icon-arrow-4{background-position:0 -80px}.ui-icon-arrow-4-diag{background-position:-16px -80px}.ui-icon-extlink{background-position:-32px -80px}.ui-icon-newwin{background-position:-48px -80px}.ui-icon-refresh{background-position:-64px -80px}.ui-icon-shuffle{background-position:-80px -80px}.ui-icon-transfer-e-w{background-position:-96px -80px}.ui-icon-transferthick-e-w{background-position:-112px -80px}.ui-icon-folder-collapsed{background-position:0 -96px}.ui-icon-folder-open{background-position:-16px -96px}.ui-icon-document{background-position:-32px -96px}.ui-icon-document-b{background-position:-48px -96px}.ui-icon-note{background-position:-64px -96px}.ui-icon-mail-closed{background-position:-80px -96px}.ui-icon-mail-open{background-position:-96px -96px}.ui-icon-suitcase{background-position:-112px -96px}.ui-icon-comment{background-position:-128px -96px}.ui-icon-person{background-position:-144px -96px}.ui-icon-print{background-position:-160px -96px}.ui-icon-trash{background-position:-176px -96px}.ui-icon-locked{background-position:-192px -96px}.ui-icon-unlocked{background-position:-208px -96px}.ui-icon-bookmark{background-position:-224px -96px}.ui-icon-tag{background-position:-240px -96px}.ui-icon-home{background-position:0 -112px}.ui-icon-flag{background-position:-16px -112px}.ui-icon-calendar{background-position:-32px -112px}.ui-icon-cart{background-position:-48px -112px}.ui-icon-pencil{background-position:-64px -112px}.ui-icon-clock{background-position:-80px -112px}.ui-icon-disk{background-position:-96px -112px}.ui-icon-calculator{background-position:-112px -112px}.ui-icon-zoomin{background-position:-128px -112px}.ui-icon-zoomout{background-position:-144px -112px}.ui-icon-search{background-position:-160px -112px}.ui-icon-wrench{background-position:-176px -112px}.ui-icon-gear{background-position:-192px -112px}.ui-icon-heart{background-position:-208px -112px}.ui-icon-star{background-position:-224px -112px}.ui-icon-link{background-position:-240px -112px}.ui-icon-cancel{background-position:0 -128px}.ui-icon-plus{background-position:-16px -128px}.ui-icon-plusthick{background-position:-32px -128px}.ui-icon-minus{background-position:-48px -128px}.ui-icon-minusthick{background-position:-64px -128px}.ui-icon-close{background-position:-80px -128px}.ui-icon-closethick{background-position:-96px -128px}.ui-icon-key{background-position:-112px -128px}.ui-icon-lightbulb{background-position:-128px -128px}.ui-icon-scissors{background-position:-144px -128px}.ui-icon-clipboard{background-position:-160px -128px}.ui-icon-copy{background-position:-176px -128px}.ui-icon-contact{background-position:-192px -128px}.ui-icon-image{background-position:-208px -128px}.ui-icon-video{background-position:-224px -128px}.ui-icon-script{background-position:-240px -128px}.ui-icon-alert{background-position:0 -144px}.ui-icon-info{background-position:-16px -144px}.ui-icon-notice{background-position:-32px -144px}.ui-icon-help{background-position:-48px -144px}.ui-icon-check{background-position:-64px -144px}.ui-icon-bullet{background-position:-80px -144px}.ui-icon-radio-on{background-position:-96px -144px}.ui-icon-radio-off{background-position:-112px -144px}.ui-icon-pin-w{background-position:-128px -144px}.ui-icon-pin-s{background-position:-144px -144px}.ui-icon-play{background-position:0 -160px}.ui-icon-pause{background-position:-16px -160px}.ui-icon-seek-next{background-position:-32px -160px}.ui-icon-seek-prev{background-position:-48px -160px}.ui-icon-seek-end{background-position:-64px -160px}.ui-icon-seek-start{background-position:-80px -160px}.ui-icon-seek-first{background-position:-80px -160px}.ui-icon-stop{background-position:-96px -160px}.ui-icon-eject{background-position:-112px -160px}.ui-icon-volume-off{background-position:-128px -160px}.ui-icon-volume-on{background-position:-144px -160px}.ui-icon-power{background-position:0 -176px}.ui-icon-signal-diag{background-position:-16px -176px}.ui-icon-signal{background-position:-32px -176px}.ui-icon-battery-0{background-position:-48px -176px}.ui-icon-battery-1{background-position:-64px -176px}.ui-icon-battery-2{background-position:-80px -176px}.ui-icon-battery-3{background-position:-96px -176px}.ui-icon-circle-plus{background-position:0 -192px}.ui-icon-circle-minus{background-position:-16px -192px}.ui-icon-circle-close{background-position:-32px -192px}.ui-icon-circle-triangle-e{background-position:-48px -192px}.ui-icon-circle-triangle-s{background-position:-64px -192px}.ui-icon-circle-triangle-w{background-position:-80px -192px}.ui-icon-circle-triangle-n{background-position:-96px -192px}.ui-icon-circle-arrow-e{background-position:-112px -192px}.ui-icon-circle-arrow-s{background-position:-128px -192px}.ui-icon-circle-arrow-w{background-position:-144px -192px}.ui-icon-circle-arrow-n{background-position:-160px -192px}.ui-icon-circle-zoomin{background-position:-176px -192px}.ui-icon-circle-zoomout{background-position:-192px -192px}.ui-icon-circle-check{background-position:-208px -192px}.ui-icon-circlesmall-plus{background-position:0 -208px}.ui-icon-circlesmall-minus{background-position:-16px -208px}.ui-icon-circlesmall-close{background-position:-32px -208px}.ui-icon-squaresmall-plus{background-position:-48px -208px}.ui-icon-squaresmall-minus{background-position:-64px -208px}.ui-icon-squaresmall-close{background-position:-80px -208px}.ui-icon-grip-dotted-vertical{background-position:0 -224px}.ui-icon-grip-dotted-horizontal{background-position:-16px -224px}.ui-icon-grip-solid-vertical{background-position:-32px -224px}.ui-icon-grip-solid-horizontal{background-position:-48px -224px}.ui-icon-gripsmall-diagonal-se{background-position:-64px -224px}.ui-icon-grip-diagonal-se{background-position:-80px -224px}.ui-corner-all,.ui-corner-top,.ui-corner-left,.ui-corner-tl{border-top-left-radius:3px}.ui-corner-all,.ui-corner-top,.ui-corner-right,.ui-corner-tr{border-top-right-radius:3px}.ui-corner-all,.ui-corner-bottom,.ui-corner-left,.ui-corner-bl{border-bottom-left-radius:3px}.ui-corner-all,.ui-corner-bottom,.ui-corner-right,.ui-corner-br{border-bottom-right-radius:3px}.ui-widget-overlay{background:#aaa;opacity:.3;filter:Alpha(Opacity=30)}.ui-widget-shadow{-webkit-box-shadow:0 0 5px #666;box-shadow:0 0 5px #666}
\ No newline at end of file
diff --git a/inferno/jqueryui/jquery-ui.min.js b/inferno/jqueryui/jquery-ui.min.js
deleted file mode 100644
index a26b9c0..0000000
--- a/inferno/jqueryui/jquery-ui.min.js
+++ /dev/null
@@ -1,6 +0,0 @@
-/*! jQuery UI - v1.12.1 - 2017-08-18
-* http://jqueryui.com
-* Includes: widget.js, disable-selection.js, widgets/resizable.js, widgets/mouse.js
-* Copyright jQuery Foundation and other contributors; Licensed MIT */
-
-(function(t){"function"==typeof define&&define.amd?define(["jquery"],t):t(jQuery)})(function(t){t.ui=t.ui||{},t.ui.version="1.12.1";var e=0,i=Array.prototype.slice;t.cleanData=function(e){return function(i){var s,n,o;for(o=0;null!=(n=i[o]);o++)try{s=t._data(n,"events"),s&&s.remove&&t(n).triggerHandler("remove")}catch(a){}e(i)}}(t.cleanData),t.widget=function(e,i,s){var n,o,a,r={},l=e.split(".")[0];e=e.split(".")[1];var h=l+"-"+e;return s||(s=i,i=t.Widget),t.isArray(s)&&(s=t.extend.apply(null,[{}].concat(s))),t.expr[":"][h.toLowerCase()]=function(e){return!!t.data(e,h)},t[l]=t[l]||{},n=t[l][e],o=t[l][e]=function(t,e){return this._createWidget?(arguments.length&&this._createWidget(t,e),void 0):new o(t,e)},t.extend(o,n,{version:s.version,_proto:t.extend({},s),_childConstructors:[]}),a=new i,a.options=t.widget.extend({},a.options),t.each(s,function(e,s){return t.isFunction(s)?(r[e]=function(){function t(){return i.prototype[e].apply(this,arguments)}function n(t){return i.prototype[e].apply(this,t)}return function(){var e,i=this._super,o=this._superApply;return this._super=t,this._superApply=n,e=s.apply(this,arguments),this._super=i,this._superApply=o,e}}(),void 0):(r[e]=s,void 0)}),o.prototype=t.widget.extend(a,{widgetEventPrefix:n?a.widgetEventPrefix||e:e},r,{constructor:o,namespace:l,widgetName:e,widgetFullName:h}),n?(t.each(n._childConstructors,function(e,i){var s=i.prototype;t.widget(s.namespace+"."+s.widgetName,o,i._proto)}),delete n._childConstructors):i._childConstructors.push(o),t.widget.bridge(e,o),o},t.widget.extend=function(e){for(var s,n,o=i.call(arguments,1),a=0,r=o.length;r>a;a++)for(s in o[a])n=o[a][s],o[a].hasOwnProperty(s)&&void 0!==n&&(e[s]=t.isPlainObject(n)?t.isPlainObject(e[s])?t.widget.extend({},e[s],n):t.widget.extend({},n):n);return e},t.widget.bridge=function(e,s){var n=s.prototype.widgetFullName||e;t.fn[e]=function(o){var a="string"==typeof o,r=i.call(arguments,1),l=this;return a?this.length||"instance"!==o?this.each(function(){var i,s=t.data(this,n);return"instance"===o?(l=s,!1):s?t.isFunction(s[o])&&"_"!==o.charAt(0)?(i=s[o].apply(s,r),i!==s&&void 0!==i?(l=i&&i.jquery?l.pushStack(i.get()):i,!1):void 0):t.error("no such method '"+o+"' for "+e+" widget instance"):t.error("cannot call methods on "+e+" prior to initialization; "+"attempted to call method '"+o+"'")}):l=void 0:(r.length&&(o=t.widget.extend.apply(null,[o].concat(r))),this.each(function(){var e=t.data(this,n);e?(e.option(o||{}),e._init&&e._init()):t.data(this,n,new s(o,this))})),l}},t.Widget=function(){},t.Widget._childConstructors=[],t.Widget.prototype={widgetName:"widget",widgetEventPrefix:"",defaultElement:"<div>",options:{classes:{},disabled:!1,create:null},_createWidget:function(i,s){s=t(s||this.defaultElement||this)[0],this.element=t(s),this.uuid=e++,this.eventNamespace="."+this.widgetName+this.uuid,this.bindings=t(),this.hoverable=t(),this.focusable=t(),this.classesElementLookup={},s!==this&&(t.data(s,this.widgetFullName,this),this._on(!0,this.element,{remove:function(t){t.target===s&&this.destroy()}}),this.document=t(s.style?s.ownerDocument:s.document||s),this.window=t(this.document[0].defaultView||this.document[0].parentWindow)),this.options=t.widget.extend({},this.options,this._getCreateOptions(),i),this._create(),this.options.disabled&&this._setOptionDisabled(this.options.disabled),this._trigger("create",null,this._getCreateEventData()),this._init()},_getCreateOptions:function(){return{}},_getCreateEventData:t.noop,_create:t.noop,_init:t.noop,destroy:function(){var e=this;this._destroy(),t.each(this.classesElementLookup,function(t,i){e._removeClass(i,t)}),this.element.off(this.eventNamespace).removeData(this.widgetFullName),this.widget().off(this.eventNamespace).removeAttr("aria-disabled"),this.bindings.off(this.eventNamespace)},_destroy:t.noop,widget:function(){return this.element},option:function(e,i){var s,n,o,a=e;if(0===arguments.length)return t.widget.extend({},this.options);if("string"==typeof e)if(a={},s=e.split("."),e=s.shift(),s.length){for(n=a[e]=t.widget.extend({},this.options[e]),o=0;s.length-1>o;o++)n[s[o]]=n[s[o]]||{},n=n[s[o]];if(e=s.pop(),1===arguments.length)return void 0===n[e]?null:n[e];n[e]=i}else{if(1===arguments.length)return void 0===this.options[e]?null:this.options[e];a[e]=i}return this._setOptions(a),this},_setOptions:function(t){var e;for(e in t)this._setOption(e,t[e]);return this},_setOption:function(t,e){return"classes"===t&&this._setOptionClasses(e),this.options[t]=e,"disabled"===t&&this._setOptionDisabled(e),this},_setOptionClasses:function(e){var i,s,n;for(i in e)n=this.classesElementLookup[i],e[i]!==this.options.classes[i]&&n&&n.length&&(s=t(n.get()),this._removeClass(n,i),s.addClass(this._classes({element:s,keys:i,classes:e,add:!0})))},_setOptionDisabled:function(t){this._toggleClass(this.widget(),this.widgetFullName+"-disabled",null,!!t),t&&(this._removeClass(this.hoverable,null,"ui-state-hover"),this._removeClass(this.focusable,null,"ui-state-focus"))},enable:function(){return this._setOptions({disabled:!1})},disable:function(){return this._setOptions({disabled:!0})},_classes:function(e){function i(i,o){var a,r;for(r=0;i.length>r;r++)a=n.classesElementLookup[i[r]]||t(),a=e.add?t(t.unique(a.get().concat(e.element.get()))):t(a.not(e.element).get()),n.classesElementLookup[i[r]]=a,s.push(i[r]),o&&e.classes[i[r]]&&s.push(e.classes[i[r]])}var s=[],n=this;return e=t.extend({element:this.element,classes:this.options.classes||{}},e),this._on(e.element,{remove:"_untrackClassesElement"}),e.keys&&i(e.keys.match(/\S+/g)||[],!0),e.extra&&i(e.extra.match(/\S+/g)||[]),s.join(" ")},_untrackClassesElement:function(e){var i=this;t.each(i.classesElementLookup,function(s,n){-1!==t.inArray(e.target,n)&&(i.classesElementLookup[s]=t(n.not(e.target).get()))})},_removeClass:function(t,e,i){return this._toggleClass(t,e,i,!1)},_addClass:function(t,e,i){return this._toggleClass(t,e,i,!0)},_toggleClass:function(t,e,i,s){s="boolean"==typeof s?s:i;var n="string"==typeof t||null===t,o={extra:n?e:i,keys:n?t:e,element:n?this.element:t,add:s};return o.element.toggleClass(this._classes(o),s),this},_on:function(e,i,s){var n,o=this;"boolean"!=typeof e&&(s=i,i=e,e=!1),s?(i=n=t(i),this.bindings=this.bindings.add(i)):(s=i,i=this.element,n=this.widget()),t.each(s,function(s,a){function r(){return e||o.options.disabled!==!0&&!t(this).hasClass("ui-state-disabled")?("string"==typeof a?o[a]:a).apply(o,arguments):void 0}"string"!=typeof a&&(r.guid=a.guid=a.guid||r.guid||t.guid++);var l=s.match(/^([\w:-]*)\s*(.*)$/),h=l[1]+o.eventNamespace,c=l[2];c?n.on(h,c,r):i.on(h,r)})},_off:function(e,i){i=(i||"").split(" ").join(this.eventNamespace+" ")+this.eventNamespace,e.off(i).off(i),this.bindings=t(this.bindings.not(e).get()),this.focusable=t(this.focusable.not(e).get()),this.hoverable=t(this.hoverable.not(e).get())},_delay:function(t,e){function i(){return("string"==typeof t?s[t]:t).apply(s,arguments)}var s=this;return setTimeout(i,e||0)},_hoverable:function(e){this.hoverable=this.hoverable.add(e),this._on(e,{mouseenter:function(e){this._addClass(t(e.currentTarget),null,"ui-state-hover")},mouseleave:function(e){this._removeClass(t(e.currentTarget),null,"ui-state-hover")}})},_focusable:function(e){this.focusable=this.focusable.add(e),this._on(e,{focusin:function(e){this._addClass(t(e.currentTarget),null,"ui-state-focus")},focusout:function(e){this._removeClass(t(e.currentTarget),null,"ui-state-focus")}})},_trigger:function(e,i,s){var n,o,a=this.options[e];if(s=s||{},i=t.Event(i),i.type=(e===this.widgetEventPrefix?e:this.widgetEventPrefix+e).toLowerCase(),i.target=this.element[0],o=i.originalEvent)for(n in o)n in i||(i[n]=o[n]);return this.element.trigger(i,s),!(t.isFunction(a)&&a.apply(this.element[0],[i].concat(s))===!1||i.isDefaultPrevented())}},t.each({show:"fadeIn",hide:"fadeOut"},function(e,i){t.Widget.prototype["_"+e]=function(s,n,o){"string"==typeof n&&(n={effect:n});var a,r=n?n===!0||"number"==typeof n?i:n.effect||i:e;n=n||{},"number"==typeof n&&(n={duration:n}),a=!t.isEmptyObject(n),n.complete=o,n.delay&&s.delay(n.delay),a&&t.effects&&t.effects.effect[r]?s[e](n):r!==e&&s[r]?s[r](n.duration,n.easing,o):s.queue(function(i){t(this)[e](),o&&o.call(s[0]),i()})}}),t.widget,t.fn.extend({disableSelection:function(){var t="onselectstart"in document.createElement("div")?"selectstart":"mousedown";return function(){return this.on(t+".ui-disableSelection",function(t){t.preventDefault()})}}(),enableSelection:function(){return this.off(".ui-disableSelection")}}),t.ui.ie=!!/msie [\w.]+/.exec(navigator.userAgent.toLowerCase());var s=!1;t(document).on("mouseup",function(){s=!1}),t.widget("ui.mouse",{version:"1.12.1",options:{cancel:"input, textarea, button, select, option",distance:1,delay:0},_mouseInit:function(){var e=this;this.element.on("mousedown."+this.widgetName,function(t){return e._mouseDown(t)}).on("click."+this.widgetName,function(i){return!0===t.data(i.target,e.widgetName+".preventClickEvent")?(t.removeData(i.target,e.widgetName+".preventClickEvent"),i.stopImmediatePropagation(),!1):void 0}),this.started=!1},_mouseDestroy:function(){this.element.off("."+this.widgetName),this._mouseMoveDelegate&&this.document.off("mousemove."+this.widgetName,this._mouseMoveDelegate).off("mouseup."+this.widgetName,this._mouseUpDelegate)},_mouseDown:function(e){if(!s){this._mouseMoved=!1,this._mouseStarted&&this._mouseUp(e),this._mouseDownEvent=e;var i=this,n=1===e.which,o="string"==typeof this.options.cancel&&e.target.nodeName?t(e.target).closest(this.options.cancel).length:!1;return n&&!o&&this._mouseCapture(e)?(this.mouseDelayMet=!this.options.delay,this.mouseDelayMet||(this._mouseDelayTimer=setTimeout(function(){i.mouseDelayMet=!0},this.options.delay)),this._mouseDistanceMet(e)&&this._mouseDelayMet(e)&&(this._mouseStarted=this._mouseStart(e)!==!1,!this._mouseStarted)?(e.preventDefault(),!0):(!0===t.data(e.target,this.widgetName+".preventClickEvent")&&t.removeData(e.target,this.widgetName+".preventClickEvent"),this._mouseMoveDelegate=function(t){return i._mouseMove(t)},this._mouseUpDelegate=function(t){return i._mouseUp(t)},this.document.on("mousemove."+this.widgetName,this._mouseMoveDelegate).on("mouseup."+this.widgetName,this._mouseUpDelegate),e.preventDefault(),s=!0,!0)):!0}},_mouseMove:function(e){if(this._mouseMoved){if(t.ui.ie&&(!document.documentMode||9>document.documentMode)&&!e.button)return this._mouseUp(e);if(!e.which)if(e.originalEvent.altKey||e.originalEvent.ctrlKey||e.originalEvent.metaKey||e.originalEvent.shiftKey)this.ignoreMissingWhich=!0;else if(!this.ignoreMissingWhich)return this._mouseUp(e)}return(e.which||e.button)&&(this._mouseMoved=!0),this._mouseStarted?(this._mouseDrag(e),e.preventDefault()):(this._mouseDistanceMet(e)&&this._mouseDelayMet(e)&&(this._mouseStarted=this._mouseStart(this._mouseDownEvent,e)!==!1,this._mouseStarted?this._mouseDrag(e):this._mouseUp(e)),!this._mouseStarted)},_mouseUp:function(e){this.document.off("mousemove."+this.widgetName,this._mouseMoveDelegate).off("mouseup."+this.widgetName,this._mouseUpDelegate),this._mouseStarted&&(this._mouseStarted=!1,e.target===this._mouseDownEvent.target&&t.data(e.target,this.widgetName+".preventClickEvent",!0),this._mouseStop(e)),this._mouseDelayTimer&&(clearTimeout(this._mouseDelayTimer),delete this._mouseDelayTimer),this.ignoreMissingWhich=!1,s=!1,e.preventDefault()},_mouseDistanceMet:function(t){return Math.max(Math.abs(this._mouseDownEvent.pageX-t.pageX),Math.abs(this._mouseDownEvent.pageY-t.pageY))>=this.options.distance},_mouseDelayMet:function(){return this.mouseDelayMet},_mouseStart:function(){},_mouseDrag:function(){},_mouseStop:function(){},_mouseCapture:function(){return!0}}),t.ui.plugin={add:function(e,i,s){var n,o=t.ui[e].prototype;for(n in s)o.plugins[n]=o.plugins[n]||[],o.plugins[n].push([i,s[n]])},call:function(t,e,i,s){var n,o=t.plugins[e];if(o&&(s||t.element[0].parentNode&&11!==t.element[0].parentNode.nodeType))for(n=0;o.length>n;n++)t.options[o[n][0]]&&o[n][1].apply(t.element,i)}},t.widget("ui.resizable",t.ui.mouse,{version:"1.12.1",widgetEventPrefix:"resize",options:{alsoResize:!1,animate:!1,animateDuration:"slow",animateEasing:"swing",aspectRatio:!1,autoHide:!1,classes:{"ui-resizable-se":"ui-icon ui-icon-gripsmall-diagonal-se"},containment:!1,ghost:!1,grid:!1,handles:"e,s,se",helper:!1,maxHeight:null,maxWidth:null,minHeight:10,minWidth:10,zIndex:90,resize:null,start:null,stop:null},_num:function(t){return parseFloat(t)||0},_isNumber:function(t){return!isNaN(parseFloat(t))},_hasScroll:function(e,i){if("hidden"===t(e).css("overflow"))return!1;var s=i&&"left"===i?"scrollLeft":"scrollTop",n=!1;return e[s]>0?!0:(e[s]=1,n=e[s]>0,e[s]=0,n)},_create:function(){var e,i=this.options,s=this;this._addClass("ui-resizable"),t.extend(this,{_aspectRatio:!!i.aspectRatio,aspectRatio:i.aspectRatio,originalElement:this.element,_proportionallyResizeElements:[],_helper:i.helper||i.ghost||i.animate?i.helper||"ui-resizable-helper":null}),this.element[0].nodeName.match(/^(canvas|textarea|input|select|button|img)$/i)&&(this.element.wrap(t("<div class='ui-wrapper' style='overflow: hidden;'></div>").css({position:this.element.css("position"),width:this.element.outerWidth(),height:this.element.outerHeight(),top:this.element.css("top"),left:this.element.css("left")})),this.element=this.element.parent().data("ui-resizable",this.element.resizable("instance")),this.elementIsWrapper=!0,e={marginTop:this.originalElement.css("marginTop"),marginRight:this.originalElement.css("marginRight"),marginBottom:this.originalElement.css("marginBottom"),marginLeft:this.originalElement.css("marginLeft")},this.element.css(e),this.originalElement.css("margin",0),this.originalResizeStyle=this.originalElement.css("resize"),this.originalElement.css("resize","none"),this._proportionallyResizeElements.push(this.originalElement.css({position:"static",zoom:1,display:"block"})),this.originalElement.css(e),this._proportionallyResize()),this._setupHandles(),i.autoHide&&t(this.element).on("mouseenter",function(){i.disabled||(s._removeClass("ui-resizable-autohide"),s._handles.show())}).on("mouseleave",function(){i.disabled||s.resizing||(s._addClass("ui-resizable-autohide"),s._handles.hide())}),this._mouseInit()},_destroy:function(){this._mouseDestroy();var e,i=function(e){t(e).removeData("resizable").removeData("ui-resizable").off(".resizable").find(".ui-resizable-handle").remove()};return this.elementIsWrapper&&(i(this.element),e=this.element,this.originalElement.css({position:e.css("position"),width:e.outerWidth(),height:e.outerHeight(),top:e.css("top"),left:e.css("left")}).insertAfter(e),e.remove()),this.originalElement.css("resize",this.originalResizeStyle),i(this.originalElement),this},_setOption:function(t,e){switch(this._super(t,e),t){case"handles":this._removeHandles(),this._setupHandles();break;default:}},_setupHandles:function(){var e,i,s,n,o,a=this.options,r=this;if(this.handles=a.handles||(t(".ui-resizable-handle",this.element).length?{n:".ui-resizable-n",e:".ui-resizable-e",s:".ui-resizable-s",w:".ui-resizable-w",se:".ui-resizable-se",sw:".ui-resizable-sw",ne:".ui-resizable-ne",nw:".ui-resizable-nw"}:"e,s,se"),this._handles=t(),this.handles.constructor===String)for("all"===this.handles&&(this.handles="n,e,s,w,se,sw,ne,nw"),s=this.handles.split(","),this.handles={},i=0;s.length>i;i++)e=t.trim(s[i]),n="ui-resizable-"+e,o=t("<div>"),this._addClass(o,"ui-resizable-handle "+n),o.css({zIndex:a.zIndex}),this.handles[e]=".ui-resizable-"+e,this.element.append(o);this._renderAxis=function(e){var i,s,n,o;e=e||this.element;for(i in this.handles)this.handles[i].constructor===String?this.handles[i]=this.element.children(this.handles[i]).first().show():(this.handles[i].jquery||this.handles[i].nodeType)&&(this.handles[i]=t(this.handles[i]),this._on(this.handles[i],{mousedown:r._mouseDown})),this.elementIsWrapper&&this.originalElement[0].nodeName.match(/^(textarea|input|select|button)$/i)&&(s=t(this.handles[i],this.element),o=/sw|ne|nw|se|n|s/.test(i)?s.outerHeight():s.outerWidth(),n=["padding",/ne|nw|n/.test(i)?"Top":/se|sw|s/.test(i)?"Bottom":/^e$/.test(i)?"Right":"Left"].join(""),e.css(n,o),this._proportionallyResize()),this._handles=this._handles.add(this.handles[i])},this._renderAxis(this.element),this._handles=this._handles.add(this.element.find(".ui-resizable-handle")),this._handles.disableSelection(),this._handles.on("mouseover",function(){r.resizing||(this.className&&(o=this.className.match(/ui-resizable-(se|sw|ne|nw|n|e|s|w)/i)),r.axis=o&&o[1]?o[1]:"se")}),a.autoHide&&(this._handles.hide(),this._addClass("ui-resizable-autohide"))},_removeHandles:function(){this._handles.remove()},_mouseCapture:function(e){var i,s,n=!1;for(i in this.handles)s=t(this.handles[i])[0],(s===e.target||t.contains(s,e.target))&&(n=!0);return!this.options.disabled&&n},_mouseStart:function(e){var i,s,n,o=this.options,a=this.element;return this.resizing=!0,this._renderProxy(),i=this._num(this.helper.css("left")),s=this._num(this.helper.css("top")),o.containment&&(i+=t(o.containment).scrollLeft()||0,s+=t(o.containment).scrollTop()||0),this.offset=this.helper.offset(),this.position={left:i,top:s},this.size=this._helper?{width:this.helper.width(),height:this.helper.height()}:{width:a.width(),height:a.height()},this.originalSize=this._helper?{width:a.outerWidth(),height:a.outerHeight()}:{width:a.width(),height:a.height()},this.sizeDiff={width:a.outerWidth()-a.width(),height:a.outerHeight()-a.height()},this.originalPosition={left:i,top:s},this.originalMousePosition={left:e.pageX,top:e.pageY},this.aspectRatio="number"==typeof o.aspectRatio?o.aspectRatio:this.originalSize.width/this.originalSize.height||1,n=t(".ui-resizable-"+this.axis).css("cursor"),t("body").css("cursor","auto"===n?this.axis+"-resize":n),this._addClass("ui-resizable-resizing"),this._propagate("start",e),!0},_mouseDrag:function(e){var i,s,n=this.originalMousePosition,o=this.axis,a=e.pageX-n.left||0,r=e.pageY-n.top||0,l=this._change[o];return this._updatePrevProperties(),l?(i=l.apply(this,[e,a,r]),this._updateVirtualBoundaries(e.shiftKey),(this._aspectRatio||e.shiftKey)&&(i=this._updateRatio(i,e)),i=this._respectSize(i,e),this._updateCache(i),this._propagate("resize",e),s=this._applyChanges(),!this._helper&&this._proportionallyResizeElements.length&&this._proportionallyResize(),t.isEmptyObject(s)||(this._updatePrevProperties(),this._trigger("resize",e,this.ui()),this._applyChanges()),!1):!1},_mouseStop:function(e){this.resizing=!1;var i,s,n,o,a,r,l,h=this.options,c=this;return this._helper&&(i=this._proportionallyResizeElements,s=i.length&&/textarea/i.test(i[0].nodeName),n=s&&this._hasScroll(i[0],"left")?0:c.sizeDiff.height,o=s?0:c.sizeDiff.width,a={width:c.helper.width()-o,height:c.helper.height()-n},r=parseFloat(c.element.css("left"))+(c.position.left-c.originalPosition.left)||null,l=parseFloat(c.element.css("top"))+(c.position.top-c.originalPosition.top)||null,h.animate||this.element.css(t.extend(a,{top:l,left:r})),c.helper.height(c.size.height),c.helper.width(c.size.width),this._helper&&!h.animate&&this._proportionallyResize()),t("body").css("cursor","auto"),this._removeClass("ui-resizable-resizing"),this._propagate("stop",e),this._helper&&this.helper.remove(),!1},_updatePrevProperties:function(){this.prevPosition={top:this.position.top,left:this.position.left},this.prevSize={width:this.size.width,height:this.size.height}},_applyChanges:function(){var t={};return this.position.top!==this.prevPosition.top&&(t.top=this.position.top+"px"),this.position.left!==this.prevPosition.left&&(t.left=this.position.left+"px"),this.size.width!==this.prevSize.width&&(t.width=this.size.width+"px"),this.size.height!==this.prevSize.height&&(t.height=this.size.height+"px"),this.helper.css(t),t},_updateVirtualBoundaries:function(t){var e,i,s,n,o,a=this.options;o={minWidth:this._isNumber(a.minWidth)?a.minWidth:0,maxWidth:this._isNumber(a.maxWidth)?a.maxWidth:1/0,minHeight:this._isNumber(a.minHeight)?a.minHeight:0,maxHeight:this._isNumber(a.maxHeight)?a.maxHeight:1/0},(this._aspectRatio||t)&&(e=o.minHeight*this.aspectRatio,s=o.minWidth/this.aspectRatio,i=o.maxHeight*this.aspectRatio,n=o.maxWidth/this.aspectRatio,e>o.minWidth&&(o.minWidth=e),s>o.minHeight&&(o.minHeight=s),o.maxWidth>i&&(o.maxWidth=i),o.maxHeight>n&&(o.maxHeight=n)),this._vBoundaries=o},_updateCache:function(t){this.offset=this.helper.offset(),this._isNumber(t.left)&&(this.position.left=t.left),this._isNumber(t.top)&&(this.position.top=t.top),this._isNumber(t.height)&&(this.size.height=t.height),this._isNumber(t.width)&&(this.size.width=t.width)},_updateRatio:function(t){var e=this.position,i=this.size,s=this.axis;return this._isNumber(t.height)?t.width=t.height*this.aspectRatio:this._isNumber(t.width)&&(t.height=t.width/this.aspectRatio),"sw"===s&&(t.left=e.left+(i.width-t.width),t.top=null),"nw"===s&&(t.top=e.top+(i.height-t.height),t.left=e.left+(i.width-t.width)),t},_respectSize:function(t){var e=this._vBoundaries,i=this.axis,s=this._isNumber(t.width)&&e.maxWidth&&e.maxWidth<t.width,n=this._isNumber(t.height)&&e.maxHeight&&e.maxHeight<t.height,o=this._isNumber(t.width)&&e.minWidth&&e.minWidth>t.width,a=this._isNumber(t.height)&&e.minHeight&&e.minHeight>t.height,r=this.originalPosition.left+this.originalSize.width,l=this.originalPosition.top+this.originalSize.height,h=/sw|nw|w/.test(i),c=/nw|ne|n/.test(i);return o&&(t.width=e.minWidth),a&&(t.height=e.minHeight),s&&(t.width=e.maxWidth),n&&(t.height=e.maxHeight),o&&h&&(t.left=r-e.minWidth),s&&h&&(t.left=r-e.maxWidth),a&&c&&(t.top=l-e.minHeight),n&&c&&(t.top=l-e.maxHeight),t.width||t.height||t.left||!t.top?t.width||t.height||t.top||!t.left||(t.left=null):t.top=null,t},_getPaddingPlusBorderDimensions:function(t){for(var e=0,i=[],s=[t.css("borderTopWidth"),t.css("borderRightWidth"),t.css("borderBottomWidth"),t.css("borderLeftWidth")],n=[t.css("paddingTop"),t.css("paddingRight"),t.css("paddingBottom"),t.css("paddingLeft")];4>e;e++)i[e]=parseFloat(s[e])||0,i[e]+=parseFloat(n[e])||0;return{height:i[0]+i[2],width:i[1]+i[3]}},_proportionallyResize:function(){if(this._proportionallyResizeElements.length)for(var t,e=0,i=this.helper||this.element;this._proportionallyResizeElements.length>e;e++)t=this._proportionallyResizeElements[e],this.outerDimensions||(this.outerDimensions=this._getPaddingPlusBorderDimensions(t)),t.css({height:i.height()-this.outerDimensions.height||0,width:i.width()-this.outerDimensions.width||0})},_renderProxy:function(){var e=this.element,i=this.options;this.elementOffset=e.offset(),this._helper?(this.helper=this.helper||t("<div style='overflow:hidden;'></div>"),this._addClass(this.helper,this._helper),this.helper.css({width:this.element.outerWidth(),height:this.element.outerHeight(),position:"absolute",left:this.elementOffset.left+"px",top:this.elementOffset.top+"px",zIndex:++i.zIndex}),this.helper.appendTo("body").disableSelection()):this.helper=this.element},_change:{e:function(t,e){return{width:this.originalSize.width+e}},w:function(t,e){var i=this.originalSize,s=this.originalPosition;return{left:s.left+e,width:i.width-e}},n:function(t,e,i){var s=this.originalSize,n=this.originalPosition;return{top:n.top+i,height:s.height-i}},s:function(t,e,i){return{height:this.originalSize.height+i}},se:function(e,i,s){return t.extend(this._change.s.apply(this,arguments),this._change.e.apply(this,[e,i,s]))},sw:function(e,i,s){return t.extend(this._change.s.apply(this,arguments),this._change.w.apply(this,[e,i,s]))},ne:function(e,i,s){return t.extend(this._change.n.apply(this,arguments),this._change.e.apply(this,[e,i,s]))},nw:function(e,i,s){return t.extend(this._change.n.apply(this,arguments),this._change.w.apply(this,[e,i,s]))}},_propagate:function(e,i){t.ui.plugin.call(this,e,[i,this.ui()]),"resize"!==e&&this._trigger(e,i,this.ui())},plugins:{},ui:function(){return{originalElement:this.originalElement,element:this.element,helper:this.helper,position:this.position,size:this.size,originalSize:this.originalSize,originalPosition:this.originalPosition}}}),t.ui.plugin.add("resizable","animate",{stop:function(e){var i=t(this).resizable("instance"),s=i.options,n=i._proportionallyResizeElements,o=n.length&&/textarea/i.test(n[0].nodeName),a=o&&i._hasScroll(n[0],"left")?0:i.sizeDiff.height,r=o?0:i.sizeDiff.width,l={width:i.size.width-r,height:i.size.height-a},h=parseFloat(i.element.css("left"))+(i.position.left-i.originalPosition.left)||null,c=parseFloat(i.element.css("top"))+(i.position.top-i.originalPosition.top)||null;i.element.animate(t.extend(l,c&&h?{top:c,left:h}:{}),{duration:s.animateDuration,easing:s.animateEasing,step:function(){var s={width:parseFloat(i.element.css("width")),height:parseFloat(i.element.css("height")),top:parseFloat(i.element.css("top")),left:parseFloat(i.element.css("left"))};n&&n.length&&t(n[0]).css({width:s.width,height:s.height}),i._updateCache(s),i._propagate("resize",e)}})}}),t.ui.plugin.add("resizable","containment",{start:function(){var e,i,s,n,o,a,r,l=t(this).resizable("instance"),h=l.options,c=l.element,u=h.containment,d=u instanceof t?u.get(0):/parent/.test(u)?c.parent().get(0):u;d&&(l.containerElement=t(d),/document/.test(u)||u===document?(l.containerOffset={left:0,top:0},l.containerPosition={left:0,top:0},l.parentData={element:t(document),left:0,top:0,width:t(document).width(),height:t(document).height()||document.body.parentNode.scrollHeight}):(e=t(d),i=[],t(["Top","Right","Left","Bottom"]).each(function(t,s){i[t]=l._num(e.css("padding"+s))}),l.containerOffset=e.offset(),l.containerPosition=e.position(),l.containerSize={height:e.innerHeight()-i[3],width:e.innerWidth()-i[1]},s=l.containerOffset,n=l.containerSize.height,o=l.containerSize.width,a=l._hasScroll(d,"left")?d.scrollWidth:o,r=l._hasScroll(d)?d.scrollHeight:n,l.parentData={element:d,left:s.left,top:s.top,width:a,height:r}))},resize:function(e){var i,s,n,o,a=t(this).resizable("instance"),r=a.options,l=a.containerOffset,h=a.position,c=a._aspectRatio||e.shiftKey,u={top:0,left:0},d=a.containerElement,p=!0;d[0]!==document&&/static/.test(d.css("position"))&&(u=l),h.left<(a._helper?l.left:0)&&(a.size.width=a.size.width+(a._helper?a.position.left-l.left:a.position.left-u.left),c&&(a.size.height=a.size.width/a.aspectRatio,p=!1),a.position.left=r.helper?l.left:0),h.top<(a._helper?l.top:0)&&(a.size.height=a.size.height+(a._helper?a.position.top-l.top:a.position.top),c&&(a.size.width=a.size.height*a.aspectRatio,p=!1),a.position.top=a._helper?l.top:0),n=a.containerElement.get(0)===a.element.parent().get(0),o=/relative|absolute/.test(a.containerElement.css("position")),n&&o?(a.offset.left=a.parentData.left+a.position.left,a.offset.top=a.parentData.top+a.position.top):(a.offset.left=a.element.offset().left,a.offset.top=a.element.offset().top),i=Math.abs(a.sizeDiff.width+(a._helper?a.offset.left-u.left:a.offset.left-l.left)),s=Math.abs(a.sizeDiff.height+(a._helper?a.offset.top-u.top:a.offset.top-l.top)),i+a.size.width>=a.parentData.width&&(a.size.width=a.parentData.width-i,c&&(a.size.height=a.size.width/a.aspectRatio,p=!1)),s+a.size.height>=a.parentData.height&&(a.size.height=a.parentData.height-s,c&&(a.size.width=a.size.height*a.aspectRatio,p=!1)),p||(a.position.left=a.prevPosition.left,a.position.top=a.prevPosition.top,a.size.width=a.prevSize.width,a.size.height=a.prevSize.height)},stop:function(){var e=t(this).resizable("instance"),i=e.options,s=e.containerOffset,n=e.containerPosition,o=e.containerElement,a=t(e.helper),r=a.offset(),l=a.outerWidth()-e.sizeDiff.width,h=a.outerHeight()-e.sizeDiff.height;e._helper&&!i.animate&&/relative/.test(o.css("position"))&&t(this).css({left:r.left-n.left-s.left,width:l,height:h}),e._helper&&!i.animate&&/static/.test(o.css("position"))&&t(this).css({left:r.left-n.left-s.left,width:l,height:h})}}),t.ui.plugin.add("resizable","alsoResize",{start:function(){var e=t(this).resizable("instance"),i=e.options;t(i.alsoResize).each(function(){var e=t(this);e.data("ui-resizable-alsoresize",{width:parseFloat(e.width()),height:parseFloat(e.height()),left:parseFloat(e.css("left")),top:parseFloat(e.css("top"))})})},resize:function(e,i){var s=t(this).resizable("instance"),n=s.options,o=s.originalSize,a=s.originalPosition,r={height:s.size.height-o.height||0,width:s.size.width-o.width||0,top:s.position.top-a.top||0,left:s.position.left-a.left||0};t(n.alsoResize).each(function(){var e=t(this),s=t(this).data("ui-resizable-alsoresize"),n={},o=e.parents(i.originalElement[0]).length?["width","height"]:["width","height","top","left"];t.each(o,function(t,e){var i=(s[e]||0)+(r[e]||0);i&&i>=0&&(n[e]=i||null)}),e.css(n)})},stop:function(){t(this).removeData("ui-resizable-alsoresize")}}),t.ui.plugin.add("resizable","ghost",{start:function(){var e=t(this).resizable("instance"),i=e.size;e.ghost=e.originalElement.clone(),e.ghost.css({opacity:.25,display:"block",position:"relative",height:i.height,width:i.width,margin:0,left:0,top:0}),e._addClass(e.ghost,"ui-resizable-ghost"),t.uiBackCompat!==!1&&"string"==typeof e.options.ghost&&e.ghost.addClass(this.options.ghost),e.ghost.appendTo(e.helper)},resize:function(){var e=t(this).resizable("instance");e.ghost&&e.ghost.css({position:"relative",height:e.size.height,width:e.size.width})},stop:function(){var e=t(this).resizable("instance");e.ghost&&e.helper&&e.helper.get(0).removeChild(e.ghost.get(0))}}),t.ui.plugin.add("resizable","grid",{resize:function(){var e,i=t(this).resizable("instance"),s=i.options,n=i.size,o=i.originalSize,a=i.originalPosition,r=i.axis,l="number"==typeof s.grid?[s.grid,s.grid]:s.grid,h=l[0]||1,c=l[1]||1,u=Math.round((n.width-o.width)/h)*h,d=Math.round((n.height-o.height)/c)*c,p=o.width+u,f=o.height+d,g=s.maxWidth&&p>s.maxWidth,m=s.maxHeight&&f>s.maxHeight,_=s.minWidth&&s.minWidth>p,v=s.minHeight&&s.minHeight>f;s.grid=l,_&&(p+=h),v&&(f+=c),g&&(p-=h),m&&(f-=c),/^(se|s|e)$/.test(r)?(i.size.width=p,i.size.height=f):/^(ne)$/.test(r)?(i.size.width=p,i.size.height=f,i.position.top=a.top-d):/^(sw)$/.test(r)?(i.size.width=p,i.size.height=f,i.position.left=a.left-u):((0>=f-c||0>=p-h)&&(e=i._getPaddingPlusBorderDimensions(this)),f-c>0?(i.size.height=f,i.position.top=a.top-d):(f=c-e.height,i.size.height=f,i.position.top=a.top+o.height-f),p-h>0?(i.size.width=p,i.position.left=a.left-u):(p=h-e.width,i.size.width=p,i.position.left=a.left+o.width-p))}}),t.ui.resizable});
\ No newline at end of file
diff --git a/inferno/script.js b/inferno/script.js
index 834504a..c2c16a8 100644
--- a/inferno/script.js
+++ b/inferno/script.js
@@ -1,248 +1,263 @@
+/*
+ * Copyright (C) 2017 The Android Open Source Project
+ *
+ * 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.
+ */
'use strict';
-function init() {
- let x = document.getElementsByTagName('svg');
- for (let i = 0; i < x.length; i++) {
- createZoomHistoryStack(x[i]);
- }
+function flamegraphInit() {
+ let flamegraph = document.getElementById('flamegraph_id');
+ let svgs = flamegraph.getElementsByTagName('svg');
+ for (let i = 0; i < svgs.length; ++i) {
+ createZoomHistoryStack(svgs[i]);
+ adjust_text_size(svgs[i]);
+ }
+
+ function throttle(callback) {
+ let running = false;
+ return function() {
+ if (!running) {
+ running = true;
+ window.requestAnimationFrame(function () {
+ callback();
+ running = false;
+ });
+ }
+ };
+ }
+ window.addEventListener('resize', throttle(function() {
+ let flamegraph = document.getElementById('flamegraph_id');
+ let svgs = flamegraph.getElementsByTagName('svg');
+ for (let i = 0; i < svgs.length; ++i) {
+ adjust_text_size(svgs[i]);
+ }
+ }));
}
// Create a stack add the root svg element in it.
function createZoomHistoryStack(svgElement) {
- svgElement.zoomStack = [svgElement.getElementById(svgElement.attributes['rootid'].value)];
-}
-
-function dumpStack(svgElement) {
- // Disable (enable for debugging)
- return;
- stack = svgElement.zoomStack;
- for (i=0; i < stack.length; i++) {
- let title = stack[i].getElementsByTagName('title')[0];
- console.log('[' +i+ ']-' + title.textContent);
- }
+ svgElement.zoomStack = [svgElement.getElementById(svgElement.attributes['rootid'].value)];
}
function adjust_node_text_size(x, svgWidth) {
- let title = x.getElementsByTagName('title')[0];
- let text = x.getElementsByTagName('text')[0];
- let rect = x.getElementsByTagName('rect')[0];
+ let title = x.getElementsByTagName('title')[0];
+ let text = x.getElementsByTagName('text')[0];
+ let rect = x.getElementsByTagName('rect')[0];
- let width = parseFloat(rect.attributes['width'].value) * svgWidth * 0.01;
+ let width = parseFloat(rect.attributes['width'].value) * svgWidth * 0.01;
- // Don't even bother trying to find a best fit. The area is too small.
- if (width < 28) {
- text.textContent = '';
- return;
- }
- // Remove dso and #samples which are here only for mouseover purposes.
- let methodName = title.textContent.split(' | ')[0];
+ // Don't even bother trying to find a best fit. The area is too small.
+ if (width < 28) {
+ text.textContent = '';
+ return;
+ }
+ // Remove dso and #samples which are here only for mouseover purposes.
+ let methodName = title.textContent.split(' | ')[0];
- let numCharacters;
- for (numCharacters = methodName.length; numCharacters > 4; numCharacters--) {
- // Avoid reflow by using hard-coded estimate instead of text.getSubStringLength(0, numCharacters)
- // if (text.getSubStringLength(0, numCharacters) <= width) {
- if (numCharacters * 7.5 <= width) {
- break;
- }
- }
+ let numCharacters;
+ for (numCharacters = methodName.length; numCharacters > 4; numCharacters--) {
+ // Avoid reflow by using hard-coded estimate instead of
+ // text.getSubStringLength(0, numCharacters).
+ if (numCharacters * 7.5 <= width) {
+ break;
+ }
+ }
- if (numCharacters == methodName.length) {
- text.textContent = methodName;
- return;
- }
+ if (numCharacters == methodName.length) {
+ text.textContent = methodName;
+ return;
+ }
- text.textContent = methodName.substring(0, numCharacters-2) + '..';
- }
+ text.textContent = methodName.substring(0, numCharacters-2) + '..';
+}
function adjust_text_size(svgElement) {
- let svgWidth = $(svgElement).parent().width();
- let x = svgElement.getElementsByTagName('g');
- for (let i = 0; i < x.length; i++) {
- adjust_node_text_size(x[i], svgWidth);
- }
+ let svgWidth = window.innerWidth;
+ let x = svgElement.getElementsByTagName('g');
+ for (let i = 0; i < x.length; i++) {
+ adjust_node_text_size(x[i], svgWidth);
+ }
}
function zoom(e) {
- let svgElement = e.ownerSVGElement;
- let zoomStack = svgElement.zoomStack;
- zoomStack.push(e);
- displayFromElement(e);
- select(e);
- dumpStack(svgElement);
+ let svgElement = e.ownerSVGElement;
+ let zoomStack = svgElement.zoomStack;
+ zoomStack.push(e);
+ displayFromElement(e);
+ select(e);
- // Show zoom out button.
- svgElement.getElementById('zoom_rect').style.display = 'block';
- svgElement.getElementById('zoom_text').style.display = 'block';
+ // Show zoom out button.
+ svgElement.getElementById('zoom_rect').style.display = 'block';
+ svgElement.getElementById('zoom_text').style.display = 'block';
}
function displayFromElement(e) {
- let clicked_rect = e.getElementsByTagName('rect')[0];
- let clicked_origin_x = clicked_rect.attributes['ox'].value;
- let clicked_origin_y = clicked_rect.attributes['oy'].value;
- let clicked_origin_width = clicked_rect.attributes['owidth'].value;
+ let clicked_rect = e.getElementsByTagName('rect')[0];
+ let clicked_origin_x = clicked_rect.attributes['ox'].value;
+ let clicked_origin_y = clicked_rect.attributes['oy'].value;
+ let clicked_origin_width = clicked_rect.attributes['owidth'].value;
- let svgBox = e.ownerSVGElement.getBoundingClientRect();
- let svgBoxHeight = svgBox.height;
- let svgBoxWidth = 100;
- let scaleFactor = svgBoxWidth / clicked_origin_width;
+ let svgBox = e.ownerSVGElement.getBoundingClientRect();
+ let svgBoxHeight = svgBox.height;
+ let svgBoxWidth = 100;
+ let scaleFactor = svgBoxWidth / clicked_origin_width;
- let callsites = e.ownerSVGElement.getElementsByTagName('g');
- for (let i = 0; i < callsites.length; i++) {
- let text = callsites[i].getElementsByTagName('text')[0];
- let rect = callsites[i].getElementsByTagName('rect')[0];
+ let callsites = e.ownerSVGElement.getElementsByTagName('g');
+ for (let i = 0; i < callsites.length; i++) {
+ let text = callsites[i].getElementsByTagName('text')[0];
+ let rect = callsites[i].getElementsByTagName('rect')[0];
- let rect_o_x = parseFloat(rect.attributes['ox'].value);
- let rect_o_y = parseFloat(rect.attributes['oy'].value);
+ let rect_o_x = parseFloat(rect.attributes['ox'].value);
+ let rect_o_y = parseFloat(rect.attributes['oy'].value);
- // Avoid multiple forced reflow by hiding nodes.
- if (rect_o_y > clicked_origin_y) {
- rect.style.display = 'none';
- text.style.display = 'none';
- continue;
- }
- rect.style.display = 'block';
- text.style.display = 'block';
+ // Avoid multiple forced reflow by hiding nodes.
+ if (rect_o_y > clicked_origin_y) {
+ rect.style.display = 'none';
+ text.style.display = 'none';
+ continue;
+ }
+ rect.style.display = 'block';
+ text.style.display = 'block';
- let newrec_x = rect.attributes['x'].value = (rect_o_x - clicked_origin_x) * scaleFactor + "%";
- let newrec_y = rect.attributes['y'].value = rect_o_y + (svgBoxHeight - clicked_origin_y
+ let newrec_x = rect.attributes['x'].value = (rect_o_x - clicked_origin_x) * scaleFactor +
+ '%';
+ let newrec_y = rect.attributes['y'].value = rect_o_y + (svgBoxHeight - clicked_origin_y
- 17 - 2);
- text.attributes['y'].value = newrec_y + 12;
- text.attributes['x'].value = newrec_x;
+ text.attributes['y'].value = newrec_y + 12;
+ text.attributes['x'].value = newrec_x;
- rect.attributes['width'].value = (rect.attributes['owidth'].value * scaleFactor) + "%";
- }
+ rect.attributes['width'].value = (rect.attributes['owidth'].value * scaleFactor) + '%';
+ }
- adjust_text_size(e.ownerSVGElement);
+ adjust_text_size(e.ownerSVGElement);
}
function unzoom(e) {
- let svgOwner = e.ownerSVGElement;
- let stack = svgOwner.zoomStack;
+ let svgOwner = e.ownerSVGElement;
+ let stack = svgOwner.zoomStack;
- // Unhighlight whatever was selected.
- if (selected) {
- selected.classList.remove('s');
- }
+ // Unhighlight whatever was selected.
+ if (selected) {
+ selected.classList.remove('s');
+ }
- // Stack management: Never remove the last element which is the flamegraph root.
- if (stack.length > 1) {
- let previouslySelected = stack.pop();
- select(previouslySelected);
- }
- let nextElement = stack[stack.length-1];
+ // Stack management: Never remove the last element which is the flamegraph root.
+ if (stack.length > 1) {
+ let previouslySelected = stack.pop();
+ select(previouslySelected);
+ }
+ let nextElement = stack[stack.length-1];
- // Hide zoom out button.
- if (stack.length == 1) {
- svgOwner.getElementById('zoom_rect').style.display = 'none';
- svgOwner.getElementById('zoom_text').style.display = 'none';
- }
+ // Hide zoom out button.
+ if (stack.length == 1) {
+ svgOwner.getElementById('zoom_rect').style.display = 'none';
+ svgOwner.getElementById('zoom_text').style.display = 'none';
+ }
- displayFromElement(nextElement);
- dumpStack(svgOwner);
+ displayFromElement(nextElement);
}
function search(e) {
- let term = prompt('Search for:', '');
- let svgOwner = e.ownerSVGElement;
- let callsites = e.ownerSVGElement.getElementsByTagName('g');
+ let term = prompt('Search for:', '');
+ let callsites = e.ownerSVGElement.getElementsByTagName('g');
- if (!term) {
+ if (!term) {
+ for (let i = 0; i < callsites.length; i++) {
+ let rect = callsites[i].getElementsByTagName('rect')[0];
+ rect.attributes['fill'].value = rect.attributes['ofill'].value;
+ }
+ return;
+ }
+
for (let i = 0; i < callsites.length; i++) {
- let rect = callsites[i].getElementsByTagName('rect')[0];
- rect.attributes['fill'].value = rect.attributes['ofill'].value;
+ let title = callsites[i].getElementsByTagName('title')[0];
+ let rect = callsites[i].getElementsByTagName('rect')[0];
+ if (title.textContent.indexOf(term) != -1) {
+ rect.attributes['fill'].value = 'rgb(230,100,230)';
+ } else {
+ rect.attributes['fill'].value = rect.attributes['ofill'].value;
+ }
}
- return;
- }
-
- for (let i = 0; i < callsites.length; i++) {
- let title = callsites[i].getElementsByTagName('title')[0];
- let rect = callsites[i].getElementsByTagName('rect')[0];
- if (title.textContent.indexOf(term) != -1) {
- rect.attributes['fill'].value = 'rgb(230,100,230)';
- } else {
- rect.attributes['fill'].value = rect.attributes['ofill'].value;
- }
- }
}
let selected;
document.onkeydown = function handle_keyboard_input(e) {
- if (!selected) {
- return;
- }
+ if (!selected) {
+ return;
+ }
- let title = selected.getElementsByTagName('title')[0];
- let nav = selected.attributes['nav'].value.split(',');
- let navigation_index;
- switch (e.keyCode) {
- // case 38: // ARROW UP
- case 87: navigation_index = 0; break; // W
+ let nav = selected.attributes['nav'].value.split(',');
+ let navigation_index;
+ switch (e.keyCode) {
+ // case 38: // ARROW UP
+ case 87: navigation_index = 0; break; // W
- // case 32 : // ARROW LEFT
- case 65: navigation_index = 1; break; // A
+ // case 32 : // ARROW LEFT
+ case 65: navigation_index = 1; break; // A
- // case 43: // ARROW DOWN
- case 68: navigation_index = 3; break; // S
+ // case 43: // ARROW DOWN
+ case 68: navigation_index = 3; break; // S
- // case 39: // ARROW RIGHT
- case 83: navigation_index = 2; break; // D
+ // case 39: // ARROW RIGHT
+ case 83: navigation_index = 2; break; // D
- case 32: zoom(selected); return false; // SPACE
+ case 32: zoom(selected); return false; // SPACE
- case 8: // BACKSPACE
- unzoom(selected); return false;
- default: return true;
- }
+ case 8: // BACKSPACE
+ unzoom(selected); return false;
+ default: return true;
+ }
- if (nav[navigation_index] == '0') {
+ if (nav[navigation_index] == '0') {
+ return false;
+ }
+
+ let target_element = selected.ownerSVGElement.getElementById(nav[navigation_index]);
+ select(target_element);
return false;
- }
-
- let target_element = selected.ownerSVGElement.getElementById(nav[navigation_index]);
- select(target_element);
- return false;
};
function select(e) {
- if (selected) {
- selected.classList.remove('s');
- }
- selected = e;
- selected.classList.add('s');
-
- // Update info bar
- let titleElement = selected.getElementsByTagName('title')[0];
- let text = titleElement.textContent;
-
- // Parse title
- let method_and_info = text.split(' | ');
- let methodName = method_and_info[0];
- let info = method_and_info[1];
-
- // Parse info
- // '/system/lib64/libhwbinder.so (4 samples: 0.28%)'
- let regexp = /(.*) \(.* ([0-9**\.[0-9]*%)\)/g;
- let match = regexp.exec(info);
- if (match.length > 2) {
- let percentage = match[2];
- // Write percentage
- let percentageTextElement = selected.ownerSVGElement.getElementById('percent_text');
- percentageTextElement.textContent = percentage;
- // console.log("'" + percentage + "'")
- }
-
- // Set fields
- let barTextElement = selected.ownerSVGElement.getElementById('info_text');
- barTextElement.textContent = methodName;
-}
-
-$(document).ready(function() {
- $(".flamegraph_block").resizable({
- handles: "e",
- resize: function(event, ui) {
- adjust_text_size(ui.element.find("svg")[0]);
+ if (selected) {
+ selected.classList.remove('s');
}
- });
-});
\ No newline at end of file
+ selected = e;
+ selected.classList.add('s');
+
+ // Update info bar
+ let titleElement = selected.getElementsByTagName('title')[0];
+ let text = titleElement.textContent;
+
+ // Parse title
+ let method_and_info = text.split(' | ');
+ let methodName = method_and_info[0];
+ let info = method_and_info[1];
+
+ // Parse info
+ // '/system/lib64/libhwbinder.so (4 events: 0.28%)'
+ let regexp = /(.*) \(.* (\d*\.\d*%)\)/g;
+ let match = regexp.exec(info);
+ if (match.length > 2) {
+ let percentage = match[2];
+ // Write percentage
+ let percentageTextElement = selected.ownerSVGElement.getElementById('percent_text');
+ percentageTextElement.textContent = percentage;
+ // console.log("'" + percentage + "'")
+ }
+
+ // Set fields
+ let barTextElement = selected.ownerSVGElement.getElementById('info_text');
+ barTextElement.textContent = methodName;
+}
\ No newline at end of file
diff --git a/inferno/svg_renderer.py b/inferno/svg_renderer.py
index 3950572..9b654ee 100644
--- a/inferno/svg_renderer.py
+++ b/inferno/svg_renderer.py
@@ -16,7 +16,6 @@
import sys
-SVG_CANVAS_WIDTH = 1124
SVG_NODE_HEIGHT = 17
FONT_SIZE = 12
@@ -124,9 +123,9 @@
def renderSearchNode(f):
f.write(
"""<rect id="search_rect" style="stroke:rgb(0,0,0);" onclick="search(this);" class="t"
- rx="10" ry="10" x="%d" y="10" width="80" height="30" fill="rgb(255,255,255)""/>
- <text id="search_text" class="t" x="%d" y="30" onclick="search(this);">Search</text>
- """ % (SVG_CANVAS_WIDTH - 95, SVG_CANVAS_WIDTH - 80))
+ rx="10" ry="10" x="1029" y="10" width="80" height="30" fill="rgb(255,255,255)""/>
+ <text id="search_text" class="t" x="1044" y="30" onclick="search(this);">Search</text>
+ """)
def renderUnzoomNode(f):
@@ -135,41 +134,34 @@
onclick="unzoom(this);" rx="10" ry="10" x="10" y="10" width="80" height="30"
fill="rgb(255,255,255)"/>
<text id="zoom_text" style="display:none;" class="t" x="19" y="30"
- onclick="unzoom(this);">Zoom out</text>"""
- )
+ onclick="unzoom(this);">Zoom out</text>""")
def renderInfoNode(f):
f.write(
"""<clipPath id="info_clip_path"> <rect id="info_rect" style="stroke:rgb(0,0,0);"
- rx="10" ry="10" x="120" y="10" width="%d" height="30" fill="rgb(255,255,255)"/>
+ rx="10" ry="10" x="120" y="10" width="789" height="30" fill="rgb(255,255,255)"/>
</clipPath>
<rect id="info_rect" style="stroke:rgb(0,0,0);"
- rx="10" ry="10" x="120" y="10" width="%d" height="30" fill="rgb(255,255,255)"/>
- <text clip-path="url(#info_clip_path)" id="info_text" x="128" y="30"></text>
- """ % (SVG_CANVAS_WIDTH - 335, SVG_CANVAS_WIDTH - 325)
- )
+ rx="10" ry="10" x="120" y="10" width="799" height="30" fill="rgb(255,255,255)"/>
+ <text clip-path="url(#info_clip_path)" id="info_text" x="128" y="30"></text>""")
def renderPercentNode(f):
f.write(
"""<rect id="percent_rect" style="stroke:rgb(0,0,0);"
- rx="10" ry="10" x="%d" y="10" width="82" height="30" fill="rgb(255,255,255)"/>
- <text id="percent_text" text-anchor="end" x="%d" y="30">100.00%%</text>
- """ % (SVG_CANVAS_WIDTH - (95 * 2), SVG_CANVAS_WIDTH - (125))
- )
+ rx="10" ry="10" x="934" y="10" width="82" height="30" fill="rgb(255,255,255)"/>
+ <text id="percent_text" text-anchor="end" x="999" y="30">100.00%%</text>""")
-def renderSVG(flamegraph, f, color_scheme, width):
- global SVG_CANVAS_WIDTH
- SVG_CANVAS_WIDTH = width
+def renderSVG(flamegraph, f, color_scheme):
height = (flamegraph.get_max_depth() + 2) * SVG_NODE_HEIGHT
- f.write("""<div class="flamegraph_block" style="width:%dpx; height:%dpx;">
- """ % (SVG_CANVAS_WIDTH, height))
+ f.write("""<div class="flamegraph_block" style="width:100%%; height:%dpx;">
+ """ % height)
f.write("""<svg xmlns="http://www.w3.org/2000/svg"
xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1"
width="100%%" height="100%%" style="border: 1px solid black;"
- onload="adjust_text_size(this);" rootid="%d">
+ rootid="%d">
""" % (flamegraph.children[0].id))
f.write("""<defs > <linearGradient id="background_gradiant" y1="0" y2="1" x1="0" x2="0" >
<stop stop-color="#eeeeee" offset="5%" /> <stop stop-color="#efefb1" offset="90%" />
diff --git a/repo.prop b/repo.prop
index d8fba98..aaaa702 100644
--- a/repo.prop
+++ b/repo.prop
@@ -1,13 +1,13 @@
-device/asus/fugu c462a793bb04e9f9783592d6b1ba452f2d0266e3
-device/asus/fugu-kernel 6c50d39cf6e1dabdfc5ba6ce22973b96650ae0b5
-device/common 1401af3a0c8e9cf74f70bbdcae9299be30866563
+device/asus/fugu dafedf866f507762306dff5269e06af6a0de5204
+device/asus/fugu-kernel e4a8a3b9991f0c7243677adb7b868e94eba91607
+device/common 1dca2f2bfac43921adc4ba1283cf6f8120e41ac6
device/generic/arm64 392eec3cb4c64721cd5b76fa36535954b5d45309
device/generic/armv7-a-neon 9e96e041bf3c9d5b57b11f921c6339e076478a7a
-device/generic/car c4d08a9e781e9bc69b861428765e87f6cbb696ac
-device/generic/common 14bdf66f0d1f9007140fabe7152de111cc24ec5c
-device/generic/goldfish ee12b2db7e7d0b1dadb16fb601844ede4d47caf4
-device/generic/goldfish-opengl 9b2d1c3cee91b7ce39e22c47b58843037ebb78fc
-device/generic/mini-emulator-arm64 1aba9d054f2e5e906000fe69eff8d168e99e74fc
+device/generic/car be665121124618099978c75d5f69f867bc0e7113
+device/generic/common 55abed9b08565d0c72ab458f479c0f5746ad3a4e
+device/generic/goldfish c7237199bf74ffc64f2513e0f8b3c9a7f8432ab5
+device/generic/goldfish-opengl c7d47aa5c0df8b0926d7a3f605359ea7fa7aa8a2
+device/generic/mini-emulator-arm64 77d394326f7fc17831ee7702cb8b750fe0b21443
device/generic/mini-emulator-armv7-a-neon 258706fc0ddcf2c9be48967f4766ec90efb736a5
device/generic/mini-emulator-mips 544e69736cff911c286a5d1d300dac228e23a73f
device/generic/mini-emulator-mips64 b1c36d0847fe6366527efd38fdb29d757ea887ee
@@ -16,585 +16,598 @@
device/generic/mips bf27f9bcb56e483310507f78f0a4b93d3f5499c5
device/generic/mips64 34300f7753b7a9fa2f7ba16bdc1f1fb8625dd4e4
device/generic/qemu 97ef4f962aec451b00bfe562c1cb67471c4563d9
-device/generic/uml f71ea7d194988fd6a64e071918f6e00e8db9ff0c
+device/generic/uml 275d215105cc2e15eeba4f91a8762e19a4800a53
device/generic/x86 645e6e711b32f94ecbd9b1cc77a51559dfce5ebe
device/generic/x86_64 506952733fdebb5a2ef086b519ceebe86a95ae31
device/google/accessory/arduino abc5159a3ca9dbb5c7e364a1eab99901a4440ac5
device/google/accessory/demokit 7dfe7f89a3b174709c773fe319531006e46440d9
-device/google/atv a14f6c4a89589a7cc6919702f9f3fd3140cc273b
-device/google/contexthub 0334f4068ae77ffc4e2579726e5274fec51c64b1
-device/google/dragon 7424436ee4249cf5f356eef34a0dd4b57c16c3dc
-device/google/dragon-kernel 01a873b040f63d8d030b2ec34338bc86aadd15b2
-device/google/marlin 83de65b605f3a5f3115aec90a737d917c46217f8
-device/google/marlin-kernel 8d4b71409708a68a7b1aa9614fecaf17a3e8749f
-device/google/vrservices 957cfc2a4e2244a83ddfaf6ba2da74aee24f0767
-device/huawei/angler 8c68626c01750f93ac8412de79dcf1e396e9af5c
-device/huawei/angler-kernel 5263f25b197ee09005570da6363b766281969c22
-device/lge/bullhead f91eab2f09279fb6159686b566563b9a4a8899f1
-device/lge/bullhead-kernel 1d824f51ffa8950d7695829d849da183233d71e3
-device/linaro/bootloader/OpenPlatformPkg ca8a8eebc29ebbcc65edc8f3b4533867cbed9205
-device/linaro/bootloader/arm-trusted-firmware d2baddd32bbfdce3faf3c8af7b1758ea04230543
-device/linaro/bootloader/edk2 99d893a486546aa62d4088a16e86617ed3c74897
-device/linaro/hikey 0b1d49abda38e09479faa1d9dea6473d99d94997
-device/linaro/hikey-kernel 7517d7dbe37f7c9cd59bb4e2bb0056384d25f717
-device/sample c0515d458620138a0b8539d9880c6e3b2b3c99c3
-kernel/tests ddc9df77efa3e8ebc8277a98622f0b0ce098bd9f
-platform/art e6c8e3caa4fddf3afb64081d3585cfd2320a632c
-platform/bionic 0f6dd63e068440142bf07d2edc035e394053d2f0
-platform/bootable/recovery deb5de0bc144de9f62b0621d8a23db3415e4ac78
-platform/build cfde9a84bc12f8b9769eea352ea33c2f576f9e45
-platform/build/blueprint b47f82f0173707a293a6319dc1ef8c6ffea8cfd2
-platform/build/kati d18d5bb2122043a37570d8fcaf2f212ec505ff3f
-platform/build/soong 0bf1f85c489d3b649c3f5fd9d064586ed3376a36
-platform/compatibility/cdd 644a1933b78f1b08e86cecb58e199cd43ff7bd26
-platform/cts 9fba547b6b1ea6f968cbfcc35f391cdecb653e11
-platform/dalvik 363daca666204c364be91b3f3f30fc02c483e88a
-platform/developers/build dac7653ebb52f2de72352a3ba8e91b7d70837ee5
-platform/developers/demos baed7559c8272ae802aa581148851a7f81307d47
-platform/developers/samples/android 0f2a748e3111611b65421dc42840def7ae2797de
-platform/development 6defd7bd2eff635cabfed28ff0dcc0a49c9131b5
-platform/docs/source.android.com 6fcef5f72c8b2ea92f472b6335bc86408da97b33
-platform/external/ImageMagick 5691a8ecaba367d04b2c2a13ec1e9e05f4f17b7c
-platform/external/Microsoft-GSL c1b94b10a13f22e951c429e056b151d340238e27
-platform/external/aac 32b8bd3dc952952a097fbaa87a626d15445f15c7
+device/google/atv cbbd0bb53e6a40191fbebf18df6d1a67587721e8
+device/google/contexthub a8d1f4f523cf20a0317233cdd752ba90442a8487
+device/google/dragon 4d9672465da3e5db0ec3cbbad56b188f49a151c5
+device/google/dragon-kernel 4b7bdcda6d72d7497c5aa7566f8078e8f91115ce
+device/google/marlin 6f1cc2c0b99498410c008254400a689b095dad68
+device/google/marlin-kernel e3dc8bcb4021fada94c15c96b0c8aa041578c7ce
+device/google/muskie d86f50b8bc86e2d4ae75b67009eb66b72995806b
+device/google/taimen e6c72abc13cd36ca1a122adec0abeb41139b780f
+device/google/vrservices cf0636cbc21bf95f7de6c525e8357eb8d9fffce0
+device/google/wahoo 31411fcf06ab29c09fdb4c459e269a546b72b91c
+device/google/wahoo-kernel 05b4cb375b46d9a567c3a46bfeb82616ae47aa43
+device/huawei/angler f7097ab83091cb59154bede747edb832d2b7e57b
+device/huawei/angler-kernel ec23a43db4563f9020323608d10f8f768c1855fa
+device/lge/bullhead b5525165db93b411ac29e648a5a969a4587e025e
+device/lge/bullhead-kernel c682a6d043b5b70bbebd6f8e0ef921713eb6c318
+device/linaro/bootloader/OpenPlatformPkg 8ec03d3d46f6823baac1890b749b8e632ca85d75
+device/linaro/bootloader/arm-trusted-firmware 8fa3096942d2faed55122efd47348dacca991141
+device/linaro/bootloader/edk2 16164a6d52c3ce842891d64420b119fe8410fe29
+device/linaro/hikey bc86535529da148828db594772723ef472ae9f5f
+device/linaro/hikey-kernel b6a77bc3d4916a12f65520dacef2136e28e6df5c
+device/sample 978ab19455a9078ef20bffa6d4c17b3af8aa1ead
+kernel/configs 393cac1ab8f0f16a844797385681618db715cb0f
+kernel/tests 4d0cc9cfe431350d038d869eafeba9865fd422f4
+platform/art 3e6c13997373efac343a65740da0c4f6e77338b9
+platform/bionic 048e18045d6e7bcaf0870d8bb7cb671214c9ad50
+platform/bootable/recovery 1bc0117755615b0c022ca738cc40e47a243e3847
+platform/build 2f05802a78be87250b8ca4afb23c96c02aac1cfb
+platform/build/blueprint 886dc6b72e080639938e045ba48170c499675718
+platform/build/kati 330fd37d4e4fce12ceb47764879f202b527bbe00
+platform/build/soong b563989f97c2579061ccb12152efecebdf762a33
+platform/compatibility/cdd ece19098fbc22ce5c3b76642a71019010bd161a8
+platform/cts eec2fcbceb86301a220d160370adb0dc7b8eb34c
+platform/dalvik 2799b7380e5e2e32ba992cb6a8db8d9c970cc357
+platform/developers/build 53092eae9edf0d4d2a177344f4308ea89845d582
+platform/developers/demos 48bb8cdfa431833041345406f2030f92d62e3e4f
+platform/developers/samples/android 8da42e6763864ce074159ce175a5f045fb52c21d
+platform/development 8620d08ce4c84d6657b81dca21a154f03717ce7b
+platform/docs/source.android.com 96deba20c83e0d2756e4c7861622a554955fe8db
+platform/external/ImageMagick f3397017e3b682548667ff0964b87408ea3a25fd
+platform/external/Microsoft-GSL b699d3dc833cd1555752d3080ea023452c60470c
+platform/external/aac 82440fab169e32ae16f6ea514cbdacffa752aef8
platform/external/abi-compliance-checker 5453ac991183dc8dc856e8beced059edab94acaf
-platform/external/abi-dumper d972c9eaf4e7ccd0f79e779fe4621538cc9c9e90
-platform/external/adt-infra 633951cf054fe937eab7ff3ebe151ed72966f41e
-platform/external/android-clat a9ac3414a215e41ac2e5298d0de1c670b9f5936d
+platform/external/abi-dumper 815ccfe2318cdeb44b01b9b5c2aa9e2421a08f77
+platform/external/adt-infra 09f9d910259cebfaf6547a53c1bd5a783b1ef2f3
+platform/external/android-clat 0ca87dfcd8255ac9d68c294ae43a1e13335dfb6f
platform/external/androidplot c66727ebf001607cee14521c35bc852b55fd9845
+platform/external/annotation-tools c81da9298e7d2f08ff3ec9eb3240c7aff2fd25e1
platform/external/ant-glob a73228afa9540b9c5518d360c5ae630bb634f975
platform/external/antlr dd5fa6d48b827c5d98b625adbc209f4a05567534
platform/external/apache-commons-math 18f62fca59d387e3c1ccd8f80087d9c9af40bcc8
-platform/external/apache-harmony f0c97e611fc9f6d064a76c48a503aa0bf9d3e34d
-platform/external/apache-http 6d301174f0502912ee7df6db4bc5c1f5cf4ee230
+platform/external/apache-harmony 1f77e5999dbde45b9aa258c42f10849f60e49701
+platform/external/apache-http 40be99b8679c9d93a7ded8f2daaf37ce0d464533
platform/external/apache-xml 31d7642eb8f37a9166db7f1c9e313ab651bdb8ba
-platform/external/archive-patcher a764fc9e8810be26a51add8a79502e411e3ee1ee
+platform/external/archive-patcher 2d9c08d9c336bc53a3569342003a5af905a06d3d
platform/external/arm-neon-tests 104e2a662ec8443e9a7603b22efbb3b86c1e7094
-platform/external/autotest 471196b166a43790e2b3f9aa666eb597725755ca
-platform/external/avb fc2531374e305b93b62855d659521f50cdb0b53e
-platform/external/bison 7467d52af437dd3dfea237a6865ab84bf258dee0
+platform/external/autotest a9e4741d303718abf397fb5d83f0aa727fe1424d
+platform/external/avb 85081ffb38b823418d5b4d72725a26cf9e1f8b81
+platform/external/bart eb9f1bd727e47e1d87300d908ee041464a6515f7
+platform/external/bison 9daf3ca0930745604c1a0b790223269c48877b4b
platform/external/blktrace d345431f16b8f76f30a58193ff2b26d5853e1109
-platform/external/boringssl 05e49fc79f61f4be37c1bb5bee2931524d1939c6
-platform/external/bouncycastle 410db1cb56f59fbef788f0236dac582f80f91e1a
-platform/external/brotli 20f8b631b833531fe1c72e5d31805a447a24d8ba
-platform/external/bsdiff 84fc23869cb9a881736cd965fa58b4db7c313662
-platform/external/bzip2 10161487330cae1326cd75bdef14a44920e094c1
-platform/external/c-ares 9f1b2827cb88c58a80c470a4bddb2485677ec57f
-platform/external/caliper fc57a58a3420e4f3e2ab86233ef932efa266ec1e
-platform/external/cblas 10c02ec5bc1eef9b34896eb3d3d0dbadfd51d91f
+platform/external/boringssl a7dc4759b5f834b50a9be8dd3ec1abb7ff3e5bc5
+platform/external/bouncycastle 7377028a95a04fd7de0a8b4d821cfb2db7e66f3e
+platform/external/brotli b24c03c6f537dc6d916cda6edce1cf6ae9361c9d
+platform/external/bsdiff 2991153473fdacff2437e1c8367cd224042db29c
+platform/external/bzip2 94d2512ef4ba303b00cbfecef0236f53f036a364
+platform/external/c-ares ee4a8ea891d21e4e1d706e089959a21d248bf1b1
+platform/external/caliper 6e6c481ab7de884ff5cbea6339465229e6f64f2f
+platform/external/cblas 22f61bd1f525bb66de703b712b42105ab2ec857c
platform/external/chromium-libpac 8c47eebd81c996f17198edb2fa504c04d48d567e
-platform/external/chromium-trace 88bb68148bd0510a1564d673421f051011b4240c
-platform/external/chromium-webview 955f673b041b96ffc383190797c3613ac8dba844
-platform/external/clang dc1b97084f35b0f1c7f6a5356d68a09f472a6620
+platform/external/chromium-trace 3e601f2c29e63f5151aa982790deea52645bc6ea
+platform/external/chromium-webview 4976d3ad0fe241cbc245c7798a2a5ce6b12edc10
+platform/external/clang 794bc3852146342eef74c568b397e2edd0da2b7b
platform/external/cmockery 9199c7bfafefea32d1884182fa655b6e4578c1c4
-platform/external/compiler-rt 954aa6618cc6ca0deaabb460e4d9fba2caa7a0af
-platform/external/conscrypt 387e377237aeea22be57357abf11d94ae62d9edf
-platform/external/crcalc 5559610bea3b267abef3273868f2d6b8c01409a5
-platform/external/cros/system_api 5f36ab6399bac201c34b9520cec5b7b53007627b
-platform/external/curl 8eff9c322bc3706bafe2e0e3608526b949493263
-platform/external/dagger2 4adf2d360d1a32a1e053cd181fcbc45ca7a0bd2a
-platform/external/deqp 2a3edba962ff9efe07a759bfdc763e50d5400ecc
-platform/external/desugar 9aacba113f461d7ae6f25443bd79922cee967f3e
-platform/external/dexmaker d9c3d7ebd35e46ae0e417de636d9fb6126114de8
-platform/external/dhcpcd-6.8.2 54d1fc2d574d21f788abb64b10c9a6352eba0b65
+platform/external/compiler-rt 260e2e68b9b12fbb1d40a573e9a6cd0c04498c7a
+platform/external/conscrypt 0d4a10317b2c41d7ba8618cd17448a06b82ef43f
+platform/external/crcalc dfed3c6d374ce288e7ba5bf987ccaed3ba14eea6
+platform/external/cros/system_api f8cb6e381b765306f3c551c27ecaaf332428c2ed
+platform/external/curl d8a68223b1c4e6364984b86601e5d06af8a141f7
+platform/external/dagger2 e5d08c40449250c30d1225ef107df7623aa5806f
+platform/external/deqp 3e98e4c9110716e3a08d35749610830152e4654b
+platform/external/desugar b94c5b8efd3f113d1bf71effd3f5fb4d6c03e198
+platform/external/devlib 63fd9cc1fc6e6b0ceb78950d96f905fdd3edee32
+platform/external/dexmaker a03f6220bfadb68fa5688d17e2f07a7fc0b63b5e
+platform/external/dhcpcd-6.8.2 10a0659b1f37ffa6bbfc7e66fb5b840c73588bca
platform/external/dlmalloc 6661f3ca66b55d8f5a57b96fec97efaf8f3897a5
-platform/external/dng_sdk fc2eb6731caf1190cdb6c547224d881e49d6623e
-platform/external/dnsmasq d37f567618c3f5266d82b971a505f5a214a03f34
-platform/external/doclava e15c52f95e09391629d4046a1ceb296ae29710cd
+platform/external/dng_sdk 9fb2201fe0d420f39f8393bbf776b05abf8fcd3e
+platform/external/dnsmasq 32112bf47f4ea1a9d03eb94989ee8811f1d5e83e
+platform/external/doclava f41e16b37b011808e2f039b15cb2539907dbf403
platform/external/donuts 005389fb31a67841b5de42f0d25fbfab843e208c
-platform/external/drm_gralloc f2936dc85a5f92d4dcdfa7b46066c9cd64086720
-platform/external/drm_hwcomposer 56a3fe4fcd5dcc2ed2266723f415b4fba22caf4b
-platform/external/droiddriver 5ef7dc3be02aedf00fc3cc3f59379d598e5c8817
+platform/external/drm_hwcomposer 1ad48d06624a684c645268a60e06394839be6044
+platform/external/droiddriver 42246bba6f931498161e7bff9f1e15fd391ce5b4
platform/external/drrickorang 8ea137d1086e621188f70ffac34a6ba8924af2c6
-platform/external/dtc e3169bcf71f41dd48c554c8b1f75f536df9be997
-platform/external/e2fsprogs 1f984ce998216296cbcb5f78d19571f0335e983d
-platform/external/easymock 8bc2748f2850c88ab6b86c6ceed1ce65bed5ee16
-platform/external/eclipse-basebuilder 076f46888ed346775b8efc564d7694063ef5eea7
-platform/external/eclipse-windowbuilder c533332008088ee5f61745b724361fbde5b7e770
-platform/external/eigen 977f4c94b5d153bed802aeeaffa13425ec017372
-platform/external/elfutils f1f212563ee672a1eca4238a5cc6ff1e969ec071
+platform/external/dtc 3281c3abb9f0d29d1ef11b0b2994f5f440c346d6
+platform/external/e2fsprogs 3db011ff8f4426a0fa051b101dbf6a2f69f1d8ec
+platform/external/easymock ae369e1738f81cbcf6118adac03ec0e8bcbb8400
+platform/external/eigen 3a6f837d1ecb2969e6349dec9682206a79f8cb4e
+platform/external/elfutils 4842833de9733d911f40e639e10ce4a28876ae20
platform/external/emma deb5711b2635ee9d332605e2e00b63b32bed6777
-platform/external/error_prone 0fc37229d73c096278e0b8d29c849e36e1ffea6f
+platform/external/error_prone 7de486a4d6682c8a30340525c4ed6f43228fbfab
platform/external/esd 943c42b6f8e9afe821744aa4c039f4943ebf29f5
-platform/external/expat f1e893626ddf5c5527fd67e2267f9e658f94d76f
+platform/external/expat 44802c7a437c6c4d3bff9e7be703dc62c66bc740
platform/external/eyes-free 16bd4c7a4d1bfe229068b637614dad7c48dd2ceb
-platform/external/f2fs-tools 27a6ad377c141f95ac208a0302a1a50cabeeff3c
-platform/external/fdlibm c541d3f40207616d6f8d03a982f593b32f3c2167
-platform/external/fec 1be76f512a85176bf5c59b6b9acf88a38f4895c8
-platform/external/fio 4b6b097c135637252e77c543aca66fe0461f636f
-platform/external/flac a44d56a9c4ad0093a1dca3e3be7ddb1905da2648
+platform/external/f2fs-tools 5d24dd5ef8988828c3cbe0afa80a86029e115d11
+platform/external/fdlibm 78b0ef5e5b31ae975bde55cf42e716fc4971e038
+platform/external/fec 780562b700b5497bf65831b87c9c535caafb66ed
+platform/external/fio 80b784cd42893d10adb9e7d889306fb1f2bbc18c
+platform/external/flac 471812d791340816e28ede30e9e75af624a3663b
platform/external/flatbuffers 83d6225e9b642ea069023e45a943d3e7f916fb62
-platform/external/fmtlib f4d6768e2595b881a146e41e275f77203db9bc56
+platform/external/fmtlib a12d391b4e37738115d409e737d95e3d08bf9db5
platform/external/fonttools fede58680958e96e6e8f61c3cc6282f798452c53
-platform/external/freetype 44d6b9bdc6cb90d7b0fa275dcf2be8631312ca9c
-platform/external/fsck_msdos 4103a4ab7d45b439b4f7297a2129a1ca45ec91dd
-platform/external/gemmlowp 345bf231ace3433152f7866bb5479497159bd1ec
-platform/external/giflib db6d1840bcd6c13cc41456835127d6d7d1b0694c
-platform/external/glide 31e64acd1e65045471124267a06241eff1f55d26
-platform/external/golang-protobuf c42093045dafe0406269b36c2e35b3acd7da63ae
-platform/external/google-benchmark c53db751162d02b22ac2622e1f20cabc21a69bea
-platform/external/google-breakpad 4125ccd15f21dbae3d521d0298ce90f1fbc29bdb
+platform/external/freetype b2bf79eebd71f498abe969e79b589a5a5d060f06
+platform/external/fsck_msdos b901fc09c3c8bcdc2d9dcbb7d97153df18bbb159
+platform/external/gemmlowp e6b22b7388af3fc85eca1f921aa616a7fd26792a
+platform/external/gflags 645025b30a03e12cd54f57621c6379badb655a13
+platform/external/giflib 186e796a54662df5e7774776b8d6bb6ba38f2050
+platform/external/glide ae9934f69312efed4f49cf6433be2a2511cb57de
+platform/external/golang-protobuf 4140b20f6edf5a730354e387dfba54ec786347c3
+platform/external/google-benchmark 97a79a69d13de5a705fd82ed2cf0f2a55589f7a9
+platform/external/google-breakpad 5b9dc7d25a702b4fb8377ea2953c88beb10e2e21
platform/external/google-fonts/carrois-gothic-sc 0062a10458d4c357f3082d66bcb129d11913aaae
platform/external/google-fonts/coming-soon 2c5cb418c690815545bbb0316eae5fd33b9fc859
platform/external/google-fonts/cutive-mono bce2136662854076023066602526ba299e6556b2
platform/external/google-fonts/dancing-script 7b6623bd54cee3e48ae8a4f477f616366643cc78
platform/external/google-tv-pairing-protocol c731915e80d9e2ccb755b97e7fdd280bbea07f70
-platform/external/googletest dca68eca3f6328ed43ac70b61fa7bbc5a8c1ce1d
-platform/external/gptfdisk 8c834de12a855fee178f037a36705e72fe981c67
-platform/external/guava 5d824723beff81b246b327386fdb458decfd2af5
+platform/external/googletest 6b6a5bdf2d123418d460e10ff2ab11f08406013d
+platform/external/gptfdisk 0c7af6fa274abeed5172035d897c2afc43050463
+platform/external/guava d137d8b4611318b51fa9c72186dc34f08e63f121
platform/external/guice 2f4c6dfce39d38e64a6e4993a808270e16d7dc1b
-platform/external/hamcrest 2c2a845e1e8fdda704e4f2ccaa335d69d043ebf3
-platform/external/harfbuzz_ng b8de12e1ad4706f81411d67a6a5887013a043a87
-platform/external/hyphenation-patterns ff853262e0f613e0ba2f677ae8b1f155544bb2ed
-platform/external/icu c98207ca6ad0232486b01501f1ed30017f47799c
-platform/external/ims 12c79bb691c753c069facd0097293776bd7b56cd
-platform/external/iproute2 842c7f0c1a62af00b79750de5b207c2782cbcb6a
+platform/external/hamcrest a87b7d3237de0251588e1c3147586acd7c874071
+platform/external/harfbuzz_ng 26f528b332286f3a401bc79648ba1775e85dea0c
+platform/external/hyphenation-patterns f0ce79b3e6624eff80848e16c9d96d7410b95d1d
+platform/external/icu cc5c613d9922dc1f400dc2958c8a0def3756d627
+platform/external/ims e28e150f1e399030946f49788ed0d98bb6eede18
+platform/external/iproute2 2af93e3e401d1809ad618da5374fa6d85c989faa
platform/external/ipsec-tools 16477a409310e8daa3ec5c8ecaa7a2f2d471fb93
-platform/external/iptables 8ae1dffe73609570ce73ab5430111a3d47400fb8
-platform/external/iputils 2857b8009e1dfc751ef0659c9ba7358a89f83c34
+platform/external/iptables f86d7ad9adfead8f829257f72371c799084e3199
+platform/external/iputils a90e2a2f2c97668118eb53bec73c25fe1e292973
platform/external/iw 63d8bad2a5a8a9b443595762233692c5f6a6d36b
-platform/external/jacoco 9ef5ef62baa610d324b392e06bac956b383e870b
-platform/external/jarjar 38057a61ceda3974233b5d8e48a5da00911ab8bb
-platform/external/javasqlite ee41d81872eedc632f42bd231b95bc945690d159
+platform/external/jacoco 4bce1453ad265894de72c81b2946d604c047a79f
+platform/external/jarjar 9ccadaf6553aaf0d86ec8885bfc6a8cc9bec76c1
+platform/external/javasqlite 19c2f855c28431269da38bb2135dc29694823aa9
platform/external/javassist f7c4b954072e563b75f6910c25bb689bbf38a3d1
platform/external/jcommander 501341b4efff9c378ac4003899dd064586231550
-platform/external/jdiff 915f756d47a2d2e5b1b5249ef1a05ae83d4c0ff1
-platform/external/jemalloc cd80f1f946388ed096b6ca4a11b622900683fee7
+platform/external/jdiff fe347a5d71a8d09178dab8443385a5ac0a923afe
+platform/external/jemalloc 882e73dec85a551a5df0f5bf7eb1c5bd3e90ff4e
platform/external/jline 74812032f8d8eddbef387f18c96de9e5c38b8fdb
platform/external/jmdns 0c71647deb7f7835c473fd3dfb45943083e47794
platform/external/jsilver b9b84920ba47ddf7f15baa01c56b28e3d3f951ea
-platform/external/jsmn 228d04d8c2b1c918c702a8132eaf2048aa24ff5f
-platform/external/jsoncpp 2aed88e521abb2b4e13966f73df8e753ac96f112
-platform/external/jsr305 f162f4b5d952208e4b6e0a7b2e1bf01058e429b4
+platform/external/jsmn 96a4b65b380a1bd5137bb2e9f61f62a8ee089265
+platform/external/jsoncpp a10332b83510b8bbe0cd3542495b821fb43096ba
+platform/external/jsr305 92cedd29e0a021945b4be001246bf6c091e3321f
platform/external/jsr330 d15be3f228885ce6ed2098ff2dcbf6c1085d83ba
-platform/external/junit 18fb5125096c703c1b2b83b79076a68dc6f1d743
-platform/external/junit-params 3422818d91dfc550fe93355f6345c489d962c8f8
-platform/external/kernel-headers 8357d9a3fa7a6ae0f7eda1bcd3ed69e46d2ae945
-platform/external/kmod 4082ee0d1ad5ce92c5ea6fdc5ca08e89cdd9801d
-platform/external/kotlinc a69f6332d6fb99610d163b103338ca5c2e547805
+platform/external/junit 833c14eda2691fa92267a5b5ee03180c7b8110fc
+platform/external/junit-params 1eda771728c4acf78df93ed56b110d5ef4117c9c
+platform/external/kernel-headers e9e8d7a3a97a37ccdf4607106b0ac914d4f3977a
+platform/external/kmod 7a6a120af3e347158602e87c3dca5adedbe625e9
+platform/external/kotlinc 2212d1f09d35ce51c72f5093180756ee638fe9ef
platform/external/ksoap2 9fdc4a6bb2b882bd1d6c822d2ad0c66585d66ba9
-platform/external/libavc 330821ddd93fe8266e400cad8a3d27986f97b7e2
-platform/external/libbackup 2c4cc2cc8dcd77f5f72db5c5c36db808c781dfca
-platform/external/libbrillo 548b90b26b43f5766297e4395da5bc9819a096ab
-platform/external/libcap 7be03e8484ce8ed12fd70b8f05af931253aa25c1
-platform/external/libcap-ng 835f318b9785a70415980fba96c34ee5c6b7e415
-platform/external/libchrome 54cfa4a2bacc9c43b06c1582d929da42bb165100
-platform/external/libconstrainedcrypto e2607d54fdb00903023e44a65220357ac66ffa30
-platform/external/libcups 8eacdf6145fd6b8d32ab431f867b08cf3e17e346
-platform/external/libcxx 6e22fc8c73fc5ab0af1444b0190affaa323e73eb
-platform/external/libcxxabi 324bc2a10ccd0a44fb5ab27af34c707ec83c1d79
-platform/external/libdaemon e2f604066d97431c95856c73d7b9ee46b348d37e
-platform/external/libdivsufsort 6e3b2d3dcc906a64d84ec2567de89b204d4a0c7e
-platform/external/libdrm 486358728120226bf5b48cc1ec7e14744bdaff33
+platform/external/libavc a93c0e8b1878650fa79bc35bb25ae8ed46d36193
+platform/external/libbackup 4599ef0948368d2e10f11ec69f697095e64e8e38
+platform/external/libbrillo 357c368b8fa8fff79ef9e9cc59c2b4afc485a668
+platform/external/libcap 21ebb55835948443be8480a3892e088cff560d6f
+platform/external/libcap-ng 7ee71735dfa662f78c72e514ff3029f7e8716378
+platform/external/libchrome 17dead5d3aedc684a6a73052319bf44513e998c2
+platform/external/libconstrainedcrypto 8fdf11ce575df32049ce5be752d5bee50ae11d07
+platform/external/libcups 1d9caee49b41baa96cd5180ad5556bcad1e3c14e
+platform/external/libcxx ba19c01cab035056f9a27459d908d73f9992f985
+platform/external/libcxxabi aa42493d878601250c3bfe1cb69875fc0b6cb7e3
+platform/external/libdaemon f999a5ef4dfc8b481e3874022c57a9baccbd3073
+platform/external/libdivsufsort 87d09ce8421551eb436fd08edf2e355508097a90
+platform/external/libdrm 82d838b6dc31b7087af50c80e787cc8f726b5dde
platform/external/libedit 67e14dfc833aafa400a3aad8cb329cbaec503445
-platform/external/libese 69d613eee955a7c8bb90283f2efef352ecf3bde4
-platform/external/libevent 9dfd4017adef7eaf179f743bf746254917a4fb74
-platform/external/libexif be3218e783f8cfd8545009d6b8be6ef0a06e6d1e
-platform/external/libgsm 95bc71811a7900d19723efa2dfc65eb7fc49c6d3
-platform/external/libhevc 402e431f4c31d900fff5123804744bd770d60077
-platform/external/libjpeg-turbo a89b5b13bdbfac6826e14f063002658ec9d18dfa
-platform/external/libldac 2cbcf572ff376f43f49a265e676b892ddc8c727d
-platform/external/liblzf a88b9629447deabe8697d2f8fd4cc70aa6e1b563
-platform/external/libmicrohttpd 1e68f5d827a859ba3b7ab6a70a60247e0b96afa5
-platform/external/libmojo ff6fc7f0fe90ae03e78e6cfccc4558d635d87cb8
-platform/external/libmpeg2 859f0b022bcb147180636b81a0a6a01bf9d0c86b
-platform/external/libmtp 7ed2065db3ab9851f47426f3a35ba7045ce528b2
-platform/external/libnetfilter_conntrack 9e09f14b95885b73c72533a32b4ec86f84a36428
-platform/external/libnfnetlink 1c3e0db7d701d8bed90e9a65c0d581552912f8ea
-platform/external/libnl 6eb7e2dbe99a65c6faba042c627a93b17906467b
-platform/external/libogg 5bab317e01c15cdccde1ac550aac701f436518a6
-platform/external/libopus be4939eea9e476612b0e92d8fe250236111d49ca
-platform/external/libpcap 7b7f0b469ad512b9b5c759344eb0663e7c2eb191
-platform/external/libphonenumber 6b5ff49a91b66b137b210e4869e980d083683f34
-platform/external/libpng 849b9859cf5c1ae17534d7ca6fe0ec88e85ef140
-platform/external/libtextclassifier 5e5fa7f6ed116972ebccc123c2e49427a882f4c5
-platform/external/libunwind 98996149c95c9f6fca5c2589f4664ec1a394439d
-platform/external/libunwind_llvm 4c7cfe52bb51808aac373d5dc073633da323aba6
-platform/external/libusb 723eef28307737e21348b124234c18eba2038764
+platform/external/libese 7caa0ee088b3ced3dcf789e77fa82a4a182ecf01
+platform/external/libevent 0895a49cc55794c7fc41e78c53ffa9882377dcc7
+platform/external/libexif cb8825c1ceee787a450b9dfe41a6dde7bd02edd0
+platform/external/libgsm c7ac7a2ae4d0e022d9db00a0da59e5ef6d86403a
+platform/external/libhevc d6c8637b3eb0990ef37f0db526ed0788ab9b8186
+platform/external/libjpeg-turbo fa6d695d4e286bb298ac8df7a2a4af9e4c70f4a7
+platform/external/libldac 13da357fad7e8bca22086d2e340870ebf2acc412
+platform/external/liblzf 9c7fad232e559e5cd6d21dfcfad3dc7f41240a7d
+platform/external/libmicrohttpd 856d0e7719b08239847512aa325ee85b760c5705
+platform/external/libmojo 40aea3375468790cc985f638ce71a49885c17874
+platform/external/libmpeg2 0f4d331bb1b0d06c309d3565194c1670eeeffe84
+platform/external/libmtp 70b080a741fc401f24b0432fff1f95c834311cda
+platform/external/libnetfilter_conntrack f62edd6e51ef44358c7a7d42cde29a4aaa1bbeda
+platform/external/libnfnetlink b6efdfdd3129067e956611de19476dba9a9685e6
+platform/external/libnl 6cc1308f54de1982dbfb88269fc2f4d7175966f9
+platform/external/libogg 5e34e696f26ac67d54950371e59c4e68e10bb5cf
+platform/external/libopus 93264cbc8ca69346b3078abe5eb657e6169854b2
+platform/external/libpcap b6668cec55b8ca9e69b40c9a22fd1cd07c2e548e
+platform/external/libphonenumber 7d0c1e1535448a901f435c67d4e716710ea912cd
+platform/external/libpng adebdf2260417081d101d032214f599ef93a5fee
+platform/external/libtextclassifier 98d9452bb8072ec0fef49e38c45646981e6540aa
+platform/external/libunwind 64403ab46fe704393e9f0f0185822eaf75aee0ed
+platform/external/libunwind_llvm 98b55b853bbb34882b915366afe1012091b614a8
+platform/external/libusb 8008b1867cd3207e2c7175c876a81a1f7490e522
platform/external/libusb-compat 759481ae400d02fe99488dcdcd653b4f8139a39c
-platform/external/libvncserver ed7c226e0c0c22704da70c2cfe85d8a0043a3ca0
-platform/external/libvorbis dee2e0a12822ec62c89c2d7582f1a7d32cd0875b
-platform/external/libvpx 811c563cd84877f5cab826756d977be33789c96e
-platform/external/libvterm a70e1c33cb5e1f298de2affa000950123656c4f6
-platform/external/libxml2 805910f172353ae20bbcc687097bb8b448302fe4
-platform/external/libyuv c802d97f013cc20af2d0bb6a4d2473a16347552e
-platform/external/linux-kselftest 99082400b64b9aba0efbf7c7a7587a70f836d9a7
+platform/external/libvncserver 6f5f6638329cf031a8e37bc05aa4823c18b0729a
+platform/external/libvorbis e623471c48f2dc653fead4350f954e5b89f3c8a3
+platform/external/libvpx e9c5a1ba6f57a51ac2d6285f10bc1084c3a787a8
+platform/external/libvterm 4687070036d46da043685b726a10d7feb8f20fb3
+platform/external/libxml2 908b56f927d5efbd664f50748643236e1ed8a0d9
+platform/external/libyuv f296afbce6baaa59683ee27958515ede65487ec5
+platform/external/linux-kselftest 99162db134f5dde93f26347814556cb3620d27d1
+platform/external/lisa 58e57434aa253491f81868e2bb846245046836c4
platform/external/littlemock 4dd43c74801149ded3e18eaba7d43ff6dfa192c8
platform/external/lld 26c9bb3b51a7ac4bc45f73d532a03cfd9982043a
-platform/external/llvm 839c1277f49a1e5c5fa633e96cadc170803ae828
-platform/external/ltp c68c7da9634d5e1d5d47f1b94810184a051d4da4
-platform/external/lz4 d49376538eb85f252057b36374d0d61328c1e910
-platform/external/lzma 45d1ce03842801b212647d177d9a6c21726b80af
+platform/external/llvm 6e6821471f158f7c1300af9f418251b39d02f4d8
+platform/external/lmfit 22828a210c111f351709da4542f10f22d8e54408
+platform/external/ltp 92d53594196d56d3ec457e5762c0a9c6148712da
+platform/external/lz4 2332741e72e1eb934beda8139b8580cd9fc8ea78
+platform/external/lzma ac40f2d015443c24e6e8eadedc3e8d7c64ad4043
platform/external/markdown 06a8f8b914d477183f68b84424bce8ff4dae7e84
-platform/external/mdnsresponder a72480b05ef9b5f933a1c76cffc5d3591b3f781b
-platform/external/mesa3d 34a10fc4f3202536d6557921d4b3c7f3f59563c9
+platform/external/mdnsresponder d49103b84a0dfe8a083d42a70b05a47c94a76bbb
+platform/external/mesa3d 239c51fdbcd1f7e8336d6072ad2098045ee9d641
platform/external/messageformat 7941896999a8a623683b888fd39095f32a7f49b6
-platform/external/minijail 39aecb1614d2656ef385cc19154efaaa845e866e
-platform/external/mksh 8547d5a6a98fa6540ba5c4b6e358cc341451201c
+platform/external/minijail d775f4c8559d829a897899e5be857661fdbbcf84
+platform/external/mksh 46fbc9c70135db1847558e4c0e0a26a477986e5e
platform/external/mmc-utils f105db5bd1934f0c3e5f74bbc9a5aaeac7a6bf2b
platform/external/mockftpserver a98e55fca4d6aa55549f8c2c80034f2d04118cf6
-platform/external/mockito ddc7186be0a984376dccb13ab3850787abe8e821
+platform/external/mockito 56daa9c3caef0cc3ead27a85fe50997743ecc415
platform/external/mockwebserver 2440623105409d48ec0a3717bb6f3da711a0f15d
-platform/external/modp_b64 3df5466891f819567a709d03f08ac340bd678c48
+platform/external/modp_b64 6f11b5c9b4ffb38dc9375f3cf344da4300a2741a
platform/external/mp4parser 88bd0c6cfa7fcfbbebcc15c2c565f714cb36b065
-platform/external/mtpd cf7afdb28ea47ebe8a6b298a03cf89ba06ca333d
+platform/external/mtpd 8dd3de4c6b4f55f42095a20dce0b746f1f551e59
platform/external/nanohttpd c6783c32331b3fe78ffd5077e7f180995d0e268e
-platform/external/nanopb-c 3c0841c913e25ea59a52a17a1317d86dcb0dc158
+platform/external/nanopb-c 9731ff6d2b2c4f193598827406c0bf13553bbd6e
platform/external/naver-fonts 91e6e9f94d1d769a8f742649674149ba98ce7d45
-platform/external/netcat b023a43765b15f0b0fd5b52b7d8021f515c59c23
-platform/external/netperf e100a0c4105b94f7044b243287a5da22f5e8e4e7
-platform/external/neven f03656c6632b57b4c52705847a7bcc96437e2ac9
-platform/external/nfacct 5df745ccb4ef5356a786bf63e075efd0ccaad333
+platform/external/netcat 127af3e66e635df446011edb1b001c6be8951836
+platform/external/netperf 5cb77b2e10572f947cfe0c9da8fe5bcabcab490b
+platform/external/neven caf2eef6111d83d17b89f348f8f710cec0d35223
+platform/external/nfacct e3c0e3cb33d5e13994cb1404e27a7bf2d2564c00
platform/external/nist-pkits fe530ae2c3d88eb5326151387028507b4949d751
-platform/external/nist-sip 8445067c3b53e5bd56c32f1c4fb688083c4f4005
-platform/external/noto-fonts c1172c51af9734ce03c0495c3a0a658590f51487
+platform/external/nist-sip a6844ac6224619a7211ff6fce60e756349095170
+platform/external/noto-fonts 1bf56f26dc01171eb1a741800a6b7842aac7a96d
platform/external/oauth 49f3624a6d3307b640a012f15b94d04174473501
-platform/external/objenesis fa27358374a5375586ac703a514860563323d94d
-platform/external/okhttp f3436b166e01056d7b66abaad5b2e553338e0e3b
-platform/external/opencv 507669d9a3c8adb2b415488783efe10bf9a6b3a3
+platform/external/objenesis e5e9f043e4fb9b1e16ccce07f0c7d0896326bae7
+platform/external/okhttp ad46d06f88db38f40dcf36c5133bc63126f2a0f6
+platform/external/opencv ba9cf7de2d6891b012e78a12c79f6022f25bf735
platform/external/owasp/sanitizer bbfb25464ff30c5a62dce351d719a8c533afb2a3
-platform/external/parameter-framework 3b04336f88873ec4b63717035c7a30d76da89a7c
-platform/external/pcre c3f1349fc07b4a0ac8f33b0ef20d72db1668b206
-platform/external/pdfium e56ff935bcb3a395a841c069ef98c0077376d8b3
-platform/external/piex 72d810c7226d9cf1d98c211038df8fac5ee469e6
+platform/external/parameter-framework f51c185f94076080fd0161be20804c34a8eea976
+platform/external/pcre 0215a7f02a317946649cf748452e972b382a19af
+platform/external/pdfium c0e90549084a2f4b36ccebb397f941ce32badbe7
+platform/external/perfetto 46c66db013d605c7fcd1bec36b528564454efce1
+platform/external/piex 75fe364a1653ba6f5fb016c818aa6a7abc80e967
+platform/external/ply 5734a9be97a53f0290935d0f4a54ca687462b28b
platform/external/ppp 90da8c8f15db2c82aece6e4085aaf9c116d481f2
platform/external/proguard af893cdc1242e4cfbeeb10e2323f9a689a825177
-platform/external/protobuf a518d2ff2dd91bb87a17d6fda7235a30b6fb064c
-platform/external/python/cpython2 35adbed7df90bd6d6af50dbd90ad95425a5ea3f8
-platform/external/python/cpython3 9458f8e38d4fe0d58890c5867485dbbf2d538955
-platform/external/r8 46e2aa62eafa8d8b3056b144e3b06bcaa8d00a9b
-platform/external/regex-re2 79cce43a82abc1bc56c65de07a7df47d54e163a9
+platform/external/protobuf bcc5a4247ca6181e823dd708462e23e942dcac34
+platform/external/python/cpython2 48d1b2352be3d073be9585f48cf4732d5d354668
+platform/external/python/cpython3 02d71e78adfa0858889a2f0b84adb98dbe3e6fa0
+platform/external/r8 6a6117560d9f9bf6c0c82c1205f539d22adf7f8d
+platform/external/regex-re2 f76f017f28f77908d92e3001929dbea589dab2e3
platform/external/replicaisland 7d9b8c6fd062bc37ee2265010eed37786b76a324
-platform/external/rmi4utils d45cae258a3d831cd035ccecb3daa763d26346df
-platform/external/robolectric c35e9c5a25433d1e4beb65993bf8a326916fb2cb
-platform/external/roboto-fonts 0fcc0daf8155d1f44eea124faa7309b01c4fbaaf
-platform/external/rootdev ac440d6d708934c2b5df3867a2d43a275016c9ce
-platform/external/safe-iop cd76f998688d145235de78ecd5b340d0eac9239d
-platform/external/scrypt 581021037c04fb2cd8dcbfd643cd9b9545f6a2b2
-platform/external/seccomp-tests c5c3309739e1178676139c0751f8ebdc8bcfca26
-platform/external/selinux 252a2d8491ae39b9f9d2413ef5a4aa2d93c73300
-platform/external/sfntly 313d3399e181bb1eb213cde006896c5ed253045c
+platform/external/rmi4utils 36df17c8118133b4b1da5ea4a7d0f802f9bfccfd
+platform/external/robolectric ce5e27d3a738886ef6e4e13601a1ca12d8abe722
+platform/external/roboto-fonts a49bcc0972b3772872e36ca06817e950e26593bf
+platform/external/rootdev 857775eb362e821563f0733babb84ad0316ff3eb
+platform/external/safe-iop b9863012c0e03e1dd8a2c6d1cbd511676f93ed35
+platform/external/scrypt d95a0100c0df1e671008483b244cabc74300b487
+platform/external/seccomp-tests a4e66cc6d6dbf932db378f09abfb7bf6f198adaf
+platform/external/selinux adec3cd8b37ffef548ca8ca9056f6cb2d74f916b
+platform/external/sfntly 0bad802dc17fc283171923479960d6dab6342d3c
platform/external/shaderc/spirv-headers b252c470366118af0db2b4395217bedffb4c2e01
platform/external/shflags c4876e01829b8cf110ee33267bb1bad1f8ebb51d
-platform/external/skia d9ead8a8b93aac3131f453526865beaff2251ac3
-platform/external/sl4a 097b3146c9889a1175c738f2449b820a2d8b82fe
+platform/external/skia 96119fead1326ec68be2832b1fbc009b347ff44a
+platform/external/sl4a 65e7cc8cd4819f25dcfd9b00e65126ef76ab87d1
platform/external/slf4j 63b140791fbb0bceabc496bdd29e108d9597b044
-platform/external/smali 8bfa6a16459b84c945728bc197f01b46ccde41ed
-platform/external/snakeyaml 462c5b2ece22a176d8f7ce81e7321a877e891815
-platform/external/sonic a2f81e8e1f8c67a888b0e56dfc2f07e7e2030575
-platform/external/sonivox 4c1426c96fd8848540aa903069b04afd29364ca0
-platform/external/speex e5e09f60ac161b4d67ff555b359201cf923790ea
-platform/external/spirv-llvm 2c95b1ab3064e01b4a9acc7f19fe43097ec704bb
-platform/external/sqlite ed6a0cb1f19290ffc824f4ccfcaeffe1effae47c
-platform/external/squashfs-tools 8c214c801b59ca22b66a0f5cbb3c61af39311fc3
-platform/external/strace c130cbdaa4031a102193fa26b232b6344cc6f16a
+platform/external/smali c7b02ead65f98e1dc574a3c326bba1d2076ca30e
+platform/external/snakeyaml c2d7be0abdbdfae49db372518b7d460f5ac02b5f
+platform/external/sonic f1db1383385ac1ffd88e89fa5506d0e21fa80944
+platform/external/sonivox b74b3f259117b70a962bfdd8713c5e10a5ec1f5d
+platform/external/speex 39a2e16cf342222ea34ca77c208de7b8c7c91395
+platform/external/spirv-llvm df754c85256efe467743002f54c4e96b3d54e089
+platform/external/sqlite ebd1125140e3b96b89b3f0287c190285daef0eec
+platform/external/squashfs-tools de36999a73617500d83379e36c23ae4cc2b12a7e
+platform/external/strace d0279047c4d6eb6d276f4c17481f750ac2eeb89a
platform/external/stressapptest ea469443e9c2cd1fda87188df2b1ffe0e30656b2
-platform/external/svox 1416c244af01c9b4b1952bae1b84c6a347a68b2f
-platform/external/swiftshader 0781219b0b2e38ab4ee4dfe24b038052cefcd61e
+platform/external/swiftshader 2531a2f0512ca533e50e38703e40106ad5e2f3a7
platform/external/syslinux 76d05dc695b06c4e987bb8078f78032441e1430c
-platform/external/tagsoup 9c02d9f506855965ec513685788890dfc856a5bc
-platform/external/tcpdump 5c80f87b56474c42c598880611a29ae013bef07a
+platform/external/tagsoup 877497e7b01037009e131866683ca1a6650e3ced
+platform/external/tcpdump a9b933680b81642b4498c2db1d91f7ea4982d7f6
platform/external/testng 2d09ec828c313edd6f9f5021c9b166854d742521
platform/external/timezonepicker-support 99e91a76fd74bad10266623d67cdb98d011f709e
-platform/external/tinyalsa e88d49713693d9380209698130292315c82da37a
-platform/external/tinycompress 846573ce25e334e2709b3b4d7c5ab39d0a2b3bec
-platform/external/tinyxml 4949b8a792682f59a573fa87ec20b14e850af273
-platform/external/tinyxml2 d4164674daae9f26a9eb5d72dfaa0f0da0f1f3ae
-platform/external/toolchain-utils 18caef1edd870ff3850fe2c417bbcec00c2ba7e0
-platform/external/toybox 11385503eeac5298a56fc6cdd8e222be1917559a
-platform/external/tpm2 0cd05a180254e631b783b3100beecd90bb93fea9
-platform/external/tremolo 38127ca33ab9436168c8595ee02233b1dc694a71
-platform/external/turbine 7e17302c6479d49c477c51c2878a00549e56e618
-platform/external/unicode aa5100a5dfae7ac21df1ac13749faa55cf9f48df
+platform/external/tinyalsa 301705c36a95a8575a689ebdbe60ce490458b738
+platform/external/tinycompress 5e02fdc8ebb10c5b0aa70ccde2dc823ccbd3bf88
+platform/external/tinyxml c1b9f4cb1286dd01048d90e16e81670a817ed8c7
+platform/external/tinyxml2 b28b1628389ef48d3186a6ff1f2c9b7edb9b6c61
+platform/external/toolchain-utils ddfea1f7e75062a350bd5a9418562e3b5af5b6e9
+platform/external/toybox cf77ae240a1d3797c535c4f7ba000cd008394618
+platform/external/tpm2 08c43b205383fe7ed23f53ac7d15f545716bc236
+platform/external/trappy e3e1107fc64daad1e5950479951b8627cb2251e8
+platform/external/tremolo ac70b75ffcfc4cd4e3b525735e4ed1c5841195d1
+platform/external/turbine 98b82cc095a22973eb0fd53deaac3a17362f5ac6
+platform/external/unicode 1d15b7695581bf8fca39b4486615b014e7b72ce6
platform/external/universal-tween-engine efe8193526dea3b430aab4cf6d9a6c3d280ea5d3
-platform/external/v8 37a3083841acc8d69e78889991dd4e6092f77445
-platform/external/valgrind 523778cf4342740cca740f6756a204b61bf268b1
-platform/external/vboot_reference 936bcaa501354e5ea008aa3088fd3852b49e418f
-platform/external/vixl fce3db7edbd1d6e619e9fd66f1ac020d20cedd57
-platform/external/vogar 77441876b34ccbb57208a55ed35a9ca3a0bf1ad8
-platform/external/volley 8b42d211a58fc5b434921a51cae2c99a806e00eb
-platform/external/vulkan-validation-layers 727ecf10b5c219bf14c82bd1e58d856861f0cb86
-platform/external/webp 75f26557017b2859b054802100a334552d3497dc
-platform/external/webrtc 60c71f62eacd925c491863be665dfea23331a6f5
-platform/external/wpa_supplicant_8 f7bcce468f1c1d8b2ba61744c305b29f46ce34e9
+platform/external/v4l2_codec2 ab7829b032adc58c00f3f62ae1481f7f3c2dd272
+platform/external/v8 25eea94176b419fc0efee84a0ab2b2b05f037448
+platform/external/valgrind 20da257d3834dfcc1e9477744b803cc957c3efe8
+platform/external/vboot_reference c54978e619b13e4cf6de2c922284b82a6141b0ab
+platform/external/vixl 498ccef20c806a86234b8dce398e0f9e1c2b7953
+platform/external/vogar 4f04889b350a9576fe78f53986cc178e98bde219
+platform/external/volley fa08a1a64c39d7a4b8699912603e6580e00ba71d
+platform/external/vulkan-validation-layers 8f560e426198082b39ad0953480367df8e0b40ff
+platform/external/walt a67c00187b247fbcfddf1ecee255828e9800943e
+platform/external/webp 3b38baff9f66ff8725e54e72f29f06988b1bb84d
+platform/external/webrtc c333849418e52f0b07572738f0baecb81f642435
+platform/external/wpa_supplicant_8 8ed76874ba9ac67bfc83d30dd858a11395ad2112
platform/external/wycheproof 46d28ac44a242d84d610a294a640266d3b0e033b
platform/external/x264 7bf0b975cdfd5692a4598ab09e1258abe5ea5fff
platform/external/xmlrpcpp c79d4eae757a81270915c2a7cf01c81b8d42785e
platform/external/xmlwriter e95d92246ee35273dde2bee8b00485cc14c12be5
platform/external/xmp_toolkit 42ea4dc6d1fc2206a7778029070ed9213e3b0fbf
-platform/external/xz-embedded eb4f490242a36248365747f532c12b6621009210
-platform/external/zlib 6b637b605dfb9cbd6bf77e0f86d28867fa08ee58
-platform/external/zopfli 069cb023c39358bc0e1b3dc0397b8fcdd84b4d44
+platform/external/xz-embedded 056e99d35531918b777aff9a3eb7a68b045844ae
+platform/external/zlib 80092b6bba524246abdaaddc5a9b77f098a10a78
+platform/external/zopfli 7f0667c45a5289e9ce6fc07d681d356345546958
platform/external/zxing fedf8f2d8099bf7bb85dc3db8699343d56617deb
-platform/frameworks/av 7fa5bce9882698c578141fe48984c93f561a9caa
-platform/frameworks/base cc3525748e4b1317afb1c396587207867e95f842
-platform/frameworks/compile/libbcc 86a5f1143277a1b24ef580072ee80863248a9281
-platform/frameworks/compile/mclinker 8b0d49f3aa14a959ae1ea886c1934c9024cdbd5b
-platform/frameworks/compile/slang 4c0eb48014c6ff64e427f179f4873563ef1b4222
-platform/frameworks/data-binding 0bb88fe21d26e675ceaaf455b46e74bbef5f6e03
-platform/frameworks/ex 77dfa2a9b4c5acafa45d7f09bd95dc44da54fc4d
-platform/frameworks/hardware/interfaces ecef1a8347e85af5a3fe4728e7b17dbec86db8eb
-platform/frameworks/minikin 26886917cc4b1e7b5daa1de182adaeb56ceb4a14
-platform/frameworks/ml 00cee902ec6ac3d80f3dab80097dbf13e0a475d9
-platform/frameworks/multidex afb74c2dfc474e50b2a0d3e7fb0ccba747681953
-platform/frameworks/native 7e7457f225ab3c4778a46b6d718acad69d305d4e
+platform/frameworks/av c2a068aa3bd9f7286c96c5998e59520e26102350
+platform/frameworks/base 9009eabbfefdc685f176419001bc2aeff4790089
+platform/frameworks/compile/libbcc 4855f0b2e8103b7fd3baad9b09574b5e8bd77968
+platform/frameworks/compile/mclinker eb76075156324051c23b6d47fd984921d146fcc3
+platform/frameworks/compile/slang 9f03d2447a458fe0ed2958a49f290eb15bf759fc
+platform/frameworks/data-binding a53fdf4a28693a8ebbc5d3fef1e44423a560dc90
+platform/frameworks/ex 9ac18aa601e927ecfa3948836b388844082971b5
+platform/frameworks/hardware/interfaces fdea7efd2c82d2b569833fc90dd0def376e10aae
+platform/frameworks/layoutlib 2546a8559cd45130bed820eabc02c14e0995e37e
+platform/frameworks/minikin c995aba0d1fcdd54a92faa81befc18588653f8ff
+platform/frameworks/ml ffaf7cd39aed2328e97bed76bb8e280c1ab787c8
+platform/frameworks/multidex ca11166173d7509b64ca051d73705c38fa763bd9
+platform/frameworks/native 263921f53cdddc6becac4bfa07e3fb27764adc12
platform/frameworks/opt/bitmap a0d4e3108663202564a6833b76770075b8e5b767
-platform/frameworks/opt/bluetooth 9de9c5d49aaed7083ae886df6c61750529e51751
+platform/frameworks/opt/bluetooth f6011e13ba30a5e32f6c2c76e300c622a329756e
platform/frameworks/opt/calendar 3f01563afad6b3d2f0546a6260ef958a8173deb6
-platform/frameworks/opt/chips 27b256c8555b93cc69ca962051a8fe12b769efdc
-platform/frameworks/opt/colorpicker 578f24e997333479fe5ff697d65e994033a62679
-platform/frameworks/opt/datetimepicker 433cb3227c83234b2c58f3a8cf320bb441163daf
-platform/frameworks/opt/emoji 92eede13edbf22b501edb9aeb92366f91eab9781
+platform/frameworks/opt/chips 6804c987e2f2ee243bd7b21dc5665fb83c98ffe4
+platform/frameworks/opt/colorpicker 6519ee31643ef95bb6f9204e817a7ab6484d6ecf
+platform/frameworks/opt/datetimepicker 74631d4581fe01cfcbc38f63740d165f388d5044
platform/frameworks/opt/inputconnectioncommon 3baece9b20fa480da46d860acd7320fd9eee3386
platform/frameworks/opt/inputmethodcommon 990fcb1a6dbb5d1204cc8ec86e4bc3f691f4aeeb
-platform/frameworks/opt/net/ethernet 230c0aae03841b1147e75bbd68dcc4feacded6f6
-platform/frameworks/opt/net/ims e511a2031ecb8b5986b7db5548b58d0648b59209
-platform/frameworks/opt/net/voip 07e4fc6758c6c8053dafcbd1bbc03b163647081d
-platform/frameworks/opt/net/wifi e436be523fdfaca40084c68d729f74e97047a747
-platform/frameworks/opt/photoviewer ca71bde33a57b9c0384fd185aabe41653806666f
-platform/frameworks/opt/setupwizard eaabb4f5914ace1a775675acc4f6d73a3ac5f9a2
-platform/frameworks/opt/telephony 39b22f4a3d02af539dcfe9daf16370fb2ab39fae
-platform/frameworks/opt/timezonepicker 34864043247588c0a1cceabf127fcf009ee74a9f
-platform/frameworks/opt/vcard 8ed59aea836a09915c3d8cbf639627ad4f8120ac
-platform/frameworks/rs b8629f45c928d9dffe1ba7f2b7845895f26931d1
-platform/frameworks/support 6e67840e5cd3cad180863b666d369b4a47182a65
-platform/frameworks/webview f205dee72d0bacf11b373c1bdaa4dcb049f9bdea
-platform/frameworks/wilhelm 29862d783a985bc0c46de45253010b8ce4e0ea7d
-platform/hardware/akm 99e8b227cd00678ce863c04ebde0b8d9a6f7df5c
-platform/hardware/broadcom/libbt 50b90f45144aacb1f67c9c18993be9aba82bc747
-platform/hardware/broadcom/wlan 84029467a8866bf82e413310d11f78e1cc3fed4e
-platform/hardware/google/apf 063c56133a805ce2e6ee65c35c30b76cfd25d997
+platform/frameworks/opt/net/ethernet 7fdb2fd4499a733926d14e30602212394130eabe
+platform/frameworks/opt/net/ims 29ec0fb19061f2fd1daec19aa38b106626d201ed
+platform/frameworks/opt/net/lowpan 7bf3ff0542b52af0693d7c093159bcd8691a10a5
+platform/frameworks/opt/net/voip 7e2aff9fdb02d4b90203bb0456defd93a0b39e8c
+platform/frameworks/opt/net/wifi 3847f15a7ca90260e0017aec743106f36bfaaef4
+platform/frameworks/opt/photoviewer 23b3b42b15df4be5631bd37964e5094911b1b8a1
+platform/frameworks/opt/setupwizard 1ab21582adb7e4c6b55aa7a818b96e8d74dc45b6
+platform/frameworks/opt/telephony 4b08927aeddcc8fbc87245cb6e3b52138a309d2b
+platform/frameworks/opt/timezonepicker d18b9e83709ee2957b9c6da317f6254b9ffcc0e1
+platform/frameworks/opt/vcard d90a30beaf9c1e3b7a9633bb32a49f26fdb3b1e2
+platform/frameworks/rs 1a2c80c04d8f4f01db1d70347590072b05181ca9
+platform/frameworks/support 7f813ee17e68748ca59801956fb6f79a00a54094
+platform/frameworks/webview 091c161d12cd3eb0bc79b82fcb884380cedea435
+platform/frameworks/wilhelm f105e3465e72ffe48a3cdd1ac466853821123a4e
+platform/hardware/akm 2704929e0021651e0a54c629b36ca4ae6cc0e381
+platform/hardware/broadcom/libbt 16447ea18eab3ea0223f206d11c97357b3d6ca6c
+platform/hardware/broadcom/wlan ce868768b2f37e0d0b8df564b8ad6294a2245ae6
+platform/hardware/google/apf 1bc3e766dd99eede2d6b0b0ce6a74153b197b361
+platform/hardware/google/easel 2b2bfd44f1ef4ef88091bef5c7bbee843e7a1eb5
platform/hardware/intel/audio_media 0e60b77af3fa70393ee9fa0e80811d1b39c5f6fc
platform/hardware/intel/bootstub c759e5127aa582ac515ee1446da15f601b15a99a
platform/hardware/intel/common/bd_prov 8af329f2d2b54dfcfa84051d3ce1fae95f79011a
-platform/hardware/intel/common/libmix 550aee8200015a67e32a5eb9d8d1f438f2028331
+platform/hardware/intel/common/libmix 15d847b9d52ca42ea90f016db93d795704de2da7
platform/hardware/intel/common/libstagefrighthw bfc9bda74643eedd4414e996d2a089a36d9acb24
-platform/hardware/intel/common/libva 2901b4bfb210c1c644e44f214fe7a3b825608da7
-platform/hardware/intel/common/libwsbm 8cae226f534a504f385084d1b04f4af466a16990
-platform/hardware/intel/common/omx-components 34bb282eb4b4cf0bdb9cfdeb85d59f86e220e85f
-platform/hardware/intel/common/utils 8712cbc7851511b4aa61d1544078c8a9521578b1
-platform/hardware/intel/common/wrs_omxil_core 8309367a3ad4446a1e2f07779955bc043ade8deb
-platform/hardware/intel/img/hwcomposer ad2a78606b8c3fc85667f26cc46d5f4291badd55
-platform/hardware/intel/img/psb_headers 333777baeae5d75efc1afa6719302ac5f93723dd
-platform/hardware/intel/img/psb_video 35ffe308d254bbc6213929935439510c3bd252fa
+platform/hardware/intel/common/libva a49a95403f6b214f91fb9c6251dd7440d5069d48
+platform/hardware/intel/common/libwsbm a5760c68f2880c7451d4fc84b629c9142fe65dae
+platform/hardware/intel/common/omx-components 78abbca377701d4e705ad5347052b521bdb972ff
+platform/hardware/intel/common/utils 4f71df5045b817c88585ae55d8dc038a4346859d
+platform/hardware/intel/common/wrs_omxil_core 689411deff0f97b7904e86a371a92dfd2605177c
+platform/hardware/intel/img/hwcomposer 44c0ce2d4ba48dde6f9117566fd6811aa705ddea
+platform/hardware/intel/img/psb_headers f36ef0a5ca5f9c3546a58e053a4fc45aa3b39efe
+platform/hardware/intel/img/psb_video decfd6003f2188fcbb958f4963463e23efe0cbf6
platform/hardware/intel/sensors 68dc9e70b79dacddc4e0bf00af0de7f764b04eed
-platform/hardware/interfaces a3b0f1f43142d204e4b3fe4d8c2aaf1caf4eafe2
+platform/hardware/interfaces cd769859090b02d23d01990a73d300ccaf026fe3
platform/hardware/invensense 11e5ff75af866f91622b6008fa13db1c3685ae69
-platform/hardware/libhardware 0b2872c78458e142ee7fc441243afa7903178c27
-platform/hardware/libhardware_legacy 78c1981b8aef93deed43552325423f260baa5763
+platform/hardware/libhardware 102a326b73bcf302f5db1b3f9105ba3afd7e0be4
+platform/hardware/libhardware_legacy a35e4c72b10e2c1033bda2bcd903da53d260b6c5
platform/hardware/marvell/bt 3f33d194e8300816b94d1d7b68b1d48c8f903251
-platform/hardware/qcom/audio 545beebc5add1dfd95dd9f6ca1539cd38920692c
-platform/hardware/qcom/bootctrl 483ce31f1ef6f45447445825abb958984c306f45
-platform/hardware/qcom/bt d7d7b9e56781fd49566e0b5e4097a6d75c52cbf5
-platform/hardware/qcom/data/ipacfg-mgr 5b5af67d19baf862f5a8d6a7226feb630ec2f81a
-platform/hardware/qcom/display 0a0cfb1377fde5a0bc5be567f27e3c3959e96417
-platform/hardware/qcom/gps d8e08afc8e338db5e82cb04a90fb707894213945
-platform/hardware/qcom/keymaster 8edff26b027f2435f78aa77bca30a258e0ee9d43
-platform/hardware/qcom/media ad8d9a435b4fc96ce022709d47d9275c6db6ab4c
+platform/hardware/qcom/audio 0b85566fb532493c2bdc060cda3f13f30d8a3b0e
+platform/hardware/qcom/bootctrl 93a61da1e6caf7ed7c6ba935ead48f4f7fde4382
+platform/hardware/qcom/bt 12a198def56b4ed3769507d0fc097b06e3e724bd
+platform/hardware/qcom/camera d5d9909cd6fec0941267efea77c32b1402ac17f2
+platform/hardware/qcom/data/ipacfg-mgr c36f9def45ec0b098a6fb2ffcaf196dbf7f162d0
+platform/hardware/qcom/display 21e862dc38dd98fdb8547b30e19a16fec1268b86
+platform/hardware/qcom/gps ae0a492ed6629b6449b2ee4ca882645935c860b6
+platform/hardware/qcom/keymaster 1594c64b696692a647a46960b755da5ace26483d
+platform/hardware/qcom/media 52dc04724d6ea83e2dc2fe709f98ce3c6cafc3c6
platform/hardware/qcom/msm8960 c25a431842a26b5756b58a9d4a42c776e0457ba2
-platform/hardware/qcom/msm8994 76e6a531822718a6a9e8019e8343cd398492815e
-platform/hardware/qcom/msm8996 f81ef30068fb7ef1dfb772ffe69b5cd846e81d9f
-platform/hardware/qcom/msm8x09 cee67f33014a0d25caa53a515bf2da11225880a9
+platform/hardware/qcom/msm8994 276ba655d67d883ff0c71059e6896cbeaa4a45fe
+platform/hardware/qcom/msm8996 e6e6be1501f8b34dedbddb0d43d957040bb846e2
+platform/hardware/qcom/msm8998 e00e30c98ea591786b8cf2e7a778179c83d95bd7
+platform/hardware/qcom/msm8x09 9903d4e5f088248da507a42db251395fb6cc76b0
platform/hardware/qcom/msm8x26 85c1a5282ae28663335e55ce96a4c0487de6c578
platform/hardware/qcom/msm8x27 8ff5c0057cbdecfa09410c1710ba043e191a2862
platform/hardware/qcom/msm8x84 582b414269d8472d17eef65d8a8965aa8105042f
platform/hardware/qcom/power 3a098ee1f89c398b9d6e7b5dfae9c694994f8bc4
-platform/hardware/qcom/wlan 6c71dbce7a5b93982abc30d2bf4245247cb34efe
-platform/hardware/ril 365e630c77170b55feb10f1ccab4bff73869a902
-platform/libcore 7204dc588d686510080b8690075fd94938b7526f
-platform/libnativehelper 0c0ddf128bc3a27e7c3f09bcdd425d1fc01c15bd
-platform/manifest 31babfb21d81098c3c3e60008074db7aef8b91f3
-platform/packages/apps/BasicSmsReceiver d039943180c9cb8210d66d5e6bddfbf26ab70e17
-platform/packages/apps/Bluetooth 652e345ac44922efcbb7728a3495cac00e37ace3
-platform/packages/apps/Browser2 9180bc72bcaa6d28156bc65ad2caf62aa50fd1ac
+platform/hardware/qcom/wlan 67794f2bccb6e1310bd17934807020d866f05730
+platform/hardware/ril 9e633f39569b26a95a1c289e5a6dac9d39ce477f
+platform/libcore d7e5debc32e10bfb4218af940f2128ee02f472f9
+platform/libnativehelper 6f8f41256dadb1b7daa5fa635d2384d1c519fbe8
+platform/manifest 98ee9991e285a73927f74468651523361f5de54d
+platform/packages/apps/BasicSmsReceiver ae385481d9c91b8ab7ae0ca98992c37b70aa6499
+platform/packages/apps/Bluetooth 873fa35235496aaca39c9fee2aff4c3ef4d29eca
+platform/packages/apps/Browser2 ce9866578e8dc4b47f034ed592b5947781943d1c
platform/packages/apps/Calendar 956525423e51315015d6902e3b9487e72e9f4885
-platform/packages/apps/Camera2 a4cb3c1c2d6cfa34a462bb04b38dca2eab7a62d7
-platform/packages/apps/Car/Dialer 732aaff8cb379728ea0359ea8cf913078d3534c8
-platform/packages/apps/Car/Hvac 58772162c56dc8f893855e9fc27edc34a8f2bcac
-platform/packages/apps/Car/LatinIME 22e301a603d844c0ef92e9cd95c721d6095ca19b
-platform/packages/apps/Car/LensPicker d509b2dfcbd4a51d28c496bd2f3fd8f40f42528d
-platform/packages/apps/Car/LocalMediaPlayer 9305871fc0013c451f33e56229507be0a1e120c6
-platform/packages/apps/Car/Media 5cb58af61618d6c457685a5feba630540f25567c
-platform/packages/apps/Car/Messenger 22bc7d8f0b1a18737e951ff3352c089670cc7ab3
-platform/packages/apps/Car/Overview f6b3e8ba62d070b0870eb9b0e70aa34bb7c363a9
-platform/packages/apps/Car/Radio f5bdd9d38dd23ff55fd0fc7f7209d1ad28705f87
-platform/packages/apps/Car/Settings d19f034f4211ce0d55bb38936e39c60b13bc509e
+platform/packages/apps/Camera2 867c2a333ba82850951355504f92b98289c730f7
+platform/packages/apps/Car/Dialer 17d5ac7d0b429687765e940279726623f5f5f823
+platform/packages/apps/Car/Hvac 1ad846865c2fac8302c1ef9ff550f036aa677940
+platform/packages/apps/Car/LatinIME 0ee01641eaec3a7240cca0f41be9dcf50deb4a8f
+platform/packages/apps/Car/LensPicker b927e499771e5514ae8de5fea9d2b4bb8f31b0d2
+platform/packages/apps/Car/LocalMediaPlayer 538c69661f30d126b6f95fe82713da429ac148b1
+platform/packages/apps/Car/Media b435de51965d13509bd5a1a8e2a127470c0aaa11
+platform/packages/apps/Car/Messenger 917af90df473a739aca62bc0404fa2f25977307d
+platform/packages/apps/Car/Overview 33de93c4f02642d9d06de398f0eae219d1586022
+platform/packages/apps/Car/Radio 0d77d3ea86aaeb583ae8edcb1337f8eaf73e013a
+platform/packages/apps/Car/Settings 19177609f2b9e9dc67cc3d9d9068c64979e63cbe
platform/packages/apps/Car/Stream c19ea2d878bf135e36151e6d313eee8b1b743184
platform/packages/apps/Car/SystemUpdater 3506f3f39a1328465009ec080a741373b338bead
platform/packages/apps/Car/UserManagement 9391baedbf4ad645d7c125279a19d7a2b6b285f9
-platform/packages/apps/Car/libs 7901432126584be2601424b0969092d0d1374c65
-platform/packages/apps/CarrierConfig 5a44b01593679a6b168e5af6591b72f96819a52c
-platform/packages/apps/CellBroadcastReceiver 8cd439087aab7e0c3d530f0678a4944b697602fc
-platform/packages/apps/CertInstaller 1c43ca95cd444163e69e250f270714d7ba4cffaa
-platform/packages/apps/Contacts a33aca875ae2135e720385d07d637cf8819c7f8a
-platform/packages/apps/DeskClock 411d606453dd34b76a7af12c568872e6f98ffc86
+platform/packages/apps/Car/libs 505ccbf9a8b504e0e149ba74a344ac8b756371b7
+platform/packages/apps/CarrierConfig a6aa1c392f1a87f37c20768a9f4f72cca9b4373e
+platform/packages/apps/CellBroadcastReceiver 903204cd301ead48d3ef87ac7b833abd69ee20d8
+platform/packages/apps/CertInstaller d1d8799384570ce5933484d22a3edec99a8c4066
+platform/packages/apps/Contacts e387a54b548daa4aa8f204f2fe6a63f1c0ea8c52
+platform/packages/apps/DeskClock 6fc4a3e6bce7a2870a36365f02e547573075d098
platform/packages/apps/DevCamera 90fdbf7460fbd255194455331c6a38a7b8f41dd2
-platform/packages/apps/Dialer c39ea3c55fac807c0b98aabdf56c70dc8a49036c
-platform/packages/apps/DocumentsUI 4501139b4ac250e5437007dc7f92adcb1068565f
-platform/packages/apps/Email 1b70a229c444d3e893e6309e5191bb205fb1dcef
-platform/packages/apps/EmergencyInfo 4877fa47dcb0275079b802054104811e4fedb9f1
-platform/packages/apps/ExactCalculator 9354ca48fbf8e8ea3372d01d024dd917a4c6f553
-platform/packages/apps/Gallery 005fec50c2e22be3441ab9415ab17bd64bbd6b44
-platform/packages/apps/Gallery2 65cdfbaad6c99efc1671510daea762a0876199b5
+platform/packages/apps/Dialer e5041c35b4c0ef31e5d242a8dd61d7441d490f44
+platform/packages/apps/DocumentsUI b14cc5cb6598ad01d841db21b6d72b7444969984
+platform/packages/apps/Email 224f3f566bda37f2ee0a7bafa75a2d815c9e9687
+platform/packages/apps/EmergencyInfo 85a89c64b45e6b53eab3ccf651d339661e13d6ba
+platform/packages/apps/ExactCalculator 9223c18a08481c79f2f363278d3bc47116ae551f
+platform/packages/apps/Gallery f08154dbdd726c78990f003f3bfe0a4ce30b3849
+platform/packages/apps/Gallery2 ea2a43b6e82217b20190e8979160d931eae2b686
platform/packages/apps/HTMLViewer e3269d171111f474c9240688c3da62e26f6a2093
-platform/packages/apps/KeyChain 1e1f4535d86dd49bf23108eb1a4c3d2543fc85fe
-platform/packages/apps/Launcher2 0269d850997f5d8c8f82d2f94a8c648ad0cb1543
-platform/packages/apps/Launcher3 fa5511133a6bf784955042966162e8635ab5b930
-platform/packages/apps/LegacyCamera b717e128c1835dcee7efdbeedd5adc6c371f0b61
-platform/packages/apps/ManagedProvisioning f1dde3461ec2d47adfd49a6752770ca654410b2d
-platform/packages/apps/Messaging 3fb6abb42199b98a23a6e0569b83519919f9d015
-platform/packages/apps/Music 033375b60c90a9ecb1bb12197f978e31f95419bc
-platform/packages/apps/MusicFX e51ca133f48c3856d14a6f3230bd4f52a7a1283b
-platform/packages/apps/Nfc 6b79a8601b2a0dfc18c3d941208d877246f592b8
-platform/packages/apps/OneTimeInitializer d2762cc94e5ea13cc13b6665101220c2ddf06110
-platform/packages/apps/PackageInstaller 1677529d348d72daa1531ff5c1f1e4d5cd9ff19c
-platform/packages/apps/Phone 79731f02b7009206a01182d2cca15dfc8491da09
-platform/packages/apps/PhoneCommon 742f2254b5fd8b8239724008dadcdac7debfe1c6
+platform/packages/apps/KeyChain fd931937d42ec74a893c903bd4a3b3f0e8fe8d58
+platform/packages/apps/Launcher2 29a7128376610efd9ad19d121243b2b6ded12af4
+platform/packages/apps/Launcher3 29e4a28f8cf8af3e7dc0efe489bb44e8de328cee
+platform/packages/apps/LegacyCamera aa0ad5191cc5df7c9904d4a6ae76534e5a3f8515
+platform/packages/apps/ManagedProvisioning 2fb67984b0d218b4e09df07d4b57967a6af65524
+platform/packages/apps/Messaging 443a8888f29809a1d70bc198c9d7d10f12db1cd4
+platform/packages/apps/Music a6a28406e16f4c87a62d34b51a1e9f9de63caddb
+platform/packages/apps/MusicFX 387f835feac9aab5b1935ac20190292ff574afd4
+platform/packages/apps/Nfc 32e82fb6333c012f8d332688cb7a105e7281e3a8
+platform/packages/apps/OneTimeInitializer 1eabf23efa9b3e6c68fb30195e0c3fd10dd57d04
+platform/packages/apps/PackageInstaller 1d5b10bfb6246dcecfa26b560969248c16435347
+platform/packages/apps/PhoneCommon 61825b76ecd9b6fcbe299b0716be22ed9f7cd0d9
platform/packages/apps/Protips 1e5d8a500dce6530a99cb952681889042dcfb63a
platform/packages/apps/Provision 6b2548213a9c7cf93ac9dc32f8b82db4ad8e9c4d
-platform/packages/apps/QuickSearchBox 5f6197edeea02d084a1a042374b8f15e7ed5131f
-platform/packages/apps/RetailDemo bc538c50dae99678584ff9e2aff60ee11a78cb44
-platform/packages/apps/SafetyRegulatoryInfo e69ff22ef6c1ad43c5ff876b19fc492c94d4fc12
-platform/packages/apps/Settings 0bed3cacd7bb5fab60242fa8690a9a6b1eee8698
+platform/packages/apps/QuickSearchBox 29885db1a02334eba9a75bfb4e84de38fc9d9c66
+platform/packages/apps/RetailDemo 6c74324e41bde711ec2c756060c97e49fbedfa32
+platform/packages/apps/SafetyRegulatoryInfo 22b1961035bf349dd1f6e4a0373db9ffa13e106e
+platform/packages/apps/Settings aa8a972ea3b4f6ed7565d05c5ade6d51258b445b
platform/packages/apps/SoundRecorder ba7237540f779f1c7e67ae7a36aa7a63da8eb1c5
platform/packages/apps/SpareParts f47fff0421153f4c20d155df8d0864a7036bdb06
-platform/packages/apps/SpeechRecorder 51cc8462c082a26810f5d7c5a2648232a77e795c
-platform/packages/apps/Stk 8ffca3e4b7b42d2449cbc5000130c8e08994e280
-platform/packages/apps/StorageManager bf23305b3b201f91d9c06eafa2622b058ced93dd
-platform/packages/apps/TV 0e0082535dc276086ee61ac156581a20e86609e7
-platform/packages/apps/Tag 018a398bc877266e87d05b81a863f3b630187f71
-platform/packages/apps/Terminal 6ef2b729b4e4ac070563efba29fca2b98ba09e56
-platform/packages/apps/Test/connectivity 821545335c8e35cc43aeae463b42732fd75b46b9
-platform/packages/apps/TimeZoneData bba3dfde202b7483b1e430d75d49e0389d42d644
-platform/packages/apps/TimeZoneUpdater 4f154a7d574963466abaeded961b6c5f427a45fb
-platform/packages/apps/TvSettings 51fb032a154fa756327c5dacce6a2540b1196c3b
-platform/packages/apps/UnifiedEmail 6087c3f339a76e9033f6c2991ef26d4401743429
-platform/packages/apps/WallpaperPicker 969934f908ff2ddda77bddbb80f206daf6a24281
-platform/packages/experimental f664e1fad87d06f1ad738549db049011c84ca787
-platform/packages/inputmethods/LatinIME cfc834197c6e2ae6bdff1b7e4cde47cf0a2c0dd4
+platform/packages/apps/Stk 9a1b8610ebe21f65b53fd1ecaecc7e368f375257
+platform/packages/apps/StorageManager fc91593bb2d2656a85e61ec9004aa1f881694f7d
+platform/packages/apps/TV 8240a88ba4469279d99444bcd08eebdc1c0dc5d8
+platform/packages/apps/Tag 18565f68c4f0a3eeaf407c6c33e0061a763c96a3
+platform/packages/apps/Terminal 2af09e5df57946f0b8df7cde98c9c939d0ee12d1
+platform/packages/apps/Test/connectivity 687fec924e44d1a4cee43f62047555eba405d115
+platform/packages/apps/TimeZoneData fde6b9823389a57a0cac6d2e6293145316041fee
+platform/packages/apps/TimeZoneUpdater 1dff9355659d622b9278844a333a45bd7cbf00bf
+platform/packages/apps/TvSettings 777b4920c6450395c7917f7db504e582bb65a00d
+platform/packages/apps/UnifiedEmail a1ba4867640a96cbb6c44f656c3da18334923a9b
+platform/packages/apps/WallpaperPicker c61e5b02e0dba6d0f9d33449c3d085bd87b9bca1
+platform/packages/experimental 9fdd32e6758bcca54518b82a6876e7cd6934cd90
+platform/packages/inputmethods/LatinIME 21a3e7ef7aaab9f3684f09dfc6f5efad1d089553
platform/packages/inputmethods/OpenWnn 85ca997d8265739e94b5bd9d19460c9b87b9ec59
platform/packages/providers/ApplicationsProvider 33d26f5eedb3d3011762ce5b2de66e931bf64b35
-platform/packages/providers/BlockedNumberProvider aa0b531d60d27409d304d7e8bbad787181be4831
-platform/packages/providers/BookmarkProvider bf198c8c5cd3e128046a3f712709690734b61f43
-platform/packages/providers/CalendarProvider 1e487cd116214c0df34efea269595352b757e4c9
-platform/packages/providers/CallLogProvider df0624f9cf6274ed67c70ef0d163aa27ad5bf2d3
-platform/packages/providers/ContactsProvider 18a114896c47de3a2c29b5731afa658c57ca353c
-platform/packages/providers/DownloadProvider 372fcac69b5033bba262ce871f4f54ff1dde48b8
-platform/packages/providers/MediaProvider 428574d247cc821825721ca7fd3beb77b5e8895e
-platform/packages/providers/PartnerBookmarksProvider 9c4e2ffaf6344c6c6e3ec5884f29dc7999bdf363
-platform/packages/providers/TelephonyProvider 80178f441abc1f9352ade9c7597de11634bac269
-platform/packages/providers/TvProvider a22ab1935b10a91ac9eb9606d27f4290c5659894
-platform/packages/providers/UserDictionaryProvider ef263aa8c0557e3aab06d11c8914b53b59ee8e7e
-platform/packages/screensavers/Basic 2d9323d2deeeaa720cfb35d15767dd90a256b2c9
-platform/packages/screensavers/PhotoTable f27f650d1184819984e47ecbd797ccff162c0eeb
+platform/packages/providers/BlockedNumberProvider 5d9b02f04444771eb3a64e4f1611699deddfcc48
+platform/packages/providers/BookmarkProvider 429f5487edb04a11c1fb176b02f0490517b38159
+platform/packages/providers/CalendarProvider 8028388d715b058e191dba5b078e58567aebafb1
+platform/packages/providers/CallLogProvider cc28c0668ccb09cf131ec826977ca4c58e576e66
+platform/packages/providers/ContactsProvider a637f96cb2e2ce2fc8280741fd7cabdaa7c48a0c
+platform/packages/providers/DownloadProvider b50329927cf915576474ce4a3d9a7ff9e57c4df5
+platform/packages/providers/MediaProvider b09d81a7d73bcad57b1a6413e114ff1ebddd946f
+platform/packages/providers/PartnerBookmarksProvider 2817a5f96940035f870c745108e33dcb232f443f
+platform/packages/providers/TelephonyProvider 10caff8a090c37e7c70eba654e110d56289c9f07
+platform/packages/providers/TvProvider 2753f1f328121c6cf6fe9c4a18ae9a8ba816051c
+platform/packages/providers/UserDictionaryProvider 5591a71a53806091b2db68c5db3423c9dde25ad9
+platform/packages/screensavers/Basic 46ab2614d5cd7732b53175e9fdd2faa8bd191de6
+platform/packages/screensavers/PhotoTable 213d8f35b0a7bd0e63cfc78b799c8858e94b2214
platform/packages/screensavers/WebView 6e0a80f6faed6191acc8ce1b6c79eada09e9e042
-platform/packages/services/BuiltInPrintService 911db0bab1adfab02e551fb512e7fac8f8dccfe1
-platform/packages/services/Car 39cfdea1817951d36b08d9ac7630b73930d3ab87
-platform/packages/services/Mms 7117bff6c1186110a6381c49b7eac3cebd3d23f3
-platform/packages/services/NetworkRecommendation 9264ab18a037a78501f3ada2c1f307c436dfb8a1
-platform/packages/services/Telecomm 702f8f5870277fc298b437bda3df5882586b3b6e
-platform/packages/services/Telephony 8823dd0a36e7ab5c69d66ab8f81adfa8fa88b315
-platform/packages/wallpapers/LivePicker 5afd06f0764ce8afc99e641794ca3a57bdf87c2a
-platform/pdk c7bd144646f9e961a7ee5f10a0105d21f5bf8ff7
-platform/platform_testing 0606a72fb0360c96af13b9917e1248f9076e8f95
-platform/prebuilts/abi-dumps/ndk 93ee4dea555579076cc5313e04ed95cfbb0523cd
-platform/prebuilts/abi-dumps/vndk 83a0ec4746ab945004a9e71f2253d5fed8a4cc75
-platform/prebuilts/android-emulator 25dec9283da1c6a3c281ab0eb123c44f70c27632
-platform/prebuilts/build-tools afe22c74f958ae77d130dd87e7d01fd9f180c345
+platform/packages/services/BuiltInPrintService 3335da5994a85d1b16d62ab26bdecefe2b76128e
+platform/packages/services/Car be7d132e7c784b2a8e77f0b3c85117462cde9888
+platform/packages/services/Mms 0285e3a1c1322682bc6f45042f6aba5376dd14a8
+platform/packages/services/NetworkRecommendation 727389b2a3e52a35bb91a2ed8efb2b1fd9cc7cb9
+platform/packages/services/Telecomm 1707c239d1deafda0f296c70dcbd6748ef766e6e
+platform/packages/services/Telephony 25f76654f92e58b17f3c23e67678ff4061e62ca1
+platform/packages/wallpapers/LivePicker 8a85d46930ad6c9e63b03957ff68db9c2619bf9e
+platform/pdk a667a6faef44482c4d2c8a3ec1479ebf542e9fe4
+platform/platform_testing 659819eeb6ec2a7d9a3d2960075394615730ed30
+platform/prebuilts/abi-dumps/ndk d73817a493844b3363d3a1609b689cef202920e2
+platform/prebuilts/abi-dumps/vndk 04d5a43771aac062fc107f4cb3e7bad0b2d66ba4
+platform/prebuilts/android-emulator 0ee964619cf26195eaabfc96c35cc1ea8641f4f5
+platform/prebuilts/build-tools 54583134509ccf0a21300abdfc02aa914d0070a7
platform/prebuilts/checkcolor f6f661b424f646bc5ddf9e1b1e644abb492b7547
-platform/prebuilts/checkstyle b591a4c55c5fc6658e17875a642bb60923ed366a
-platform/prebuilts/clang/host/darwin-x86 ce827a504ccce0f032943202fbb5c1ee8fba8857
-platform/prebuilts/clang/host/linux-x86 c0674d4d25d8555d2e8a9beb7ba5e1f3fb144817
-platform/prebuilts/deqp d055450404768f16ac199b13e51eecb37abb7c07
-platform/prebuilts/devtools 6f0b60e8759e67828288feccfe0e59ad02170bd0
-platform/prebuilts/eclipse efa717742380a2fe700a62f3ba5b26e7fb3cf0f2
-platform/prebuilts/eclipse-build-deps ceb739d6a7c10f5fb5a6cf6e1f702453b1361ad3
-platform/prebuilts/gcc/darwin-x86/aarch64/aarch64-linux-android-4.9 7c946f823b985a797c63983aa3b3bd1ec2dceab3
-platform/prebuilts/gcc/darwin-x86/arm/arm-linux-androideabi-4.9 6895d0a62cfb78005df68310c61a7fa5cd20d685
+platform/prebuilts/checkstyle da9c18252bd84097b728065edfe20e6aac7e8771
+platform/prebuilts/clang/host/darwin-x86 6e4f67f07085cecbdbba832e8d8dc8492350e499
+platform/prebuilts/clang/host/linux-x86 df97928acd4a51a3d2e6f6342ba56e24631c918a
+platform/prebuilts/deqp 17c337bfe7f77ab838f83b81ba1565bb975886b0
+platform/prebuilts/devtools b6b85b68670e8df17b70a6a5eac5251d98e7b72a
+platform/prebuilts/gcc/darwin-x86/aarch64/aarch64-linux-android-4.9 ca55c760922e0e529b6d5771bea3212f75abe38d
+platform/prebuilts/gcc/darwin-x86/arm/arm-linux-androideabi-4.9 439276aed527c627e4fc35a299ab19522621a167
platform/prebuilts/gcc/darwin-x86/host/i686-apple-darwin-4.2.1 ec5aa66aaa4964c27564d0ec84dc1f18a2d72b7e
platform/prebuilts/gcc/darwin-x86/mips/mips64el-linux-android-4.9 d942564c9331f07026a8f78502d8571206eb7be4
-platform/prebuilts/gcc/darwin-x86/x86/x86_64-linux-android-4.9 9b83502b78b50217b04e66b34a3e017683093cb3
-platform/prebuilts/gcc/linux-x86/aarch64/aarch64-linux-android-4.9 b06651d136ffbab2098a2152abff4a14b69632bb
-platform/prebuilts/gcc/linux-x86/arm/arm-linux-androideabi-4.9 13f048825a14abafc9cb290f840c276564806390
-platform/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.11-4.8 a461e1b3161b4817b743ddbfb58a3878e7985c8b
-platform/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.15-4.8 6e91bb7b816df8543cffe14287acbf0412f65550
+platform/prebuilts/gcc/darwin-x86/x86/x86_64-linux-android-4.9 be3687e6f50bbb00feed9a01461f487fc2d4280f
+platform/prebuilts/gcc/linux-x86/aarch64/aarch64-linux-android-4.9 dc5072fa2f09d39b13923123f9ff255db9d3eacf
+platform/prebuilts/gcc/linux-x86/arm/arm-linux-androideabi-4.9 0299cbc8ab518f9b22239c349542f95a85b815c8
+platform/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.11-4.8 f75c4366bcb1519694962abd0d1c47933a69137f
+platform/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.15-4.8 f4efc77dafdc476d9a4f252e95b9bb1b0e1aa7ad
platform/prebuilts/gcc/linux-x86/host/x86_64-w64-mingw32-4.8 e76a9a506d7ad132f107eb2f7c27b6a8ccb68b91
platform/prebuilts/gcc/linux-x86/mips/mips64el-linux-android-4.9 388fdc4995d374d76a0c4b292afabac91638e134
-platform/prebuilts/gcc/linux-x86/x86/x86_64-linux-android-4.9 694b42b76aaf30420c2f7764dc56ed928537428e
+platform/prebuilts/gcc/linux-x86/x86/x86_64-linux-android-4.9 4e8f0879684223629cf28176fe49ddb72ce5b702
platform/prebuilts/gdb/darwin-x86 0a8c90248264a8b26970b4473770bcc3df8515fe
platform/prebuilts/gdb/linux-x86 4adfde8bc82dd39f59e0445588c3e599ada477d2
-platform/prebuilts/go/darwin-x86 1336ba94c3bda2b2db8d6eab54036ae88423be55
-platform/prebuilts/go/linux-x86 cc0b9f5238fed2fb3b3a0dfa9cd1dd163b053a8c
-platform/prebuilts/gradle-plugin 10219421eac01ee74d80c86e97bebe1fd8e16bb2
-platform/prebuilts/jdk/jdk8 0b0825979447fdf3a4d918e556f8cab675ec4312
+platform/prebuilts/go/darwin-x86 22677aa0576aa49be0fb64285fab83a78a5d2786
+platform/prebuilts/go/linux-x86 caa27bd4b435652172b6ed69d8f8f48d39e5881c
+platform/prebuilts/gradle-plugin 1a4e6eabdf72f1fd8c13fd3478c87eecf8dc9599
+platform/prebuilts/jdk/jdk8 51f4d6bf0fa83cdc5e28004b95ec6f8e947df85e
+platform/prebuilts/jdk/jdk9 7337450a075d7f2eff57382d889fb01c73eaa269
platform/prebuilts/libs/libedit d32685dba4011664b590b94ad156bc734c2c9bb5
-platform/prebuilts/maven_repo/android f8ffcaa0b464b1275af6707e931beb0b4d4b0856
-platform/prebuilts/maven_repo/bumptech ad29fe1a1b86eacbd04ac4ab23b90f38ca099f77
-platform/prebuilts/misc 85bf1819a733f1c5c87e1a1574e52badf9abfa56
-platform/prebuilts/ndk 03441d9c7276161225a04bd651eb45be84538423
+platform/prebuilts/maven_repo/android eea1d23669a7725b45e3a966f51d0f28ddcd86fd
+platform/prebuilts/maven_repo/bumptech e9660a2a8af386a9085134fd823a4f80c528b37a
+platform/prebuilts/misc 8acfb0b3e11d38d54bb972fb5f3dfdd0f777e2c5
+platform/prebuilts/ndk 8962769ddef3d98785ce5cad4dbedca8c8b2a6df
platform/prebuilts/python/darwin-x86/2.7.5 0c5958b1636c47ed7c284f859c8e805fd06a0e63
-platform/prebuilts/python/linux-x86/2.7.5 621d405f7e4db1d7046dde53c0c6e5add3e2743e
-platform/prebuilts/qemu-kernel 12bb8ccc8bd778e2e737ad68c391a977517066e0
-platform/prebuilts/sdk bc82674f7a6e29a8b7a2457cec099206bd66a607
-platform/prebuilts/tools f3fbb60b007f7a4a5f53b822d56a151bb95b8d65
-platform/sdk 2e170ea4622aa5d63e562f2eea78817ada415373
-platform/system/bt 303ff4cee3830fa9c1f9f8f30ca1b7a9dd95fe0d
+platform/prebuilts/python/linux-x86/2.7.5 7bd79b4f40b14c6746d16cbe20cfad4f8d4faf63
+platform/prebuilts/qemu-kernel f9325c3aa5493a06102036df9c3f979755495a4e
+platform/prebuilts/sdk c5c42e19654a17f46138d3ceb4b593dfa13a56d0
+platform/prebuilts/tools d58b4829529e8241cf37ad9af6374844e71a5f23
+platform/sdk 526656cc27f2f1554f7eba4990215903af097ff7
+platform/system/bt c6258565dbfd3498c152181194986e5db3a8d55b
platform/system/ca-certificates db524b256a637f18588c1d5ecc4a7d43cdebf8fd
-platform/system/chre 1355b80eaeeb2c63bdbc72a9f883d2391161700a
-platform/system/connectivity/wificond 7a83a7b6ae1e0c8d7a3ec912eafca33d418608c1
-platform/system/connectivity/wifilogd 2211b56ca924f4a810d48fc31bdec959dda8f5c1
-platform/system/core 9638729a9d9e3cbbbe1bb2784e774fef0f2e6a54
-platform/system/extras 5324d11761f52b87b9b4417a494c2da8c1e0e4ed
-platform/system/gatekeeper cfa8a4123c34ee89af12edd656b0bdfcd8cd594b
-platform/system/hardware/interfaces 5d9f8a8361743c1fe199b8f5af53cf2cb4458589
-platform/system/hwservicemanager 2f881ef152c5e18f7366bf3dec5fe502fd54a557
-platform/system/iot/attestation adb01bfee3c0f712a7aa2f811dfd6ccd10da4bad
-platform/system/keymaster 109901d94c6e3f15768f3d9e1ba32803999db672
-platform/system/libfmq 67f088af77ee069cfa69d6d04c05214321118105
-platform/system/libhidl 56eb149beaf89e56e9d27c95316e83dd4c443889
-platform/system/libhwbinder 81fc3cc8b1aa535785c02d5734a4416ec23ae131
-platform/system/libufdt bc0c8d0ef23e39813955d476676974dfd384683b
-platform/system/libvintf 48602dfa0ef4a40b58c6376bab82c6e46d6f6a0b
-platform/system/media 780c6311c55e7ec2a510a89271b0f9f69bebb647
-platform/system/netd a2d039c860379779bc079bc0e800d7b472e7bb3f
-platform/system/nfc 8c5a46cb6e052336ba3df4a09e58c66eb0961a04
-platform/system/nvram 15ec4335d3c14eba136fb86d2d9a0c099ccc96cc
-platform/system/security 78505bae4ce8d571d8543e9066d6f6a85ebefa95
-platform/system/sepolicy f58f1499c10d66c8d5785bd089db14c860c5beb8
-platform/system/timezone 7cb0d8ca7adc520b85400140a6984d9ab3d67b98
-platform/system/tools/aidl 3a67504b8909ef4371527777933c1a78fd60d167
-platform/system/tools/hidl a9cf2b2d31707989b7eb66efc05b56721ea83bab
-platform/system/tpm 06e295b58139a538745835f45ed889a30507d008
-platform/system/update_engine a48f630400429ca010c5462967607985f2ffa7e4
-platform/system/vold db725d33485d8c51bafbbf3566275abfbd152ae7
+platform/system/chre dd059e8204adcc3d7210ec2820d317acb40f4d57
+platform/system/connectivity/wificond ba2aef39e16600be9ca306e96cf21786718e549b
+platform/system/connectivity/wifilogd 6e5a94db8d7dc577a135cc81d80e0bdf39d42780
+platform/system/core 41a9b3c81d8a9871ab02589c300fd5859c95792e
+platform/system/extras 1ed0c279d86fc1fd4e19625d816acc2acd136ba9
+platform/system/gatekeeper 82a39a0d83944b66a5a58138dfc7b5658be8ea36
+platform/system/hardware/interfaces 29f5ec76f933d14785fead2dd78ed4360e8a16bb
+platform/system/hwservicemanager 2e4b54cfdc1f39ff096577ac082b6e6634bbeb93
+platform/system/iot/attestation 33de69ae5b2399206d7e28c1f8527522e5b8bef9
+platform/system/keymaster 94742cf14e0e680f234eedaa5a06a8e9a77fc0f0
+platform/system/libfmq 1afdf9f7e2859f996281895b0e56047c1ed64560
+platform/system/libhidl e949257f03552a0d3667ad393c532473ee4eb503
+platform/system/libhwbinder 1c5fe19a1bc8676d5059e9a2cde8b5314e813271
+platform/system/libufdt 32ae9915bdb1fad389cdf083201dc81acb2c75d5
+platform/system/libvintf 941137d77f03562ba0fd226a220c20f19799566b
+platform/system/media 56f00a0b6a035a1f1241a32ad8226b544b8debf9
+platform/system/netd 7a70d081a1d71e4bf361ce9aa4bda384bcd8a2a1
+platform/system/nfc b02db1a5069fc9bd2bc6f4be3f2b22fcdb3b175c
+platform/system/nvram 9d3d63729edbcb3f6b29e70e86d1e03943d50683
+platform/system/security 96b00c572e62f38aacb99f72f7dd882ed9fd6b70
+platform/system/sepolicy fe00f563aba4c6c719219e645a9ea49b7ad9532d
+platform/system/timezone 476fb2d9e89f753308b85d3edf3dd0c7cb26578d
+platform/system/tools/aidl 3a9ff7a91fa00b01a2cdb8942c60565214536b4a
+platform/system/tools/hidl 93400b6af7858599c875d41e60c49ebd79c075f3
+platform/system/tpm b369a67d4e975ec0d579cc047f2fd013c90bc69b
+platform/system/update_engine abb45a5542fedd2a7c587aa91402b21f606bc259
+platform/system/vold c923127e23056f2883f9f89e81d0a02ad79fc520
platform/test/framework 246492cf54d880214952e9cbb1ea490cbdd56f52
platform/test/vti/alert 2270d160e38f0623d708cce36754fe14168dce64
-platform/test/vti/dashboard 94115f096dc1996ea7b56c93f1cc2726a792a65e
+platform/test/vti/dashboard a064c91d790fc1fe08152b6ae35d94535df488e6
platform/test/vti/fuzz_test_serving 30d457ce6613d587901379844b9d0dfb37a67a7b
-platform/test/vti/test_serving 0a064803e1ebbfd5e587e9baf677d458a4e3bbd8
-platform/test/vts 14cf8fc743d6d6cb8fba400aecbbe8905e232b43
-platform/test/vts-testcase/fuzz 779aef521c47d7829c4b2d26dbd0f16db5c36a16
-platform/test/vts-testcase/hal c5f0cac717aa8ab9341a07a551292c48f61eb1df
-platform/test/vts-testcase/hal-trace 02a4c1be279b9afa69a0003db73652980aa3e85f
-platform/test/vts-testcase/kernel a86301d96c4a718f20f156e3c8b53af918d5d166
-platform/test/vts-testcase/performance 7b106d96253864d18850ee306c48775b6c0270a4
-platform/test/vts-testcase/security 5f9c0342d425bcaf20d6ed7027dcbc5cd9bd0943
-platform/test/vts-testcase/vndk 39b58b9604b44745b6e628291649045747e4d089
+platform/test/vti/test_serving b342941ef17f8d54e05c6a9d19987883406ea088
+platform/test/vts b11e7a1acc116e765741d87155949a9fbdda4dba
+platform/test/vts-testcase/fuzz 6b0587a698f65e396a67f766808112c41092061a
+platform/test/vts-testcase/hal 7db2153245d0f18dabf77bdc9497a9be274b25a0
+platform/test/vts-testcase/hal-trace 18e4e8fbf5a9af64b0a053c4a34014f312246bff
+platform/test/vts-testcase/kernel 75f04b2457d09440bf8ae830de6028e28a4f9aa2
+platform/test/vts-testcase/performance 777a3702a2ba5c81aaa97ce90f5a36419f3f1a09
+platform/test/vts-testcase/security f06130eedbf0c60d42772ef55c4c935fbe3dfe1f
+platform/test/vts-testcase/vndk 071b84b8b12769543dcffbbfb75fb5f6c4799ac7
platform/tools/adt/idea d3106c42c5466362da9bfb9f7453e5d60af83579
-platform/tools/apksig 8f0bf970b2475abcf7dc32a33605944617a8931e
+platform/tools/apksig 8010184556475a6a9cea4ad95e44b3125b03ccfe
platform/tools/base 908b391a9c006af569dfaff08b37f8fdd6c4da89
platform/tools/build 69c4b95102b4b9862bfba68b3eaf5b7537a705ee
-platform/tools/dexter c6b1c520015b8ff473d8f58ae253e8f94de2399d
-platform/tools/external/fat32lib 9cdd712d65a9e3ce2db93867d0d4a8288ad3bd52
-platform/tools/external/gradle f055ebc3d4b63026076c1077b6116f0b36bc514c
+platform/tools/dexter 729cfd5e01a00c922920081c850641b05ec965c0
+platform/tools/external/fat32lib aed8cb729bba5c4bc7f02b5fcd898bab2e2a4853
+platform/tools/external/gradle 4aa951119697df3b4fc58baff80fbb4e1a3d2f39
platform/tools/idea 9b5d02ac8c92b1e71523cc15cb3d168d57fbd898
-platform/tools/loganalysis 5bf6d894a874141fa8f806693b6c0a7c171f44a9
+platform/tools/loganalysis d4dcae820b34619b538ae08a15d495ad323cda82
platform/tools/motodev 69989786cefbde82527960a1e100ec9afba46a98
-platform/tools/repohooks 6abc478494f3bfb112e1d2e4ab1ff7e941667828
+platform/tools/repohooks 23e637629cd1be38b999fc0577a08cbe612e207f
platform/tools/studio/cloud 58f06e77e051fff3903adabca7acdaa9dd12ec2d
platform/tools/swt 8996e71047a2bd11efee46ef14e02435ab5fa07a
-platform/tools/test/connectivity 8b1dad0169ad9bd76aaabb9b36f510c4a04fb6dd
-platform/tools/tradefederation 7a5692d3d8fa5ec5db6f48f4a5a751b9e3080126
-platform/tools/tradefederation/contrib a5a7a9fabbf14a8d600aa612e5667ff3217c319c
-toolchain/benchmark 6236292bcfa5dcae8b572fbc978ec1edcdb04299
-toolchain/binutils 5318987751bf4e440bf3900416b36648bd41ff74
+platform/tools/test/connectivity e397db7a1bc7b0fa2af88feb4d5e0b039a63f040
+platform/tools/tradefederation 7f0789ea29e6fbd27bc2daef07b042f375d54e22
+platform/tools/tradefederation/contrib 4753a23dd42f5ee3287dbbd9830010071b1bcf98
+toolchain/benchmark 95500d7427284f3ab59a5a03f78bb56f167f0dd2
+toolchain/binutils fbc6a651310fca7acf92740565ef93168c172a7c
diff --git a/report.py b/report.py
index 1c1197d..76f56ff 100644
--- a/report.py
+++ b/report.py
@@ -320,7 +320,6 @@
while i < len(args):
if args[i] == '-h' or args[i] == '--help':
print('report.py A python wrapper for simpleperf report command.')
- print('report.py [')
print('Options supported by simpleperf report command:')
print(get_simpleperf_report_help_msg())
print('\nOptions supported by report.py:')
diff --git a/report_html.js b/report_html.js
new file mode 100644
index 0000000..fff18f2
--- /dev/null
+++ b/report_html.js
@@ -0,0 +1,883 @@
+/*
+ * Copyright (C) 2017 The Android Open Source Project
+ *
+ * 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.
+ */
+'use strict';
+
+// Use IIFE to avoid leaking names to other scripts.
+$(document).ready(function() {
+
+function openHtml(name, attrs={}) {
+ let s = `<${name} `;
+ for (let key in attrs) {
+ s += `${key}="${attrs[key]}" `;
+ }
+ s += '>';
+ return s;
+}
+
+function closeHtml(name) {
+ return `</${name}>`;
+}
+
+function getHtml(name, attrs={}) {
+ let text;
+ if ('text' in attrs) {
+ text = attrs.text;
+ delete attrs.text;
+ }
+ let s = openHtml(name, attrs);
+ if (text) {
+ s += text;
+ }
+ s += closeHtml(name);
+ return s;
+}
+
+function getTableRow(cols, colName, attrs={}) {
+ let s = openHtml('tr', attrs);
+ for (let col of cols) {
+ s += `<${colName}>${col}</${colName}>`;
+ }
+ s += '</tr>';
+ return s;
+}
+
+function toPercentageStr(percentage) {
+ return percentage.toFixed(2) + '%';
+}
+
+function getProcessName(pid) {
+ let name = gProcesses[pid];
+ return name ? `${pid} (${name})`: pid.toString();
+}
+
+function getThreadName(tid) {
+ let name = gThreads[tid];
+ return name ? `${tid} (${name})`: tid.toString();
+}
+
+function getLibName(libId) {
+ return gLibList[libId];
+}
+
+function getFuncName(funcId) {
+ return gFunctionMap[funcId][1];
+}
+
+function getLibNameOfFunction(funcId) {
+ return getLibName(gFunctionMap[funcId][0]);
+}
+
+class TabManager {
+ constructor(divContainer) {
+ this.div = $('<div>', {id: 'tabs'});
+ this.div.appendTo(divContainer);
+ this.div.append(getHtml('ul'));
+ this.tabs = [];
+ this.isDrawCalled = false;
+ }
+
+ addTab(title, tabObj) {
+ let id = 'tab_' + this.div.children().length;
+ let tabDiv = $('<div>', {id: id});
+ tabDiv.appendTo(this.div);
+ this.div.children().first().append(
+ getHtml('li', {text: getHtml('a', {href: '#' + id, text: title})}));
+ tabObj.init(tabDiv);
+ this.tabs.push(tabObj);
+ if (this.isDrawCalled) {
+ this.div.tabs('refresh');
+ }
+ return tabObj;
+ }
+
+ findTab(title) {
+ let links = this.div.find('li a');
+ for (let i = 0; i < links.length; ++i) {
+ if (links.eq(i).text() == title) {
+ return this.tabs[i];
+ }
+ }
+ return null;
+ }
+
+ draw() {
+ this.div.tabs({
+ active: 0,
+ });
+ this.tabs.forEach(function(tab) {
+ tab.draw();
+ });
+ this.isDrawCalled = true;
+ }
+
+ setActive(tabObj) {
+ for (let i = 0; i < this.tabs.length; ++i) {
+ if (this.tabs[i] == tabObj) {
+ this.div.tabs('option', 'active', i);
+ break;
+ }
+ }
+ }
+}
+
+// Show global information retrieved from the record file, including:
+// record time
+// machine type
+// Android version
+// record cmdline
+// total samples
+class RecordFileView {
+ constructor(divContainer) {
+ this.div = $('<div>');
+ this.div.appendTo(divContainer);
+ }
+
+ draw() {
+ if (gRecordInfo.recordTime) {
+ this.div.append(getHtml('p', {text: 'Record Time: ' + gRecordInfo.recordTime}));
+ }
+ if (gRecordInfo.machineType) {
+ this.div.append(getHtml('p', {text: 'Machine Type: ' + gRecordInfo.machineType}));
+ }
+ if (gRecordInfo.androidVersion) {
+ this.div.append(getHtml('p', {text: 'Android Version: ' + gRecordInfo.androidVersion}));
+ }
+ if (gRecordInfo.recordCmdline) {
+ this.div.append(getHtml('p', {text: 'Record Cmdline: ' + gRecordInfo.recordCmdline}));
+ }
+ this.div.append(getHtml('p', {text: 'Total Samples: ' + gRecordInfo.totalSamples}));
+ }
+}
+
+// Show pieChart of event count percentage of each process, thread, library and function.
+class ChartView {
+ constructor(divContainer, eventInfo) {
+ this.id = divContainer.children().length;
+ this.div = $('<div>', {id: 'chartstat_' + this.id});
+ this.div.appendTo(divContainer);
+ this.eventInfo = eventInfo;
+ this.processInfo = null;
+ this.threadInfo = null;
+ this.libInfo = null;
+ this.states = {
+ SHOW_EVENT_INFO: 1,
+ SHOW_PROCESS_INFO: 2,
+ SHOW_THREAD_INFO: 3,
+ SHOW_LIB_INFO: 4,
+ };
+ }
+
+ _getState() {
+ if (this.libInfo) {
+ return this.states.SHOW_LIB_INFO;
+ }
+ if (this.threadInfo) {
+ return this.states.SHOW_THREAD_INFO;
+ }
+ if (this.processInfo) {
+ return this.states.SHOW_PROCESS_INFO;
+ }
+ return this.states.SHOW_EVENT_INFO;
+ }
+
+ _drawTitle() {
+ if (this.eventInfo) {
+ this.div.append(getHtml('p', {text: `Event Type: ${this.eventInfo.eventName}`}));
+ }
+ if (this.processInfo) {
+ this.div.append(getHtml('p',
+ {text: `Process: ${getProcessName(this.processInfo.pid)}`}));
+ }
+ if (this.threadInfo) {
+ this.div.append(getHtml('p',
+ {text: `Thread: ${getThreadName(this.threadInfo.tid)}`}));
+ }
+ if (this.libInfo) {
+ this.div.append(getHtml('p',
+ {text: `Library: ${getLibName(this.libInfo.libId)}`}));
+ }
+ if (this.processInfo) {
+ let button = $('<button>', {text: 'Back'});
+ button.appendTo(this.div);
+ button.button().click(() => this._goBack());
+ }
+ }
+
+ _goBack() {
+ let state = this._getState();
+ if (state == this.states.SHOW_PROCESS_INFO) {
+ this.processInfo = null;
+ } else if (state == this.states.SHOW_THREAD_INFO) {
+ this.threadInfo = null;
+ } else if (state == this.states.SHOW_LIB_INFO) {
+ this.libInfo = null;
+ }
+ this.draw();
+ }
+
+ _selectHandler(chart) {
+ let selectedItem = chart.getSelection()[0];
+ if (selectedItem) {
+ let state = this._getState();
+ if (state == this.states.SHOW_EVENT_INFO) {
+ this.processInfo = this.eventInfo.processes[selectedItem.row];
+ } else if (state == this.states.SHOW_PROCESS_INFO) {
+ this.threadInfo = this.processInfo.threads[selectedItem.row];
+ } else if (state == this.states.SHOW_THREAD_INFO) {
+ this.libInfo = this.threadInfo.libs[selectedItem.row];
+ }
+ this.draw();
+ }
+ }
+
+ draw() {
+ google.charts.setOnLoadCallback(() => this.realDraw());
+ }
+
+ realDraw() {
+ this.div.empty();
+ this._drawTitle();
+ let data = new google.visualization.DataTable();
+ let title;
+ let state = this._getState();
+ if (state == this.states.SHOW_EVENT_INFO) {
+ title = 'Processes in event type ' + this.eventInfo.eventName;
+ data.addColumn('string', 'Process');
+ data.addColumn('number', 'EventCount');
+ let rows = [];
+ for (let process of this.eventInfo.processes) {
+ rows.push([getProcessName(process.pid), process.eventCount]);
+ }
+ data.addRows(rows);
+ } else if (state == this.states.SHOW_PROCESS_INFO) {
+ title = 'Threads in process ' + getProcessName(this.processInfo.pid);
+ data.addColumn('string', 'Thread');
+ data.addColumn('number', 'EventCount');
+ let rows = [];
+ for (let thread of this.processInfo.threads) {
+ rows.push([getThreadName(thread.tid), thread.eventCount]);
+ }
+ data.addRows(rows);
+ } else if (state == this.states.SHOW_THREAD_INFO) {
+ title = 'Libraries in thread ' + getThreadName(this.threadInfo.tid);
+ data.addColumn('string', 'Lib');
+ data.addColumn('number', 'EventCount');
+ let rows = [];
+ for (let lib of this.threadInfo.libs) {
+ rows.push([getLibName(lib.libId), lib.eventCount]);
+ }
+ data.addRows(rows);
+ } else if (state == this.states.SHOW_LIB_INFO) {
+ title = 'Functions in library ' + getLibName(this.libInfo.libId);
+ data.addColumn('string', 'Function');
+ data.addColumn('number', 'EventCount');
+ let rows = [];
+ for (let func of this.libInfo.functions) {
+ rows.push([getFuncName(func.g.f), func.g.e]);
+ }
+ data.addRows(rows);
+ }
+
+ let options = {
+ title: title,
+ width: 1000,
+ height: 600,
+ };
+ let wrapperDiv = $('<div>');
+ wrapperDiv.appendTo(this.div);
+ let chart = new google.visualization.PieChart(wrapperDiv.get(0));
+ chart.draw(data, options);
+ google.visualization.events.addListener(chart, 'select', () => this._selectHandler(chart));
+ }
+}
+
+
+class ChartStatTab {
+ constructor() {
+ }
+
+ init(div) {
+ this.div = div;
+ this.recordFileView = new RecordFileView(this.div);
+ this.chartViews = [];
+ for (let eventInfo of gSampleInfo) {
+ this.chartViews.push(new ChartView(this.div, eventInfo));
+ }
+ }
+
+ draw() {
+ this.recordFileView.draw();
+ for (let charView of this.chartViews) {
+ charView.draw();
+ }
+ }
+}
+
+
+class SampleTableTab {
+ constructor() {
+ }
+
+ init(div) {
+ this.div = div;
+ }
+
+ draw() {
+ for (let tId = 0; tId < gSampleInfo.length; tId++) {
+ let eventInfo = gSampleInfo[tId];
+ let eventName = eventInfo.eventName;
+ this.div.append(getHtml('p', {text: 'Sample table for event ' + eventName}));
+ let percentMul = 100.0 / eventInfo.eventCount;
+ let tableId = 'reportTable_' + tId;
+ let titles = ['Total', 'Self', 'SampleCount', 'Process', 'Thread', 'Lib', 'Function'];
+ let tableStr = openHtml('table', {id: tableId, cellspacing: '0', width: '100%'}) +
+ getHtml('thead', {text: getTableRow(titles, 'th')}) +
+ getHtml('tfoot', {text: getTableRow(titles, 'th')}) +
+ openHtml('tbody');
+
+ for (let i = 0; i < eventInfo.processes.length; ++i) {
+ let processInfo = eventInfo.processes[i];
+ let processName = getProcessName(processInfo.pid);
+ for (let j = 0; j < processInfo.threads.length; ++j) {
+ let threadInfo = processInfo.threads[j];
+ let threadName = getThreadName(threadInfo.tid);
+ for (let k = 0; k < threadInfo.libs.length; ++k) {
+ let lib = threadInfo.libs[k];
+ for (let t = 0; t < lib.functions.length; ++t) {
+ let func = lib.functions[t];
+ let key = [i, j, k, t].join('_');
+ let treePercentage = toPercentageStr(func.g.s * percentMul);
+ let selfPercenetage = toPercentageStr(func.g.e * percentMul);
+ tableStr += getTableRow([treePercentage, selfPercenetage, func.c,
+ processName, threadName, getLibName(lib.libId),
+ getFuncName(func.g.f)], 'td', {key: key});
+ }
+ }
+ }
+ }
+ tableStr += closeHtml('tbody') + closeHtml('table');
+ this.div.append(tableStr);
+ let table = this.div.find(`table#${tableId}`).dataTable({
+ lengthMenu: [10, 20, 50, 100, -1],
+ processing: true,
+ order: [0, 'desc'],
+ responsive: true,
+ });
+
+ table.find('tr').css('cursor', 'pointer');
+ table.on('click', 'tr', function() {
+ let key = this.getAttribute('key');
+ if (!key) {
+ return;
+ }
+ let indexes = key.split('_');
+ let processInfo = eventInfo.processes[indexes[0]];
+ let threadInfo = processInfo.threads[indexes[1]];
+ let lib = threadInfo.libs[indexes[2]];
+ let func = lib.functions[indexes[3]];
+ FunctionTab.showFunction(eventInfo, processInfo, threadInfo, lib, func);
+ });
+ }
+ }
+}
+
+
+// Show embedded flamegraph generated by inferno.
+class FlameGraphTab {
+ constructor() {
+ }
+
+ init(div) {
+ this.div = div;
+ }
+
+ draw() {
+ $('div#flamegraph_id').appendTo(this.div).css('display', 'block');
+ flamegraphInit();
+ }
+}
+
+
+// FunctionTab: show information of a function.
+// 1. Show the callgrpah and reverse callgraph of a function as flamegraphs.
+class FunctionTab {
+ static showFunction(eventInfo, processInfo, threadInfo, lib, func) {
+ let title = 'Function';
+ let tab = gTabs.findTab(title);
+ if (!tab) {
+ tab = gTabs.addTab(title, new FunctionTab());
+ }
+ tab.setFunction(eventInfo, processInfo, threadInfo, lib, func);
+ }
+
+ constructor() {
+ this.func = null;
+ this.selectPercent = 'thread';
+ }
+
+ init(div) {
+ this.div = div;
+ }
+
+ setFunction(eventInfo, processInfo, threadInfo, lib, func) {
+ this.eventInfo = eventInfo;
+ this.processInfo = processInfo;
+ this.threadInfo = threadInfo;
+ this.lib = lib;
+ this.func = func;
+ this.selectorView = null;
+ this.callgraphView = null;
+ this.reverseCallgraphView = null;
+ this.draw();
+ gTabs.setActive(this);
+ }
+
+ draw() {
+ if (!this.func) {
+ return;
+ }
+ this.div.empty();
+ let eventName = this.eventInfo.eventName;
+ let processName = getProcessName(this.processInfo.pid);
+ let threadName = getThreadName(this.threadInfo.tid);
+ let libName = getLibName(this.lib.libId);
+ let funcName = getFuncName(this.func.g.f);
+ let title = getHtml('p', {text: `Event ${eventName}`}) +
+ getHtml('p', {text: `Process ${processName}`}) +
+ getHtml('p', {text: `Thread ${threadName}`}) +
+ getHtml('p', {text: `Library ${libName}`}) +
+ getHtml('p', {text: `Function ${funcName}`});
+ this.div.append(title);
+
+ this.selectorView = new FunctionSampleWeightSelectorView(this.div, this.eventInfo,
+ this.processInfo, this.threadInfo, () => this.onSampleWeightChange());
+ this.selectorView.draw();
+
+ this.div.append(getHtml('hr'));
+ this.div.append(getHtml('b', {text: `Functions called by ${funcName}`}) + '<br/>');
+ this.callgraphView = new FlameGraphView(this.div, this.func.g, false);
+
+ this.div.append(getHtml('hr'));
+ this.div.append(getHtml('b', {text: `Functions calling ${funcName}`}) + '<br/>');
+ this.reverseCallgraphView = new FlameGraphView(this.div, this.func.rg, true);
+ this.onSampleWeightChange(); // Manually set sample weight function for the first time.
+ }
+
+ onSampleWeightChange() {
+ let sampleWeightFunction = this.selectorView.getSampleWeightFunction();
+ if (this.callgraphView) {
+ this.callgraphView.draw(sampleWeightFunction);
+ }
+ if (this.reverseCallgraphView) {
+ this.reverseCallgraphView.draw(sampleWeightFunction);
+ }
+ }
+}
+
+
+// Select the way to show sample weight in FunctionTab.
+// 1. Show percentage of event count relative to all processes.
+// 2. Show percentage of event count relative to the current process.
+// 3. Show percentage of event count relative to the current thread.
+// 4. Show absolute event count.
+// 5. Show event count in milliseconds, only possible for cpu-clock or task-clock events.
+class FunctionSampleWeightSelectorView {
+ constructor(divContainer, eventInfo, processInfo, threadInfo, onSelectChange) {
+ this.div = $('<div>');
+ this.div.appendTo(divContainer);
+ this.onSelectChange = onSelectChange;
+ this.eventCountForAllProcesses = eventInfo.eventCount;
+ this.eventCountForProcess = processInfo.eventCount;
+ this.eventCountForThread = threadInfo.eventCount;
+ this.options = {
+ PERCENT_TO_ALL_PROCESSES: 0,
+ PERCENT_TO_CUR_PROCESS: 1,
+ PERCENT_TO_CUR_THREAD: 2,
+ RAW_EVENT_COUNT: 3,
+ EVENT_COUNT_IN_TIME: 4,
+ }
+ let name = eventInfo.eventName;
+ this.supportEventCountInTime = name.includes('task-clock') || name.includes('cpu-clock');
+ if (this.supportEventCountInTime) {
+ this.curOption = this.options.EVENT_COUNT_IN_TIME;
+ } else {
+ this.curOption = this.options.PERCENT_TO_CUR_THREAD;
+ }
+ }
+
+ draw() {
+ let options = [];
+ options.push('Show percentage of event count relative to all processes.');
+ options.push('Show percentage of event count relative to the current process.');
+ options.push('Show percentage of event count relative to the current thread.');
+ options.push('Show event count.');
+ if (this.supportEventCountInTime) {
+ options.push('Show event count in milliseconds.');
+ }
+ let optionStr = '';
+ for (let i = 0; i < options.length; ++i) {
+ optionStr += getHtml('option', {value: i, text: options[i]});
+ }
+ this.div.append(getHtml('select', {text: optionStr}));
+ let selectMenu = this.div.children().last();
+ selectMenu.children().eq(this.curOption).attr('selected', 'selected');
+ let thisObj = this;
+ selectMenu.selectmenu({
+ change: function() {
+ thisObj.curOption = this.value;
+ thisObj.onSelectChange();
+ },
+ width: '100%',
+ });
+ }
+
+ getSampleWeightFunction() {
+ let thisObj = this;
+ if (this.curOption == this.options.PERCENT_TO_ALL_PROCESSES) {
+ return function(eventCount) {
+ let percent = eventCount * 100.0 / thisObj.eventCountForAllProcesses;
+ return percent.toFixed(2) + '%';
+ };
+ }
+ if (this.curOption == this.options.PERCENT_TO_CUR_PROCESS) {
+ return function(eventCount) {
+ let percent = eventCount * 100.0 / thisObj.eventCountForProcess;
+ return percent.toFixed(2) + '%';
+ };
+ }
+ if (this.curOption == this.options.PERCENT_TO_CUR_THREAD) {
+ return function(eventCount) {
+ let percent = eventCount * 100.0 / thisObj.eventCountForThread;
+ return percent.toFixed(2) + '%';
+ };
+ }
+ if (this.curOption == this.options.RAW_EVENT_COUNT) {
+ return function(eventCount) {
+ return '' + eventCount;
+ };
+ }
+ if (this.curOption == this.options.EVENT_COUNT_IN_TIME) {
+ return function(eventCount) {
+ let timeInMs = eventCount / 1000000.0;
+ return timeInMs.toFixed(3) + ' ms';
+ };
+ }
+ }
+}
+
+
+// Given a callgraph, show the flamegraph.
+class FlameGraphView {
+ // If reverseOrder is false, the root of the flamegraph is at the bottom,
+ // otherwise it is at the top.
+ constructor(divContainer, callgraph, reverseOrder) {
+ this.id = divContainer.children().length;
+ this.div = $('<div>', {id: 'fg_' + this.id});
+ this.div.appendTo(divContainer);
+ this.callgraph = callgraph;
+ this.reverseOrder = reverseOrder;
+ this.sampleWeightFunction = null;
+ this.svgWidth = $(window).width();
+ this.svgNodeHeight = 17;
+ this.fontSize = 12;
+
+ function getMaxDepth(node) {
+ let depth = 0;
+ for (let child of node.c) {
+ depth = Math.max(depth, getMaxDepth(child));
+ }
+ return depth + 1;
+ }
+ this.maxDepth = getMaxDepth(this.callgraph);
+ this.svgHeight = this.svgNodeHeight * (this.maxDepth + 3);
+ }
+
+ draw(sampleWeightFunction) {
+ this.sampleWeightFunction = sampleWeightFunction;
+ this.div.empty();
+ this.div.css('width', '100%').css('height', this.svgHeight + 'px');
+ let svgStr = '<svg xmlns="http://www.w3.org/2000/svg" \
+ xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" \
+ width="100%" height="100%" style="border: 1px solid black; font-family: Monospace;"> \
+ </svg>';
+ this.div.append(svgStr);
+ this.svg = this.div.find('svg');
+ this._renderBackground();
+ this._renderSvgNodes(this.callgraph, 0, 0);
+ this._renderUnzoomNode();
+ this._renderInfoNode();
+ this._renderPercentNode();
+ // Make the added nodes in the svg visible.
+ this.div.html(this.div.html());
+ this.svg = this.div.find('svg');
+ this._adjustTextSize();
+ this._enableZoom();
+ this._enableInfo();
+ this._adjustTextSizeOnResize();
+ }
+
+ _renderBackground() {
+ this.svg.append(`<defs > <linearGradient id="background_gradient_${this.id}"
+ y1="0" y2="1" x1="0" x2="0" > \
+ <stop stop-color="#eeeeee" offset="5%" /> \
+ <stop stop-color="#efefb1" offset="90%" /> \
+ </linearGradient> \
+ </defs> \
+ <rect x="0" y="0" width="100%" height="100%" \
+ fill="url(#background_gradient_${this.id})" />`);
+ }
+
+ _getYForDepth(depth) {
+ if (this.reverseOrder) {
+ return (depth + 3) * this.svgNodeHeight;
+ }
+ return this.svgHeight - (depth + 1) * this.svgNodeHeight;
+ }
+
+ _getWidthPercentage(eventCount) {
+ return eventCount * 100.0 / this.callgraph.s;
+ }
+
+ _getHeatColor(widthPercentage) {
+ return {
+ r: Math.floor(245 + 10 * (1 - widthPercentage * 0.01)),
+ g: Math.floor(110 + 105 * (1 - widthPercentage * 0.01)),
+ b: 100,
+ };
+ }
+
+ _renderSvgNodes(callNode, depth, xOffset) {
+ let x = xOffset;
+ let y = this._getYForDepth(depth);
+ let width = this._getWidthPercentage(callNode.s);
+ if (width < 0.1) {
+ return xOffset;
+ }
+ let color = this._getHeatColor(width);
+ let borderColor = {};
+ for (let key in color) {
+ borderColor[key] = Math.max(0, color[key] - 50);
+ }
+ let funcName = getFuncName(callNode.f);
+ let libName = getLibNameOfFunction(callNode.f);
+ let sampleWeight = this.sampleWeightFunction(callNode.s);
+ let title = funcName + ' | ' + libName + ' (' + callNode.s + ' events: ' +
+ sampleWeight + ')';
+ this.svg.append(`<g> <title>${title}</title> <rect x="${x}%" y="${y}" ox="${x}" \
+ depth="${depth}" width="${width}%" owidth="${width}" height="15.0" \
+ ofill="rgb(${color.r},${color.g},${color.b})" \
+ fill="rgb(${color.r},${color.g},${color.b})" \
+ style="stroke:rgb(${borderColor.r},${borderColor.g},${borderColor.b})"/> \
+ <text x="${x}%" y="${y + 12}" font-size="${this.fontSize}" \
+ font-family="Monospace"></text></g>`);
+
+ let childXOffset = xOffset;
+ for (let child of callNode.c) {
+ childXOffset = this._renderSvgNodes(child, depth + 1, childXOffset);
+ }
+ return xOffset + width;
+ }
+
+ _renderUnzoomNode() {
+ this.svg.append(`<rect id="zoom_rect_${this.id}" style="display:none;stroke:rgb(0,0,0);" \
+ rx="10" ry="10" x="10" y="10" width="80" height="30" \
+ fill="rgb(255,255,255)"/> \
+ <text id="zoom_text_${this.id}" x="19" y="30" style="display:none">Zoom out</text>`);
+ }
+
+ _renderInfoNode() {
+ this.svg.append(`<clipPath id="info_clip_path_${this.id}"> \
+ <rect style="stroke:rgb(0,0,0);" rx="10" ry="10" x="120" y="10" \
+ width="789" height="30" fill="rgb(255,255,255)"/> \
+ </clipPath> \
+ <rect style="stroke:rgb(0,0,0);" rx="10" ry="10" x="120" y="10" \
+ width="799" height="30" fill="rgb(255,255,255)"/> \
+ <text clip-path="url(#info_clip_path_${this.id})" \
+ id="info_text_${this.id}" x="128" y="30"></text>`);
+ }
+
+ _renderPercentNode() {
+ this.svg.append(`<rect style="stroke:rgb(0,0,0);" rx="10" ry="10" \
+ x="934" y="10" width="100" height="30" \
+ fill="rgb(255,255,255)"/> \
+ <text id="percent_text_${this.id}" text-anchor="end" \
+ x="1024" y="30">100.00%</text>`);
+ }
+
+ _adjustTextSizeForNode(g) {
+ let text = g.find('text');
+ let width = parseFloat(g.find('rect').attr('width')) * this.svgWidth * 0.01;
+ if (width < 28) {
+ text.text('');
+ return;
+ }
+ let methodName = g.find('title').text().split(' | ')[0];
+ let numCharacters;
+ for (numCharacters = methodName.length; numCharacters > 4; numCharacters--) {
+ if (numCharacters * 7.5 <= width) {
+ break;
+ }
+ }
+ if (numCharacters == methodName.length) {
+ text.text(methodName);
+ } else {
+ text.text(methodName.substring(0, numCharacters - 2) + '..');
+ }
+ }
+
+ _adjustTextSize() {
+ this.svgWidth = $(window).width();
+ let thisObj = this;
+ this.svg.find('g').each(function(_, g) {
+ thisObj._adjustTextSizeForNode($(g));
+ });
+ }
+
+ _enableZoom() {
+ this.zoomStack = [this.svg.find('g').first().get(0)];
+ this.svg.find('g').css('cursor', 'pointer').click(zoom);
+ this.svg.find(`#zoom_rect_${this.id}`).css('cursor', 'pointer').click(unzoom);
+ this.svg.find(`#zoom_text_${this.id}`).css('cursor', 'pointer').click(unzoom);
+
+ let thisObj = this;
+ function zoom() {
+ thisObj.zoomStack.push(this);
+ displayFromElement(this);
+ thisObj.svg.find(`#zoom_rect_${thisObj.id}`).css('display', 'block');
+ thisObj.svg.find(`#zoom_text_${thisObj.id}`).css('display', 'block');
+ }
+
+ function unzoom() {
+ if (thisObj.zoomStack.length > 1) {
+ thisObj.zoomStack.pop();
+ displayFromElement(thisObj.zoomStack[thisObj.zoomStack.length - 1]);
+ if (thisObj.zoomStack.length == 1) {
+ thisObj.svg.find(`#zoom_rect_${thisObj.id}`).css('display', 'none');
+ thisObj.svg.find(`#zoom_text_${thisObj.id}`).css('display', 'none');
+ }
+ }
+ }
+
+ function displayFromElement(g) {
+ g = $(g);
+ let clickedRect = g.find('rect');
+ let clickedOriginX = parseFloat(clickedRect.attr('ox'));
+ let clickedDepth = parseInt(clickedRect.attr('depth'));
+ let clickedOriginWidth = parseFloat(clickedRect.attr('owidth'));
+ let scaleFactor = 100.0 / clickedOriginWidth;
+ thisObj.svg.find('g').each(function(_, g) {
+ g = $(g);
+ let text = g.find('text');
+ let rect = g.find('rect');
+ let depth = parseInt(rect.attr('depth'));
+ let ox = parseFloat(rect.attr('ox'));
+ let owidth = parseFloat(rect.attr('owidth'));
+ if (depth < clickedDepth || ox < clickedOriginX - 1e-9 ||
+ ox + owidth > clickedOriginX + clickedOriginWidth + 1e-9) {
+ rect.css('display', 'none');
+ text.css('display', 'none');
+ } else {
+ rect.css('display', 'block');
+ text.css('display', 'block');
+ let nx = (ox - clickedOriginX) * scaleFactor + '%';
+ let ny = thisObj._getYForDepth(depth - clickedDepth);
+ rect.attr('x', nx);
+ rect.attr('y', ny);
+ rect.attr('width', owidth * scaleFactor + '%');
+ text.attr('x', nx);
+ text.attr('y', ny + 12);
+ thisObj._adjustTextSizeForNode(g);
+ }
+ });
+ }
+ }
+
+ _enableInfo() {
+ this.selected = null;
+ let thisObj = this;
+ this.svg.find('g').on('mouseenter', function(e) {
+ if (thisObj.selected) {
+ thisObj.selected.css('stroke-width', '0');
+ }
+ // Mark current node.
+ let g = $(this);
+ thisObj.selected = g;
+ g.css('stroke', 'black').css('stroke-width', '0.5');
+
+ // Parse title.
+ let title = g.find('title').text();
+ let methodAndInfo = title.split(' | ');
+ thisObj.svg.find(`#info_text_${thisObj.id}`).text(methodAndInfo[0]);
+
+ // Parse percentage.
+ // '/system/lib64/libhwbinder.so (4 events: 0.28%)'
+ let regexp = /.* \(.*:\s+(.*)\)/g;
+ let match = regexp.exec(methodAndInfo[1]);
+ let percentage = '';
+ if (match && match.length > 1) {
+ percentage = match[1];
+ }
+ thisObj.svg.find(`#percent_text_${thisObj.id}`).text(percentage);
+ });
+ }
+
+ _adjustTextSizeOnResize() {
+ function throttle(callback) {
+ let running = false;
+ return function() {
+ if (!running) {
+ running = true;
+ window.requestAnimationFrame(function () {
+ callback();
+ running = false;
+ });
+ }
+ };
+ }
+ $(window).resize(throttle(() => this._adjustTextSize()));
+ }
+}
+
+function initGlobalObjects() {
+ gTabs = new TabManager($('div#report_content'));
+ let recordData = $('#record_data').text();
+ gRecordInfo = JSON.parse(recordData);
+ gProcesses = gRecordInfo.processNames;
+ gThreads = gRecordInfo.threadNames;
+ gLibList = gRecordInfo.libList;
+ gFunctionMap = gRecordInfo.functionMap;
+ gSampleInfo = gRecordInfo.sampleInfo;
+}
+
+function createTabs() {
+ gTabs.addTab('Chart Statistics', new ChartStatTab());
+ gTabs.addTab('Sample Table', new SampleTableTab());
+ gTabs.addTab('Flamegraph', new FlameGraphTab());
+ gTabs.draw();
+}
+
+let gTabs;
+let gRecordInfo;
+let gProcesses;
+let gThreads;
+let gLibList;
+let gFunctionMap;
+let gSampleInfo;
+
+initGlobalObjects();
+createTabs();
+
+});
\ No newline at end of file
diff --git a/report_html.py b/report_html.py
new file mode 100644
index 0000000..91b3ef9
--- /dev/null
+++ b/report_html.py
@@ -0,0 +1,565 @@
+#!/usr/bin/python
+#
+# Copyright (C) 2017 The Android Open Source Project
+#
+# 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.
+#
+
+import argparse
+import datetime
+import json
+import os
+import subprocess
+import sys
+import tempfile
+
+from simpleperf_report_lib import ReportLib
+from utils import *
+
+
+class HtmlWriter(object):
+
+ def __init__(self, output_path):
+ self.fh = open(output_path, 'w')
+ self.tag_stack = []
+
+ def close(self):
+ self.fh.close()
+
+ def open_tag(self, tag, **attrs):
+ attr_str = ''
+ for key in attrs.keys():
+ attr_str += ' %s="%s"' % (key, attrs[key])
+ self.fh.write('<%s%s>' % (tag, attr_str))
+ self.tag_stack.append(tag)
+ return self
+
+ def close_tag(self, tag=None):
+ if tag:
+ assert tag == self.tag_stack[-1]
+ self.fh.write('</%s>\n' % self.tag_stack.pop())
+
+ def add(self, text):
+ self.fh.write(text)
+ return self
+
+ def add_file(self, file_path):
+ file_path = os.path.join(get_script_dir(), file_path)
+ with open(file_path, 'r') as f:
+ self.add(f.read())
+ return self
+
+
+class EventScope(object):
+
+ def __init__(self, name):
+ self.name = name
+ self.processes = {} # map from pid to ProcessScope
+ self.sample_count = 0
+ self.event_count = 0
+
+ def get_process(self, pid):
+ process = self.processes.get(pid)
+ if not process:
+ process = self.processes[pid] = ProcessScope(pid)
+ return process
+
+ def get_sample_info(self):
+ result = {}
+ result['eventName'] = self.name
+ result['eventCount'] = self.event_count
+ result['processes'] = [process.get_sample_info() for process in self.processes.values()]
+ return result
+
+
+class ProcessScope(object):
+
+ def __init__(self, pid):
+ self.pid = pid
+ self.name = ''
+ self.event_count = 0
+ self.threads = {} # map from tid to ThreadScope
+
+ def get_thread(self, tid, thread_name):
+ thread = self.threads.get(tid)
+ if not thread:
+ thread = self.threads[tid] = ThreadScope(tid)
+ thread.name = thread_name
+ if self.pid == tid:
+ self.name = thread_name
+ return thread
+
+ def get_sample_info(self):
+ result = {}
+ result['pid'] = self.pid
+ result['eventCount'] = self.event_count
+ result['threads'] = [thread.get_sample_info() for thread in self.threads.values()]
+ return result
+
+
+class ThreadScope(object):
+
+ def __init__(self, tid):
+ self.tid = tid
+ self.name = ''
+ self.event_count = 0
+ self.libs = {} # map from libId to LibScope
+
+ def add_callstack(self, event_count, callstack):
+ """ callstack is a list of (lib_id, func_id) pairs.
+ For each i > 0, callstack[i] calls callstack[i-1]."""
+ # When a callstack contains recursive function, we should only add event count
+ # and callchain for each recursive function once.
+ hit_func_ids = {}
+ for i in range(len(callstack)):
+ lib_id, func_id = callstack[i]
+ lib = self.libs.get(lib_id)
+ if not lib:
+ lib = self.libs[lib_id] = LibScope(lib_id)
+ function = lib.get_function(func_id)
+ if i == 0:
+ lib.event_count += event_count
+ function.sample_count += 1
+ if func_id in hit_func_ids:
+ continue
+ hit_func_ids[func_id] = True
+ function.add_reverse_callchain(callstack, i + 1, len(callstack), event_count)
+
+ hit_func_ids = {}
+ for i in range(len(callstack) - 1, -1, -1):
+ lib_id, func_id = callstack[i]
+ if func_id in hit_func_ids:
+ continue
+ hit_func_ids[func_id] = True
+ lib = self.libs.get(lib_id)
+ lib.get_function(func_id).add_callchain(callstack, i - 1, -1, event_count)
+
+ def get_sample_info(self):
+ result = {}
+ result['tid'] = self.tid
+ result['eventCount'] = self.event_count
+ result['libs'] = [lib.gen_sample_info() for lib in self.libs.values()]
+ return result
+
+
+class LibScope(object):
+
+ def __init__(self, lib_id):
+ self.lib_id = lib_id
+ self.event_count = 0
+ self.functions = {} # map from func_id to FunctionScope.
+
+ def get_function(self, func_id):
+ function = self.functions.get(func_id)
+ if not function:
+ function = self.functions[func_id] = FunctionScope(func_id)
+ return function
+
+ def gen_sample_info(self):
+ result = {}
+ result['libId'] = self.lib_id
+ result['eventCount'] = self.event_count
+ result['functions'] = [func.gen_sample_info() for func in self.functions.values()]
+ return result
+
+
+class FunctionScope(object):
+
+ def __init__(self, func_id):
+ self.sample_count = 0
+ self.call_graph = CallNode(func_id)
+ self.reverse_call_graph = CallNode(func_id)
+
+ def add_callchain(self, callchain, start, end, event_count):
+ node = self.call_graph
+ for i in range(start, end, -1):
+ node = node.get_child(callchain[i][1])
+ node.event_count += event_count
+
+ def add_reverse_callchain(self, callchain, start, end, event_count):
+ node = self.reverse_call_graph
+ for i in range(start, end):
+ node = node.get_child(callchain[i][1])
+ node.event_count += event_count
+
+ def update_subtree_event_count(self):
+ a = self.call_graph.update_subtree_event_count()
+ b = self.reverse_call_graph.update_subtree_event_count()
+ return max(a, b)
+
+ def limit_callchain_percent(self, min_callchain_percent):
+ min_limit = min_callchain_percent * 0.01 * self.call_graph.subtree_event_count
+ self.call_graph.cut_edge(min_limit)
+
+ def hit_function(self, func_id_set):
+ self.call_graph.hit_function(func_id_set)
+ self.reverse_call_graph.hit_function(func_id_set)
+
+ def gen_sample_info(self):
+ result = {}
+ result['c'] = self.sample_count
+ result['g'] = self.call_graph.gen_sample_info()
+ result['rg'] = self.reverse_call_graph.gen_sample_info()
+ return result
+
+
+class CallNode(object):
+
+ def __init__(self, func_id):
+ self.event_count = 0
+ self.subtree_event_count = 0
+ self.func_id = func_id
+ self.children = {} # map from func_id to CallNode
+
+ def get_child(self, func_id):
+ child = self.children.get(func_id)
+ if not child:
+ child = self.children[func_id] = CallNode(func_id)
+ return child
+
+ def update_subtree_event_count(self):
+ self.subtree_event_count = self.event_count
+ for child in self.children.values():
+ self.subtree_event_count += child.update_subtree_event_count()
+ return self.subtree_event_count
+
+ def cut_edge(self, min_limit):
+ to_del_children = []
+ for key in self.children:
+ child = self.children[key]
+ if child.subtree_event_count < min_limit:
+ to_del_children.append(key)
+ else:
+ child.cut_edge(min_limit)
+ for key in to_del_children:
+ del self.children[key]
+
+ def hit_function(self, func_id_set):
+ func_id_set.add(self.func_id)
+ for child in self.children.values():
+ child.hit_function(func_id_set)
+
+ def gen_sample_info(self):
+ result = {}
+ result['e'] = self.event_count
+ result['s'] = self.subtree_event_count
+ result['f'] = self.func_id
+ result['c'] = [child.gen_sample_info() for child in self.children.values()]
+ return result
+
+
+class LibSet(object):
+
+ def __init__(self):
+ self.libs = {}
+
+ def get_lib_id(self, lib_name):
+ lib_id = self.libs.get(lib_name)
+ if lib_id is None:
+ lib_id = len(self.libs)
+ self.libs[lib_name] = lib_id
+ return lib_id
+
+
+class FunctionSet(object):
+
+ def __init__(self):
+ self.functions = {}
+
+ def get_func_id(self, lib_id, func_name):
+ key = (lib_id, func_name)
+ func_id = self.functions.get(key)
+ if func_id is None:
+ func_id = len(self.functions)
+ self.functions[key] = func_id
+ return func_id
+
+
+class RecordData(object):
+
+ """RecordData reads perf.data, and generates data used by report.js in json format.
+ All generated items are listed as below:
+ 1. recordTime: string
+ 2. machineType: string
+ 3. androidVersion: string
+ 4. recordCmdline: string
+ 5. totalSamples: int
+ 6. processNames: map from pid to processName.
+ 7. threadNames: map from tid to threadName.
+ 8. libList: an array of libNames, indexed by libId.
+ 9. functionMap: map from functionId to [libId, functionName].
+ 10. sampleInfo = [eventInfo]
+ eventInfo = {
+ eventName
+ eventCount
+ processes: [processInfo]
+ }
+ processInfo = {
+ pid
+ eventCount
+ threads: [threadInfo]
+ }
+ threadInfo = {
+ tid
+ eventCount
+ libs: [libInfo],
+ }
+ libInfo = {
+ libId,
+ eventCount,
+ functions: [funcInfo]
+ }
+ funcInfo = {
+ c: sampleCount
+ g: callGraph
+ rg: reverseCallgraph
+ }
+ callGraph and reverseCallGraph are both of type CallNode.
+ callGraph shows how a function calls other functions.
+ reverseCallGraph shows how a function is called by other functions.
+ CallNode {
+ e: selfEventCount
+ s: subTreeEventCount
+ f: functionId
+ c: [CallNode] # children
+ }
+ """
+
+ def __init__(self, record_file, min_func_percent, min_callchain_percent):
+ self._load_record_file(record_file)
+ self._limit_percents(min_func_percent, min_callchain_percent)
+
+ def _load_record_file(self, record_file):
+ lib = ReportLib()
+ lib.ShowIpForUnknownSymbol()
+ lib.SetRecordFile(record_file)
+ self.meta_info = lib.MetaInfo()
+ self.cmdline = lib.GetRecordCmd()
+ self.arch = lib.GetArch()
+ self.events = {}
+ self.libs = LibSet()
+ self.functions = FunctionSet()
+ self.total_samples = 0
+ while True:
+ raw_sample = lib.GetNextSample()
+ if not raw_sample:
+ lib.Close()
+ break
+ raw_event = lib.GetEventOfCurrentSample()
+ symbol = lib.GetSymbolOfCurrentSample()
+ callchain = lib.GetCallChainOfCurrentSample()
+ event = self._get_event(raw_event.name)
+ self.total_samples += 1
+ event.sample_count += 1
+ event.event_count += raw_sample.period
+ process = event.get_process(raw_sample.pid)
+ process.event_count += raw_sample.period
+ thread = process.get_thread(raw_sample.tid, raw_sample.thread_comm)
+ thread.event_count += raw_sample.period
+
+ lib_id = self.libs.get_lib_id(symbol.dso_name)
+ callstack = [(lib_id, self.functions.get_func_id(lib_id, symbol.symbol_name))]
+ for i in range(callchain.nr):
+ symbol = callchain.entries[i].symbol
+ lib_id = self.libs.get_lib_id(symbol.dso_name)
+ callstack.append((lib_id, self.functions.get_func_id(lib_id, symbol.symbol_name)))
+ thread.add_callstack(raw_sample.period, callstack)
+
+ for event in self.events.values():
+ for process in event.processes.values():
+ for thread in process.threads.values():
+ for lib in thread.libs.values():
+ for funcId in lib.functions.keys():
+ function = lib.functions[funcId]
+ function.update_subtree_event_count()
+
+ def _limit_percents(self, min_func_percent, min_callchain_percent):
+ for event in self.events.values():
+ min_limit = event.event_count * min_func_percent * 0.01
+ for process in event.processes.values():
+ for thread in process.threads.values():
+ for lib in thread.libs.values():
+ for func_id in lib.functions.keys():
+ function = lib.functions[func_id]
+ if function.call_graph.subtree_event_count < min_limit:
+ del lib.functions[func_id]
+ else:
+ function.limit_callchain_percent(min_callchain_percent)
+
+ def _get_event(self, event_name):
+ if event_name not in self.events:
+ self.events[event_name] = EventScope(event_name)
+ return self.events[event_name]
+
+ def gen_record_info(self):
+ record_info = {}
+ timestamp = self.meta_info.get('timestamp')
+ if timestamp:
+ t = datetime.datetime.fromtimestamp(int(timestamp))
+ else:
+ t = datetime.datetime.now()
+ record_info['recordTime'] = t.strftime('%Y-%m-%d (%A) %H:%M:%S')
+
+ product_props = self.meta_info.get('product_props')
+ machine_type = self.arch
+ if product_props:
+ manufacturer, model, name = product_props.split(':')
+ machine_type = '%s (%s) by %s, arch %s' % (model, name, manufacturer, self.arch)
+ record_info['machineType'] = machine_type
+ record_info['androidVersion'] = self.meta_info.get('android_version', '')
+ record_info['recordCmdline'] = self.cmdline
+ record_info['totalSamples'] = self.total_samples
+ record_info['processNames'] = self._gen_process_names()
+ record_info['threadNames'] = self._gen_thread_names()
+ record_info['libList'] = self._gen_lib_list()
+ record_info['functionMap'] = self._gen_function_map()
+ record_info['sampleInfo'] = self._gen_sample_info()
+ return record_info
+
+ def _gen_process_names(self):
+ process_names = {}
+ for event in self.events.values():
+ for process in event.processes.values():
+ process_names[process.pid] = process.name
+ return process_names
+
+ def _gen_thread_names(self):
+ thread_names = {}
+ for event in self.events.values():
+ for process in event.processes.values():
+ for thread in process.threads.values():
+ thread_names[thread.tid] = thread.name
+ return thread_names
+
+ def _modify_name_for_html(self, name):
+ return name.replace('>', '>').replace('<', '<')
+
+ def _gen_lib_list(self):
+ ret = sorted(self.libs.libs.keys(), key=lambda k: self.libs.libs[k])
+ return [self._modify_name_for_html(x) for x in ret]
+
+ def _gen_function_map(self):
+ func_id_set = set()
+ for event in self.events.values():
+ for process in event.processes.values():
+ for thread in process.threads.values():
+ for lib in thread.libs.values():
+ for func_id in lib.functions.keys():
+ lib.functions[func_id].hit_function(func_id_set)
+
+ functions = self.functions.functions
+ func_map = {}
+ for key in functions:
+ func_id = functions[key]
+ if func_id in func_id_set:
+ func_map[func_id] = [key[0], self._modify_name_for_html(key[1])]
+ return func_map
+
+ def _gen_sample_info(self):
+ return [event.get_sample_info() for event in self.events.values()]
+
+
+class ReportGenerator(object):
+
+ def __init__(self, html_path):
+ self.hw = HtmlWriter(html_path)
+ self.hw.open_tag('html')
+ self.hw.open_tag('head')
+ self.hw.open_tag('link', rel='stylesheet', type='text/css',
+ href='https://code.jquery.com/ui/1.12.0/themes/smoothness/jquery-ui.css'
+ ).close_tag()
+
+ self.hw.open_tag('link', rel='stylesheet', type='text/css',
+ href='https://cdn.datatables.net/1.10.16/css/jquery.dataTables.min.css'
+ ).close_tag()
+ self.hw.open_tag('script', src='https://www.gstatic.com/charts/loader.js').close_tag()
+ self.hw.open_tag('script').add(
+ "google.charts.load('current', {'packages': ['corechart']});").close_tag()
+ self.hw.open_tag('script', src='https://code.jquery.com/jquery-3.2.1.js').close_tag()
+ self.hw.open_tag('script', src='https://code.jquery.com/ui/1.12.1/jquery-ui.js'
+ ).close_tag()
+ self.hw.open_tag('script',
+ src='https://cdn.datatables.net/1.10.16/js/jquery.dataTables.min.js').close_tag()
+ self.hw.open_tag('script',
+ src='https://cdn.datatables.net/1.10.16/js/dataTables.jqueryui.min.js').close_tag()
+ self.hw.close_tag('head')
+ self.hw.open_tag('body')
+ self.record_info = {}
+
+ def write_content_div(self):
+ self.hw.open_tag('div', id='report_content').close_tag()
+
+ def write_record_data(self, record_data):
+ self.hw.open_tag('script', id='record_data', type='application/json')
+ self.hw.add(json.dumps(record_data))
+ self.hw.close_tag()
+
+ def write_flamegraph(self, flamegraph):
+ self.hw.add(flamegraph)
+
+ def write_script(self):
+ self.hw.open_tag('script').add_file('report_html.js').close_tag()
+
+ def finish(self):
+ self.hw.close_tag('body')
+ self.hw.close_tag('html')
+ self.hw.close()
+
+
+def gen_flamegraph(record_file):
+ fd, flamegraph_path = tempfile.mkstemp()
+ os.close(fd)
+ inferno_script_path = os.path.join(get_script_dir(), 'inferno', 'inferno.py')
+ subprocess.check_call([sys.executable, inferno_script_path, '-sc', '-o', flamegraph_path,
+ '--record_file', record_file, '--embedded_flamegraph', '--no_browser'])
+ with open(flamegraph_path, 'r') as fh:
+ data = fh.read()
+ remove(flamegraph_path)
+ return data
+
+
+def main():
+ parser = argparse.ArgumentParser(description='report profiling data')
+ parser.add_argument('-i', '--record_file', default='perf.data', help="""
+ Set profiling data file to report.""")
+ parser.add_argument('-o', '--report_path', default='report.html', help="""
+ Set output html file.""")
+ parser.add_argument('--min_func_percent', default=0.01, type=float, help="""
+ Set min percentage of functions shown in the report.
+ For example, when set to 0.01, only functions taking >= 0.01%% of total
+ event count are collected in the report.""")
+ parser.add_argument('--min_callchain_percent', default=0.01, type=float, help="""
+ Set min percentage of callchains shown in the report.
+ It is used to limit nodes shown in the function flamegraph. For example,
+ when set to 0.01, only callchains taking >= 0.01%% of the event count of
+ the starting function are collected in the report.""")
+ parser.add_argument('--no_browser', action='store_true', help="Don't open report in browser.")
+ args = parser.parse_args()
+
+ record_data = RecordData(args.record_file, args.min_func_percent, args.min_callchain_percent)
+
+ report_generator = ReportGenerator(args.report_path)
+ report_generator.write_content_div()
+ report_generator.write_record_data(record_data.gen_record_info())
+ report_generator.write_script()
+ flamegraph = gen_flamegraph(args.record_file)
+ report_generator.write_flamegraph(flamegraph)
+ report_generator.finish()
+
+ if not args.no_browser:
+ open_report_in_browser(args.report_path)
+ log_info("Report generated at '%s'." % args.report_path)
+
+
+if __name__ == '__main__':
+ main()
diff --git a/simpleperf_report_lib.py b/simpleperf_report_lib.py
index 385df2f..cbe78ce 100644
--- a/simpleperf_report_lib.py
+++ b/simpleperf_report_lib.py
@@ -33,11 +33,13 @@
def _is_null(p):
+ if p:
+ return False
return ct.cast(p, ct.c_void_p).value is None
-def _char_pt(str):
- return str_to_bytes(str)
+def _char_pt(s):
+ return str_to_bytes(s)
def _char_pt_to_str(char_pt):
@@ -45,6 +47,18 @@
class SampleStruct(ct.Structure):
+ """ Instance of a sample in perf.data.
+ ip: the address of the instruction the cpu is running when the sample happens.
+ pid: process id (or thread group id) of the thread generating the sample.
+ tid: thread id.
+ thread_comm: thread name.
+ time: timestamp of a sample, the meaning is decided by simpleperf record --clockid option.
+ in_kernel: whether the instruction is in kernel space or user space.
+ cpu: the cpu generating the sample.
+ period: count of events have happened since last sample. For example, if we use
+ -e cpu-cycles, it means how many cpu-cycles have happened.
+ If we use -e cpu-clock, it means how many nanoseconds have passed.
+ """
_fields_ = [('ip', ct.c_uint64),
('pid', ct.c_uint32),
('tid', ct.c_uint32),
@@ -56,34 +70,65 @@
class EventStruct(ct.Structure):
+ """ Name of the event. """
_fields_ = [('name', ct.c_char_p)]
class MappingStruct(ct.Structure):
+ """ A mapping area in the monitored threads, like the content in /proc/<pid>/maps.
+ start: start addr in memory.
+ end: end addr in memory.
+ pgoff: offset in the mapped shared library.
+ """
_fields_ = [('start', ct.c_uint64),
('end', ct.c_uint64),
('pgoff', ct.c_uint64)]
class SymbolStruct(ct.Structure):
+ """ Symbol info of the instruction hit by a sample or a callchain entry of a sample.
+ dso_name: path of the shared library containing the instruction.
+ vaddr_in_file: virtual address of the instruction in the shared library.
+ symbol_name: name of the function containing the instruction.
+ symbol_addr: start addr of the function containing the instruction.
+ symbol_len: length of the function in the shared library.
+ mapping: the mapping area hit by the instruction.
+ """
_fields_ = [('dso_name', ct.c_char_p),
('vaddr_in_file', ct.c_uint64),
('symbol_name', ct.c_char_p),
('symbol_addr', ct.c_uint64),
+ ('symbol_len', ct.c_uint64),
('mapping', ct.POINTER(MappingStruct))]
class CallChainEntryStructure(ct.Structure):
+ """ A callchain entry of a sample.
+ ip: the address of the instruction of the callchain entry.
+ symbol: symbol info of the callchain entry.
+ """
_fields_ = [('ip', ct.c_uint64),
('symbol', SymbolStruct)]
class CallChainStructure(ct.Structure):
+ """ Callchain info of a sample.
+ nr: number of entries in the callchain.
+ entries: a pointer to an array of CallChainEntryStructure.
+
+ For example, if a sample is generated when a thread is running function C
+ with callchain function A -> function B -> function C.
+ Then nr = 2, and entries = [function B, function A].
+ """
_fields_ = [('nr', ct.c_uint32),
('entries', ct.POINTER(CallChainEntryStructure))]
class FeatureSectionStructure(ct.Structure):
+ """ A feature section in perf.data to store information like record cmd, device arch, etc.
+ data: a pointer to a buffer storing the section data.
+ data_size: data size in bytes.
+ """
_fields_ = [('data', ct.POINTER(ct.c_char)),
('data_size', ct.c_uint32)]
@@ -163,14 +208,15 @@
self._GetFeatureSection = self._lib.GetFeatureSection
self._GetFeatureSection.restype = ct.POINTER(FeatureSectionStructure)
self._instance = self._CreateReportLibFunc()
- assert(not _is_null(self._instance))
+ assert not _is_null(self._instance)
self.convert_to_str = (sys.version_info >= (3, 0))
self.meta_info = None
self.current_sample = None
+ self.record_cmd = None
def _load_dependent_lib(self):
- # As the windows dll is built with mingw we need to load "libwinpthread-1.dll".
+ # As the windows dll is built with mingw we need to load 'libwinpthread-1.dll'.
if is_windows():
self._libwinpthread = ct.CDLL(get_host_binary_path('libwinpthread-1.dll'))
@@ -183,17 +229,17 @@
def SetLogSeverity(self, log_level='info'):
""" Set log severity of native lib, can be verbose,debug,info,error,fatal."""
cond = self._SetLogSeverityFunc(self.getInstance(), _char_pt(log_level))
- self._check(cond, "Failed to set log level")
+ self._check(cond, 'Failed to set log level')
def SetSymfs(self, symfs_dir):
""" Set directory used to find symbols."""
cond = self._SetSymfsFunc(self.getInstance(), _char_pt(symfs_dir))
- self._check(cond, "Failed to set symbols directory")
+ self._check(cond, 'Failed to set symbols directory')
def SetRecordFile(self, record_file):
""" Set the path of record file, like perf.data."""
cond = self._SetRecordFileFunc(self.getInstance(), _char_pt(record_file))
- self._check(cond, "Failed to set record file")
+ self._check(cond, 'Failed to set record file')
def ShowIpForUnknownSymbol(self):
self._ShowIpForUnknownSymbolFunc(self.getInstance())
@@ -201,7 +247,7 @@
def SetKallsymsFile(self, kallsym_file):
""" Set the file path to a copy of the /proc/kallsyms file (for off device decoding) """
cond = self._SetKallsymsFileFunc(self.getInstance(), _char_pt(kallsym_file))
- self._check(cond, "Failed to set kallsyms file")
+ self._check(cond, 'Failed to set kallsyms file')
def GetNextSample(self):
psample = self._GetNextSampleFunc(self.getInstance())
@@ -217,46 +263,45 @@
def GetEventOfCurrentSample(self):
event = self._GetEventOfCurrentSampleFunc(self.getInstance())
- assert(not _is_null(event))
+ assert not _is_null(event)
if self.convert_to_str:
return EventStructUsingStr(event[0])
return event[0]
def GetSymbolOfCurrentSample(self):
symbol = self._GetSymbolOfCurrentSampleFunc(self.getInstance())
- assert(not _is_null(symbol))
+ assert not _is_null(symbol)
if self.convert_to_str:
return SymbolStructUsingStr(symbol[0])
return symbol[0]
def GetCallChainOfCurrentSample(self):
callchain = self._GetCallChainOfCurrentSampleFunc(self.getInstance())
- assert(not _is_null(callchain))
+ assert not _is_null(callchain)
if self.convert_to_str:
return CallChainStructureUsingStr(callchain[0])
return callchain[0]
def GetBuildIdForPath(self, path):
build_id = self._GetBuildIdForPathFunc(self.getInstance(), _char_pt(path))
- assert(not _is_null(build_id))
+ assert not _is_null(build_id)
return _char_pt_to_str(build_id)
def GetRecordCmd(self):
- if hasattr(self, "record_cmd"):
+ if self.record_cmd is not None:
return self.record_cmd
- self.record_cmd = None
- feature_data = self._GetFeatureSection(self.getInstance(), _char_pt("cmdline"))
+ self.record_cmd = ''
+ feature_data = self._GetFeatureSection(self.getInstance(), _char_pt('cmdline'))
if not _is_null(feature_data):
void_p = ct.cast(feature_data[0].data, ct.c_void_p)
- data_size = feature_data[0].data_size
arg_count = ct.cast(void_p, ct.POINTER(ct.c_uint32)).contents.value
void_p.value += 4
args = []
- for i in range(arg_count):
+ for _ in range(arg_count):
str_len = ct.cast(void_p, ct.POINTER(ct.c_uint32)).contents.value
void_p.value += 4
char_p = ct.cast(void_p, ct.POINTER(ct.c_char))
- current_str = ""
+ current_str = ''
for j in range(str_len):
c = bytes_to_str(char_p[j])
if c != '\0':
@@ -265,33 +310,53 @@
current_str = '"' + current_str + '"'
args.append(current_str)
void_p.value += str_len
- self.record_cmd = " ".join(args)
+ self.record_cmd = ' '.join(args)
return self.record_cmd
+ def _GetFeatureString(self, feature_name):
+ feature_data = self._GetFeatureSection(self.getInstance(), _char_pt(feature_name))
+ result = ''
+ if not _is_null(feature_data):
+ void_p = ct.cast(feature_data[0].data, ct.c_void_p)
+ str_len = ct.cast(void_p, ct.POINTER(ct.c_uint32)).contents.value
+ void_p.value += 4
+ char_p = ct.cast(void_p, ct.POINTER(ct.c_char))
+ for i in range(str_len):
+ c = bytes_to_str(char_p[i])
+ if c == '\0':
+ break
+ result += c
+ return result
+
+ def GetArch(self):
+ return self._GetFeatureString('arch')
def MetaInfo(self):
+ """ Return a string to string map stored in meta_info section in perf.data.
+ It is used to pass some short meta information.
+ """
if self.meta_info is None:
self.meta_info = {}
- feature_data = self._GetFeatureSection(self.getInstance(), _char_pt("meta_info"))
+ feature_data = self._GetFeatureSection(self.getInstance(), _char_pt('meta_info'))
if not _is_null(feature_data):
str_list = []
data = feature_data[0].data
data_size = feature_data[0].data_size
- current_str = ""
+ current_str = ''
for i in range(data_size):
c = bytes_to_str(data[i])
if c != '\0':
current_str += c
else:
str_list.append(current_str)
- current_str = ""
+ current_str = ''
for i in range(0, len(str_list), 2):
self.meta_info[str_list[i]] = str_list[i + 1]
return self.meta_info
def getInstance(self):
if self._instance is None:
- raise Exception("Instance is Closed")
+ raise Exception('Instance is Closed')
return self._instance
def _check(self, cond, failmsg):
diff --git a/test.py b/test.py
index cbdd27b..ab744d7 100644
--- a/test.py
+++ b/test.py
@@ -51,7 +51,7 @@
except:
has_google_protobuf = False
-inferno_script = "inferno.bat" if is_windows() else "./inferno.sh"
+inferno_script = os.path.join(get_script_dir(), "inferno.bat" if is_windows() else "./inferno.sh")
support_trace_offcpu = None
@@ -72,7 +72,9 @@
""" Collect testdata from ../testdata and ../demo. """
from_testdata_path = os.path.join('..', 'testdata')
from_demo_path = os.path.join('..', 'demo')
- if not os.path.isdir(from_testdata_path) or not os.path.isdir(from_demo_path):
+ from_script_testdata_path = 'script_testdata'
+ if (not os.path.isdir(from_testdata_path) or not os.path.isdir(from_demo_path) or
+ not from_script_testdata_path):
return
copy_testdata_list = ['perf_with_symbols.data', 'perf_with_trace_offcpu.data']
copy_demo_list = ['SimpleperfExamplePureJava', 'SimpleperfExampleWithNative',
@@ -85,7 +87,8 @@
shutil.copy(os.path.join(from_testdata_path, testdata), testdata_path)
for demo in copy_demo_list:
shutil.copytree(os.path.join(from_demo_path, demo), os.path.join(testdata_path, demo))
-
+ for f in os.listdir(from_script_testdata_path):
+ shutil.copy(os.path.join(from_script_testdata_path, f), testdata_path)
class TestBase(unittest.TestCase):
def run_cmd(self, args, return_output=False):
@@ -152,7 +155,7 @@
self.__class__.test_result = result
super(TestBase, self).run(result)
- def run_app_profiler(self, record_arg = "-g --duration 3 -e cpu-cycles:u",
+ def run_app_profiler(self, record_arg = "-g -f 1000 --duration 3 -e cpu-cycles:u",
build_binary_cache=True, skip_compile=False, start_activity=True,
native_lib_dir=None, profile_from_launch=False, add_arch=False):
args = ["app_profiler.py", "--app", self.package_name, "--apk", self.apk_path,
@@ -225,12 +228,11 @@
fulfilled[i] = True
self.assertEqual(len(fulfilled), sum([int(x) for x in fulfilled]), fulfilled)
- def check_inferno_report_html(self, check_entries):
- file = "report.html"
+ def check_inferno_report_html(self, check_entries, file="report.html"):
self.check_exist(file=file)
with open(file, 'r') as fh:
data = fh.read()
- fulfilled = [False for x in check_entries]
+ fulfilled = [False for _ in check_entries]
for line in data.split('\n'):
# each entry is a (function_name, min_percentage) pair.
for i, entry in enumerate(check_entries):
@@ -264,6 +266,7 @@
self.run_cmd(["report.py", "-i", "perf.data"])
self.run_cmd(["report.py", "-g"])
self.run_cmd(["report.py", "--self-kill-for-testing", "-g", "--gui"])
+ self.run_cmd(["report_html.py"])
def common_test_annotate(self):
self.run_cmd(["annotate.py", "-h"])
@@ -279,7 +282,7 @@
self.run_cmd(["report_sample.py"])
output = self.run_cmd(["report_sample.py", "perf.data"], return_output=True)
self.check_strings_in_content(output, check_strings)
- self.run_app_profiler(record_arg="-g --duration 3 -e cpu-cycles:u --no-dump-symbols")
+ self.run_app_profiler(record_arg="-g -f 1000 --duration 3 -e cpu-cycles:u --no-dump-symbols")
output = self.run_cmd(["report_sample.py", "--symfs", "binary_cache"], return_output=True)
self.check_strings_in_content(output, check_strings)
@@ -403,6 +406,21 @@
self.run_cmd([inferno_script, "-sc"])
self.check_inferno_report_html(
[('com.example.simpleperf.simpleperfexamplepurejava.MainActivity$1.run()', 80)])
+ self.run_cmd([inferno_script, "-sc", "-o", "report2.html"])
+ self.check_inferno_report_html(
+ [('com.example.simpleperf.simpleperfexamplepurejava.MainActivity$1.run()', 80)],
+ "report2.html")
+ remove("report2.html")
+
+ def test_inferno_in_another_dir(self):
+ test_dir = 'inferno_testdir'
+ saved_dir = os.getcwd()
+ remove(test_dir)
+ os.mkdir(test_dir)
+ os.chdir(test_dir)
+ self.run_cmd([inferno_script])
+ os.chdir(saved_dir)
+ remove(test_dir)
class TestExamplePureJavaRoot(TestExampleBase):
@@ -425,7 +443,7 @@
".SleepActivity")
def test_smoke(self):
- self.run_app_profiler(record_arg = "-g --duration 3 -e cpu-cycles:u --trace-offcpu")
+ self.run_app_profiler(record_arg="-g -f 1000 --duration 3 -e cpu-cycles:u --trace-offcpu")
self.run_cmd(["report.py", "-g", "-o", "report.txt"])
self.check_strings_in_file("report.txt",
["com.example.simpleperf.simpleperfexamplepurejava.SleepActivity$1.run()",
@@ -530,7 +548,7 @@
".SleepActivity")
def test_smoke(self):
- self.run_app_profiler(record_arg = "-g --duration 3 -e cpu-cycles:u --trace-offcpu")
+ self.run_app_profiler(record_arg="-g -f 1000 --duration 3 -e cpu-cycles:u --trace-offcpu")
self.run_cmd(["report.py", "-g", "--comms", "SleepThread", "-o", "report.txt"])
self.check_strings_in_file("report.txt",
["SleepThread(void*)",
@@ -578,8 +596,8 @@
[("MixActivity.java", 80, 0),
("run", 80, 0),
("line 26", 20, 0),
- ("native-lib.cpp", 10, 0),
- ("line 40", 10, 0)])
+ ("native-lib.cpp", 5, 0),
+ ("line 40", 5, 0)])
self.run_cmd([inferno_script, "-sc"])
@@ -688,7 +706,7 @@
".SleepActivity")
def test_smoke(self):
- self.run_app_profiler(record_arg = "-g --duration 3 -e cpu-cycles:u --trace-offcpu")
+ self.run_app_profiler(record_arg="-g -f 1000 --duration 3 -e cpu-cycles:u --trace-offcpu")
self.run_cmd(["report.py", "-g", "-o", "report.txt"])
self.check_strings_in_file("report.txt",
["void com.example.simpleperf.simpleperfexampleofkotlin.SleepActivity$createRunSleepThread$1.run()",
@@ -724,7 +742,7 @@
return
remove("perf.data")
self.run_cmd(["app_profiler.py", "-np", "surfaceflinger",
- "-r", "-g --duration 3 -e cpu-cycles:u"])
+ "-r", "-g -f 1000 --duration 3 -e cpu-cycles:u"])
self.run_cmd(["report.py", "-g", "-o", "report.txt"])
@@ -763,7 +781,7 @@
build_id = self.report_lib.GetBuildIdForPath('/data/t2')
self.assertEqual(build_id, '0x70f1fe24500fc8b0d9eb477199ca1ca21acca4de')
- def test_symbol_addr(self):
+ def test_symbol(self):
found_func2 = False
while self.report_lib.GetNextSample():
sample = self.report_lib.GetCurrentSample()
@@ -771,6 +789,7 @@
if symbol.symbol_name == 'func2(int, int)':
found_func2 = True
self.assertEqual(symbol.symbol_addr, 0x4004ed)
+ self.assertEqual(symbol.symbol_len, 0x14)
self.assertTrue(found_func2)
def test_sample(self):
@@ -837,6 +856,139 @@
self.run_cmd(['run_simpleperf_on_device.py', 'list', '--show-features'])
+class TestTools(unittest.TestCase):
+ def test_addr2nearestline(self):
+ binary_cache_path = 'testdata'
+ test_map = {
+ '/simpleperf_runtest_two_functions_arm64': [
+ {
+ 'func_addr': 0x668,
+ 'addr': 0x668,
+ 'source': 'system/extras/simpleperf/runtest/two_functions.cpp:20',
+ },
+ {
+ 'func_addr': 0x668,
+ 'addr': 0x6a4,
+ 'source': """system/extras/simpleperf/runtest/two_functions.cpp:7
+ system/extras/simpleperf/runtest/two_functions.cpp:22""",
+ },
+ ],
+ '/simpleperf_runtest_two_functions_arm': [
+ {
+ 'func_addr': 0x784,
+ 'addr': 0x7b0,
+ 'source': """system/extras/simpleperf/runtest/two_functions.cpp:14
+ system/extras/simpleperf/runtest/two_functions.cpp:23""",
+ },
+ {
+ 'func_addr': 0x784,
+ 'addr': 0x7d0,
+ 'source': """system/extras/simpleperf/runtest/two_functions.cpp:15
+ system/extras/simpleperf/runtest/two_functions.cpp:23""",
+ }
+ ],
+ '/simpleperf_runtest_two_functions_x86_64': [
+ {
+ 'func_addr': 0x840,
+ 'addr': 0x840,
+ 'source': 'system/extras/simpleperf/runtest/two_functions.cpp:7',
+ },
+ {
+ 'func_addr': 0x920,
+ 'addr': 0x94a,
+ 'source': """system/extras/simpleperf/runtest/two_functions.cpp:7
+ system/extras/simpleperf/runtest/two_functions.cpp:22""",
+ }
+ ],
+ '/simpleperf_runtest_two_functions_x86': [
+ {
+ 'func_addr': 0x6d0,
+ 'addr': 0x6da,
+ 'source': 'system/extras/simpleperf/runtest/two_functions.cpp:14',
+ },
+ {
+ 'func_addr': 0x710,
+ 'addr': 0x749,
+ 'source': """system/extras/simpleperf/runtest/two_functions.cpp:8
+ system/extras/simpleperf/runtest/two_functions.cpp:22""",
+ }
+ ],
+ }
+ addr2line = Addr2Nearestline(None, binary_cache_path)
+ for dso_path in test_map:
+ test_addrs = test_map[dso_path]
+ for test_addr in test_addrs:
+ addr2line.add_addr(dso_path, test_addr['func_addr'], test_addr['addr'])
+ addr2line.convert_addrs_to_lines()
+ for dso_path in test_map:
+ dso = addr2line.get_dso(dso_path)
+ self.assertTrue(dso is not None)
+ test_addrs = test_map[dso_path]
+ for test_addr in test_addrs:
+ source_str = test_addr['source']
+ expected_source = []
+ for line in source_str.split('\n'):
+ items = line.split(':')
+ expected_source.append((items[0].strip(), int(items[1])))
+ actual_source = dso.get_addr_source(test_addr['addr'])
+ self.assertTrue(actual_source is not None)
+ self.assertEqual(len(actual_source), len(expected_source))
+ for i in range(len(expected_source)):
+ actual_file_id, actual_line = actual_source[i]
+ actual_file_path = addr2line.get_file_path(actual_file_id)
+ self.assertEqual(actual_file_path, expected_source[i][0])
+ self.assertEqual(actual_line, expected_source[i][1])
+
+ def test_objdump(self):
+ binary_cache_path = 'testdata'
+ test_map = {
+ '/simpleperf_runtest_two_functions_arm64': {
+ 'start_addr': 0x668,
+ 'len': 116,
+ 'expected_items': [
+ ('main():', 0),
+ ('system/extras/simpleperf/runtest/two_functions.cpp:20', 0),
+ (' 694: add x20, x20, #0x6de', 0x694),
+ ],
+ },
+ '/simpleperf_runtest_two_functions_arm': {
+ 'start_addr': 0x784,
+ 'len': 80,
+ 'expected_items': [
+ ('main():', 0),
+ ('system/extras/simpleperf/runtest/two_functions.cpp:20', 0),
+ (' 7ae: bne.n 7a6 <main+0x22>', 0x7ae),
+ ],
+ },
+ '/simpleperf_runtest_two_functions_x86_64': {
+ 'start_addr': 0x920,
+ 'len': 201,
+ 'expected_items': [
+ ('main():', 0),
+ ('system/extras/simpleperf/runtest/two_functions.cpp:20', 0),
+ (' 96e: mov %edx,(%rbx,%rax,4)', 0x96e),
+ ],
+ },
+ '/simpleperf_runtest_two_functions_x86': {
+ 'start_addr': 0x710,
+ 'len': 98,
+ 'expected_items': [
+ ('main():', 0),
+ ('system/extras/simpleperf/runtest/two_functions.cpp:20', 0),
+ (' 748: cmp $0x5f5e100,%ebp', 0x748),
+ ],
+ },
+ }
+ objdump = Objdump(None, binary_cache_path)
+ for dso_path in test_map:
+ dso_info = test_map[dso_path]
+ disassemble_code = objdump.disassemble_code(dso_path, dso_info['start_addr'],
+ dso_info['len'])
+ self.assertTrue(disassemble_code)
+ for item in dso_info['expected_items']:
+ self.assertTrue(item in disassemble_code)
+
+
def main():
os.chdir(get_script_dir())
build_testdata()
@@ -846,4 +998,4 @@
unittest.main(failfast=True)
if __name__ == '__main__':
- main()
\ No newline at end of file
+ main()
diff --git a/testdata/simpleperf_runtest_two_functions_arm b/testdata/simpleperf_runtest_two_functions_arm
new file mode 100755
index 0000000..66b34bb
--- /dev/null
+++ b/testdata/simpleperf_runtest_two_functions_arm
Binary files differ
diff --git a/testdata/simpleperf_runtest_two_functions_arm64 b/testdata/simpleperf_runtest_two_functions_arm64
new file mode 100755
index 0000000..49d7613
--- /dev/null
+++ b/testdata/simpleperf_runtest_two_functions_arm64
Binary files differ
diff --git a/testdata/simpleperf_runtest_two_functions_x86 b/testdata/simpleperf_runtest_two_functions_x86
new file mode 100755
index 0000000..bb0123d
--- /dev/null
+++ b/testdata/simpleperf_runtest_two_functions_x86
Binary files differ
diff --git a/testdata/simpleperf_runtest_two_functions_x86_64 b/testdata/simpleperf_runtest_two_functions_x86_64
new file mode 100755
index 0000000..71cfe3b
--- /dev/null
+++ b/testdata/simpleperf_runtest_two_functions_x86_64
Binary files differ
diff --git a/update.py b/update.py
index b147fbd..cf09a81 100755
--- a/update.py
+++ b/update.py
@@ -118,7 +118,7 @@
for d in ['bin', 'doc', 'inferno', 'testdata']:
if os.path.isdir(d):
result.append(d)
- result += glob.glob('*.py')
+ result += glob.glob('*.py') + glob.glob('*.js')
result.remove('update.py')
result += ['inferno.sh', 'inferno.bat']
return result
@@ -134,9 +134,7 @@
# Need to check again because git won't remove directories if they have
# non-git files in them.
- for prebuilt in old_prebuilts:
- if os.path.exists(prebuilt):
- shutil.rmtree(prebuilt)
+ check_call(['rm', '-rf'] + old_prebuilts)
def install_new_release(branch, build):
@@ -175,6 +173,8 @@
shutil.rmtree('scripts')
check_call(['mv'] + glob.glob('demo/*') + ['testdata'])
shutil.rmtree('demo')
+ check_call(['mv'] + glob.glob('script_testdata/*') + ['testdata'])
+ shutil.rmtree('script_testdata')
def install_repo_prop(branch, build):
diff --git a/utils.py b/utils.py
index 77f9f72..4404c21 100644
--- a/utils.py
+++ b/utils.py
@@ -30,13 +30,19 @@
def get_script_dir():
return os.path.dirname(os.path.realpath(__file__))
-
def is_windows():
return sys.platform == 'win32' or sys.platform == 'cygwin'
def is_darwin():
return sys.platform == 'darwin'
+def get_platform():
+ if is_windows():
+ return 'windows'
+ if is_darwin():
+ return 'darwin'
+ return 'linux'
+
def is_python3():
return sys.version_info >= (3, 0)
@@ -65,8 +71,8 @@
def str_to_bytes(str):
if not is_python3():
return str
- # In python 3, str are wide strings whereas the C api expects 8 bit strings, hence we have to convert
- # For now using utf-8 as the encoding.
+ # In python 3, str are wide strings whereas the C api expects 8 bit strings,
+ # hence we have to convert. For now using utf-8 as the encoding.
return str.encode('utf-8')
def bytes_to_str(bytes):
@@ -117,57 +123,91 @@
except:
return False
-expected_tool_paths = {
+DEFAULT_NDK_PATH = {
+ 'darwin': 'Library/Android/sdk/ndk-bundle',
+ 'linux': 'Android/Sdk/ndk-bundle',
+ 'windows': 'AppData/Local/Android/sdk/ndk-bundle',
+}
+
+EXPECTED_TOOLS = {
'adb': {
+ 'is_binutils': False,
'test_option': 'version',
- 'darwin': [(True, 'Library/Android/sdk/platform-tools/adb'),
- (False, '../../platform-tools/adb')],
- 'linux': [(True, 'Android/Sdk/platform-tools/adb'),
- (False, '../../platform-tools/adb')],
- 'windows': [(True, 'AppData/Local/Android/sdk/platform-tools/adb'),
- (False, '../../platform-tools/adb')],
+ 'path_in_ndk': '../platform-tools/adb',
},
'readelf': {
- 'test_option': '--help',
- 'darwin': [(True, 'Library/Android/sdk/ndk-bundle/toolchains/aarch64-linux-android-4.9/prebuilt/darwin-x86_64/bin/aarch64-linux-android-readelf'),
- (False, '../toolchains/aarch64-linux-android-4.9/prebuilt/darwin-x86_64/bin/aarch64-linux-android-readelf')],
- 'linux': [(True, 'Android/Sdk/ndk-bundle/toolchains/aarch64-linux-android-4.9/prebuilt/linux-x86_64/bin/aarch64-linux-android-readelf'),
- (False, '../toolchains/aarch64-linux-android-4.9/prebuilt/linux-x86_64/bin/aarch64-linux-android-readelf')],
- 'windows': [(True, 'AppData/Local/Android/sdk/ndk-bundle/toolchains/aarch64-linux-android-4.9/prebuilt/windows-x86_64/bin/aarch64-linux-android-readelf'),
- (False, '../toolchains/aarch64-linux-android-4.9/prebuilt/windows-x86_64/bin/aarch64-linux-android-readelf')],
+ 'is_binutils': True,
+ 'accept_tool_without_arch': True,
},
'addr2line': {
- 'test_option': '--help',
- 'darwin': [(True, 'Library/Android/sdk/ndk-bundle/toolchains/aarch64-linux-android-4.9/prebuilt/darwin-x86_64/bin/aarch64-linux-android-addr2line'),
- (False, '../toolchains/aarch64-linux-android-4.9/prebuilt/darwin-x86_64/bin/aarch64-linux-android-addr2line')],
- 'linux': [(True, 'Android/Sdk/ndk-bundle/toolchains/aarch64-linux-android-4.9/prebuilt/linux-x86_64/bin/aarch64-linux-android-addr2line'),
- (False, '../toolchains/aarch64-linux-android-4.9/prebuilt/linux-x86_64/bin/aarch64-linux-android-addr2line')],
- 'windows': [(True, 'AppData/Local/Android/sdk/ndk-bundle/toolchains/aarch64-linux-android-4.9/prebuilt/windows-x86_64/bin/aarch64-linux-android-addr2line'),
- (False, '../toolchains/aarch64-linux-android-4.9/prebuilt/windows-x86_64/bin/aarch64-linux-android-addr2line')],
+ 'is_binutils': True,
+ 'accept_tool_without_arch': True
+ },
+ 'objdump': {
+ 'is_binutils': True,
},
}
-def find_tool_path(toolname):
- if toolname not in expected_tool_paths:
+def _get_binutils_path_in_ndk(toolname, arch, platform):
+ if not arch:
+ arch = 'arm64'
+ if arch == 'arm64':
+ name = 'aarch64-linux-android-' + toolname
+ path = 'toolchains/aarch64-linux-android-4.9/prebuilt/%s-x86_64/bin/%s' % (platform, name)
+ elif arch == 'arm':
+ name = 'arm-linux-androideabi-' + toolname
+ path = 'toolchains/arm-linux-androideabi-4.9/prebuilt/%s-x86_64/bin/%s' % (platform, name)
+ elif arch == 'x86_64':
+ name = 'x86_64-linux-android-' + toolname
+ path = 'toolchains/x86_64-4.9/prebuilt/%s-x86_64/bin/%s' % (platform, name)
+ elif arch == 'x86':
+ name = 'i686-linux-android-' + toolname
+ path = 'toolchains/x86-4.9/prebuilt/%s-x86_64/bin/%s' % (platform, name)
+ else:
+ log_fatal('unexpected arch %s' % arch)
+ return (name, path)
+
+def find_tool_path(toolname, ndk_path=None, arch=None):
+ if toolname not in EXPECTED_TOOLS:
return None
- test_option = expected_tool_paths[toolname]['test_option']
- if is_executable_available(toolname, test_option):
- return toolname
- platform = 'linux'
- if is_windows():
- platform = 'windows'
- elif is_darwin():
- platform = 'darwin'
- paths = expected_tool_paths[toolname][platform]
- home = os.environ.get('HOMEPATH') if is_windows() else os.environ.get('HOME')
- for (relative_to_home, path) in paths:
- path = path.replace('/', os.sep)
- if relative_to_home:
- path = os.path.join(home, path)
- else:
- path = os.path.join(get_script_dir(), path)
+ tool_info = EXPECTED_TOOLS[toolname]
+ is_binutils = tool_info['is_binutils']
+ test_option = tool_info.get('test_option', '--help')
+ platform = get_platform()
+ if is_binutils:
+ toolname_with_arch, path_in_ndk = _get_binutils_path_in_ndk(toolname, arch, platform)
+ else:
+ toolname_with_arch = toolname
+ path_in_ndk = tool_info['path_in_ndk']
+ path_in_ndk = path_in_ndk.replace('/', os.sep)
+
+ # 1. Find tool in the given ndk path.
+ if ndk_path:
+ path = os.path.join(ndk_path, path_in_ndk)
if is_executable_available(path, test_option):
return path
+
+ # 2. Find tool in the ndk directory containing simpleperf scripts.
+ path = os.path.join('..', path_in_ndk)
+ if is_executable_available(path, test_option):
+ return path
+
+ # 3. Find tool in the default ndk installation path.
+ home = os.environ.get('HOMEPATH') if is_windows() else os.environ.get('HOME')
+ if home:
+ default_ndk_path = os.path.join(home, DEFAULT_NDK_PATH[platform].replace('/', os.sep))
+ path = os.path.join(default_ndk_path, path_in_ndk)
+ if is_executable_available(path, test_option):
+ return path
+
+ # 4. Find tool in $PATH.
+ if is_executable_available(toolname_with_arch, test_option):
+ return toolname_with_arch
+
+ # 5. Find tool without arch in $PATH.
+ if is_binutils and tool_info.get('accept_tool_without_arch'):
+ if is_executable_available(toolname, test_option):
+ return toolname
return None
@@ -294,4 +334,291 @@
elif os.path.isdir(dir_or_file):
shutil.rmtree(dir_or_file, ignore_errors=True)
+
+def open_report_in_browser(report_path):
+ import webbrowser
+ try:
+ # Try to open the report with Chrome
+ browser_key = ''
+ for key, _ in webbrowser._browsers.items():
+ if 'chrome' in key:
+ browser_key = key
+ browser = webbrowser.get(browser_key)
+ browser.open(report_path, new=0, autoraise=True)
+ except:
+ # webbrowser.get() doesn't work well on darwin/windows.
+ webbrowser.open_new_tab(report_path)
+
+
+def find_real_dso_path(dso_path_in_record_file, binary_cache_path):
+ """ Given the path of a shared library in perf.data, find its real path in the file system. """
+ if dso_path_in_record_file[0] != '/' or dso_path_in_record_file == '//anon':
+ return None
+ if binary_cache_path:
+ tmp_path = os.path.join(binary_cache_path, dso_path_in_record_file[1:])
+ if os.path.isfile(tmp_path):
+ return tmp_path
+ if os.path.isfile(dso_path_in_record_file):
+ return dso_path_in_record_file
+ return None
+
+def get_arch_of_dso_path(readelf_path, dso_path):
+ try:
+ output = subprocess.check_output([readelf_path, '-h', dso_path])
+ if output.find('AArch64') != -1:
+ return 'arm64'
+ if output.find('ARM') != -1:
+ return 'arm'
+ if output.find('X86-64') != -1:
+ return 'x86_64'
+ if output.find('80386') != -1:
+ return 'x86'
+ except subprocess.CalledProcessError:
+ pass
+ return 'unknown'
+
+
+class Addr2Nearestline(object):
+ """ Use addr2line to convert (dso_path, func_addr, addr) to (source_file, line) pairs.
+ For instructions generated by C++ compilers without a matching statement in source code
+ (like stack corruption check, switch optimization, etc.), addr2line can't generate
+ line information. However, we want to assign the instruction to the nearest line before
+ the instruction (just like objdump -dl). So we use below strategy:
+ Instead of finding the exact line of the instruction in an address, we find the nearest
+ line to the instruction in an address. If an address doesn't have a line info, we find
+ the line info of address - 1. If still no line info, then use address - 2, address - 3,
+ etc.
+
+ The implementation steps are as below:
+ 1. Collect all (dso_path, func_addr, addr) requests before converting. This saves the
+ times to call addr2line.
+ 2. Convert addrs to (source_file, line) pairs for each dso_path as below:
+ 2.1 Check if the dso_path has .debug_line. If not, omit its conversion.
+ 2.2 Get arch of the dso_path, and decide the addr_step for it. addr_step is the step we
+ change addr each time. For example, since instructions of arm64 are all 4 bytes long,
+ addr_step for arm64 can be 4.
+ 2.3 Use addr2line to find line info for each addr in the dso_path.
+ 2.4 For each addr without line info, use addr2line to find line info for
+ range(addr - addr_step, addr - addr_step * 4 - 1, -addr_step).
+ 2.5 For each addr without line info, use addr2line to find line info for
+ range(addr - addr_step * 5, addr - addr_step * 128 - 1, -addr_step).
+ (128 is a guess number. A nested switch statement in
+ system/core/demangle/Demangler.cpp has >300 bytes without line info in arm64.)
+ """
+ class Dso(object):
+ """ Info of a dynamic shared library.
+ addrs: a map from address to Addr object in this dso.
+ """
+ def __init__(self):
+ self.addrs = {}
+
+ def get_addr_source(self, addr):
+ return self.addrs[addr].source_lines
+
+ class Addr(object):
+ """ Info of an addr request.
+ func_addr: start_addr of the function containing addr.
+ source_lines: a list of [file_id, line_number] for addr.
+ source_lines[:-1] are all for inlined functions.
+ """
+ def __init__(self, func_addr):
+ self.func_addr = func_addr
+ self.source_lines = None
+
+ def __init__(self, ndk_path, binary_cache_path):
+ self.addr2line_path = find_tool_path('addr2line', ndk_path)
+ if not self.addr2line_path:
+ log_exit("Can't find addr2line. Please set ndk path by --ndk-path option.")
+ self.readelf_path = find_tool_path('readelf', ndk_path)
+ if not self.readelf_path:
+ log_exit("Can't find readelf. Please set ndk path by --ndk-path option.")
+ self.dso_map = {} # map from dso_path to Dso.
+ self.binary_cache_path = binary_cache_path
+ # Saving file names for each addr takes a lot of memory. So we store file ids in Addr,
+ # and provide data structures connecting file id and file name here.
+ self.file_name_to_id = {}
+ self.file_id_to_name = []
+
+ def add_addr(self, dso_path, func_addr, addr):
+ dso = self.dso_map.get(dso_path)
+ if dso is None:
+ dso = self.dso_map[dso_path] = self.Dso()
+ if addr not in dso.addrs:
+ dso.addrs[addr] = self.Addr(func_addr)
+
+ def convert_addrs_to_lines(self):
+ for dso_path in self.dso_map:
+ self._convert_addrs_in_one_dso(dso_path, self.dso_map[dso_path])
+
+ def _convert_addrs_in_one_dso(self, dso_path, dso):
+ real_path = find_real_dso_path(dso_path, self.binary_cache_path)
+ if not real_path:
+ if dso_path not in ['//anon', 'unknown', '[kernel.kallsyms]']:
+ log_debug("Can't find dso %s" % dso_path)
+ return
+
+ if not self._check_debug_line_section(real_path):
+ log_debug("file %s doesn't contain .debug_line section." % real_path)
+ return
+
+ addr_step = self._get_addr_step(real_path)
+ self._collect_line_info(dso, real_path, [0])
+ self._collect_line_info(dso, real_path, range(-addr_step, -addr_step * 4 - 1, -addr_step))
+ self._collect_line_info(dso, real_path,
+ range(-addr_step * 5, -addr_step * 128 - 1, -addr_step))
+
+ def _check_debug_line_section(self, real_path):
+ try:
+ output = subprocess.check_output([self.readelf_path, '-S', real_path])
+ return output.find('.debug_line') != -1
+ except subprocess.CalledProcessError:
+ return False
+
+ def _get_addr_step(self, real_path):
+ arch = get_arch_of_dso_path(self.readelf_path, real_path)
+ if arch == 'arm64':
+ return 4
+ if arch == 'arm':
+ return 2
+ return 1
+
+ def _collect_line_info(self, dso, real_path, addr_shifts):
+ """ Use addr2line to get line info in a dso, with given addr shifts. """
+ # 1. Collect addrs to send to addr2line.
+ addr_set = set()
+ for addr in dso.addrs:
+ addr_obj = dso.addrs[addr]
+ if addr_obj.source_lines: # already has source line, no need to search.
+ continue
+ for shift in addr_shifts:
+ # The addr after shift shouldn't change to another function.
+ shifted_addr = max(addr + shift, addr_obj.func_addr)
+ addr_set.add(shifted_addr)
+ if shifted_addr == addr_obj.func_addr:
+ break
+ if not addr_set:
+ return
+ addr_request = '\n'.join(['%x' % addr for addr in sorted(addr_set)])
+
+ # 2. Use addr2line to collect line info.
+ try:
+ subproc = subprocess.Popen([self.addr2line_path, '-ai', '-e', real_path],
+ stdin=subprocess.PIPE, stdout=subprocess.PIPE)
+ (stdoutdata, _) = subproc.communicate(str_to_bytes(addr_request))
+ stdoutdata = bytes_to_str(stdoutdata)
+ except:
+ return
+ addr_map = {}
+ cur_line_list = None
+ for line in stdoutdata.strip().split('\n'):
+ if line[:2] == '0x':
+ # a new address
+ cur_line_list = addr_map[int(line, 16)] = []
+ else:
+ # a file:line.
+ if cur_line_list is None:
+ continue
+ # Handle lines like "C:\Users\...\file:32".
+ items = line.rsplit(':', 1)
+ if len(items) != 2:
+ continue
+ if '?' in line:
+ # if ? in line, it doesn't have a valid line info.
+ # An addr can have a list of (file, line), when the addr belongs to an inlined
+ # function. Sometimes only part of the list has ? mark. In this case, we think
+ # the line info is valid if the first line doesn't have ? mark.
+ if not cur_line_list:
+ cur_line_list = None
+ continue
+ (file_path, line_number) = items
+ line_number = line_number.split()[0] # Remove comments after line number
+ try:
+ line_number = int(line_number)
+ except ValueError:
+ continue
+ file_id = self._get_file_id(file_path)
+ cur_line_list.append((file_id, line_number))
+
+ # 3. Fill line info in dso.addrs.
+ for addr in dso.addrs:
+ addr_obj = dso.addrs[addr]
+ if addr_obj.source_lines:
+ continue
+ for shift in addr_shifts:
+ shifted_addr = max(addr + shift, addr_obj.func_addr)
+ lines = addr_map.get(shifted_addr)
+ if lines:
+ addr_obj.source_lines = lines
+ break
+ if shifted_addr == addr_obj.func_addr:
+ break
+
+ def _get_file_id(self, file_path):
+ file_id = self.file_name_to_id.get(file_path)
+ if file_id is None:
+ file_id = self.file_name_to_id[file_path] = len(self.file_id_to_name)
+ self.file_id_to_name.append(file_path)
+ return file_id
+
+ def get_file_path(self, file_id):
+ return self.file_id_to_name[file_id]
+
+ def get_dso(self, dso_path):
+ return self.dso_map.get(dso_path)
+
+
+class Objdump(object):
+ """ A wrapper of objdump to disassemble code. """
+ def __init__(self, ndk_path, binary_cache_path):
+ self.ndk_path = ndk_path
+ self.binary_cache_path = binary_cache_path
+ self.readelf_path = find_tool_path('readelf', ndk_path)
+ if not self.readelf_path:
+ log_exit("Can't find readelf. Please set ndk path by --ndk-path option.")
+ self.objdump_paths = {}
+
+ def disassemble_code(self, dso_path, start_addr, addr_len):
+ """ Disassemble [start_addr, start_addr + addr_len] of dso_path.
+ Return a list of pair (disassemble_code_line, addr).
+ """
+ # 1. Find real path.
+ real_path = find_real_dso_path(dso_path, self.binary_cache_path)
+ if real_path is None:
+ return None
+
+ # 2. Get path of objdump.
+ arch = get_arch_of_dso_path(self.readelf_path, real_path)
+ objdump_path = self.objdump_paths.get(arch)
+ if not objdump_path:
+ objdump_path = find_tool_path('objdump', self.ndk_path, arch)
+ if not objdump_path:
+ log_exit("Can't find objdump. Please set ndk path by --ndk-path option.")
+ self.objdump_paths[arch] = objdump_path
+
+ # 3. Run objdump.
+ args = [objdump_path, '-dlC', '--no-show-raw-insn',
+ '--start-address=0x%x' % start_addr,
+ '--stop-address=0x%x' % (start_addr + addr_len),
+ real_path]
+ try:
+ subproc = subprocess.Popen(args, stdin=subprocess.PIPE, stdout=subprocess.PIPE)
+ (stdoutdata, _) = subproc.communicate()
+ stdoutdata = bytes_to_str(stdoutdata)
+ except:
+ return None
+
+ if not stdoutdata:
+ return None
+ result = []
+ for line in stdoutdata.split('\n'):
+ line = line.rstrip() # Remove '\r' on Windows.
+ items = line.split(':', 1)
+ try:
+ addr = int(items[0], 16)
+ except ValueError:
+ addr = 0
+ result.append((line, addr))
+ return result
+
+
logging.getLogger().setLevel(logging.DEBUG)