Merge "Allow compare_apk_sizes to compare based on code sizes instead of full dex size"
diff --git a/tools/compare_apk_sizes.py b/tools/compare_apk_sizes.py
index e477320..78327c2 100755
--- a/tools/compare_apk_sizes.py
+++ b/tools/compare_apk_sizes.py
@@ -23,6 +23,9 @@
   result = optparse.OptionParser(usage=USAGE)
   result.add_option('--temp',
                     help='Temporary directory to store extracted classes in')
+  result.add_option('--use_code_size',
+      help='Use the size of code segments instead of the full size of the dex.',
+      default=False, action='store_true')
   result.add_option('--report',
                     help='Print comparison to this location instead of stdout')
   return result.parse_args()
@@ -50,20 +53,35 @@
   if toolhelper.run('d8', args) is not 0:
     raise Exception('Failed running d8')
 
+def get_code_size(path):
+  segments = toolhelper.run('dexsegments',
+                            [path],
+                            build=False,
+                            return_stdout=True)
+  for line in segments.splitlines():
+    if 'Code' in line:
+      # The code size line looks like:
+      #  - Code: 264 / 4
+      splits = line.split(' ')
+      return int(splits[3])
+
 class FileInfo:
-  def __init__(self, path, root):
+  def __init__(self, path, root, use_code_size):
     self.path = path
     self.full_path = os.path.join(root, path)
-    self.size = os.path.getsize(self.full_path)
+    if use_code_size:
+      self.size = get_code_size(self.full_path)
+    else:
+      self.size = os.path.getsize(self.full_path)
 
-def generate_file_info(path):
+def generate_file_info(path, options):
   file_info_map = {}
   with utils.ChangedWorkingDirectory(path):
     for root, dirs, files in os.walk('.'):
       for f in files:
         assert f.endswith('dex')
         file_path = os.path.join(root, f)
-        entry = FileInfo(file_path, path)
+        entry = FileInfo(file_path, path, use_code_size=options.use_code_size)
         file_info_map[file_path] = entry
   return file_info_map
 
@@ -85,9 +103,9 @@
   output.write('\n\n')
 
 
-def compare(app1_classes_dir, app2_classes_dir, app1, app2, report):
-  app1_files = generate_file_info(app1_classes_dir)
-  app2_files = generate_file_info(app2_classes_dir)
+def compare(app1_classes_dir, app2_classes_dir, app1, app2, options):
+  app1_files = generate_file_info(app1_classes_dir, options)
+  app2_files = generate_file_info(app2_classes_dir, options)
   only_in_app1 = [k for k in app1_files if k not in app2_files]
   only_in_app2 = [k for k in app2_files if k not in app1_files]
   in_both = [k for k in app2_files if k in app1_files]
@@ -105,12 +123,12 @@
       bigger_in_app2[f] = app2_entry.size - app1_entry.size
     else:
       same_size.append(f)
-  output = open(report, 'w') if report else sys.stdout
+  output = open(options.report, 'w') if options.report else sys.stdout
   print_info(app1, app1_files, only_in_app1, bigger_in_app1, output)
   print_info(app2, app2_files, only_in_app2, bigger_in_app2, output)
   output.write('Same size\n')
   output.write('\n'.join(['  %s' % x for x in same_size]))
-  if report:
+  if options.report:
     output.close()
 
 def Main():
@@ -137,7 +155,7 @@
 
     extract_classes(app1_input, app1_classes_dir)
     extract_classes(app2_input, app2_classes_dir)
-    compare(app1_classes_dir, app2_classes_dir, app1, app2, options.report)
+    compare(app1_classes_dir, app2_classes_dir, app1, app2, options)
 
 if __name__ == '__main__':
   sys.exit(Main())
diff --git a/tools/toolhelper.py b/tools/toolhelper.py
index 86787be..d7cf222 100644
--- a/tools/toolhelper.py
+++ b/tools/toolhelper.py
@@ -10,7 +10,7 @@
 
 def run(tool, args, build=None, debug=True,
         profile=False, track_memory_file=None, extra_args=None,
-        stderr=None, stdout=None):
+        stderr=None, stdout=None, return_stdout=False):
   if build is None:
     build, args = extract_build_from_args(args)
   if build:
@@ -36,6 +36,8 @@
     cmd.extend(["--lib", lib])
   cmd.extend(args)
   utils.PrintCmd(cmd)
+  if return_stdout:
+    return subprocess.check_output(cmd)
   return subprocess.call(cmd, stdout=stdout, stderr=stderr)
 
 def run_in_tests(tool, args, build=None, debug=True, extra_args=None):