1#!/usr/bin/python3
2#
3# Copyright (C) 2015 The Android Open Source Project
4#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9#     http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16
17"""
18Generate java test files for test 966.
19"""
20
21import generate_smali as base
22import os
23import sys
24from pathlib import Path
25
26BUILD_TOP = os.getenv("ANDROID_BUILD_TOP")
27if BUILD_TOP is None:
28  print("ANDROID_BUILD_TOP not set. Please run build/envsetup.sh", file=sys.stderr)
29  sys.exit(1)
30
31# Allow us to import mixins.
32sys.path.append(str(Path(BUILD_TOP)/"art"/"test"/"utils"/"python"))
33
34import testgen.mixins as mixins
35import functools
36import operator
37import subprocess
38
39class JavaConverter(mixins.DumpMixin, mixins.Named, mixins.JavaFileMixin):
40  """
41  A class that can convert a SmaliFile to a JavaFile.
42  """
43  def __init__(self, inner):
44    self.inner = inner
45
46  def get_name(self):
47    """Gets the name of this file."""
48    return self.inner.get_name()
49
50  def __str__(self):
51    out = ""
52    for line in str(self.inner).splitlines(keepends = True):
53      if line.startswith("#"):
54        out += line[1:]
55    return out
56
57class Compiler:
58  def __init__(self, sources, javac, temp_dir, classes_dir):
59    self.javac = javac
60    self.temp_dir = temp_dir
61    self.classes_dir = classes_dir
62    self.sources = sources
63
64  def compile_files(self, args, files):
65    """
66    Compile the files given with the arguments given.
67    """
68    args = args.split()
69    files = list(map(str, files))
70    cmd = ['sh', '-a', '-e', '--', str(self.javac)] + args + sorted(files)
71    subprocess.check_call(cmd)
72
73  def execute(self):
74    """
75    Compiles this test, doing partial compilation as necessary.
76    """
77    # Compile Main and all classes first. Force all interfaces to be default so that there will be
78    # no compiler problems (works since classes only implement 1 interface).
79    for f in self.sources:
80      if isinstance(f, base.TestInterface):
81        JavaConverter(f.get_specific_version(base.InterfaceType.default)).dump(self.temp_dir)
82      else:
83        JavaConverter(f).dump(self.temp_dir)
84    self.compile_files("-d {}".format(self.classes_dir), self.temp_dir.glob("*.java"))
85
86    # Now we compile the interfaces
87    ifaces = set(i for i in self.sources if isinstance(i, base.TestInterface))
88    while len(ifaces) != 0:
89      # Find those ifaces where there are no (uncompiled) interfaces that are subtypes.
90      tops = set(filter(lambda a: not any(map(lambda i: a in i.get_super_types(), ifaces)), ifaces))
91      files = []
92      # Dump these ones, they are getting compiled.
93      for f in tops:
94        out = JavaConverter(f)
95        out.dump(self.temp_dir)
96        files.append(self.temp_dir / out.get_file_name())
97      # Force all superinterfaces of these to be empty so there will be no conflicts
98      overrides = functools.reduce(operator.or_, map(lambda i: i.get_super_types(), tops), set())
99      for overridden in overrides:
100        out = JavaConverter(overridden.get_specific_version(base.InterfaceType.empty))
101        out.dump(self.temp_dir)
102        files.append(self.temp_dir / out.get_file_name())
103      self.compile_files("-d {outdir} -cp {outdir}".format(outdir = self.classes_dir), files)
104      # Remove these from the set of interfaces to be compiled.
105      ifaces -= tops
106    return
107
108def main(argv):
109  javac_exec = Path(argv[1])
110  if not javac_exec.exists() or not javac_exec.is_file():
111    print("{} is not a shell script".format(javac_exec), file=sys.stderr)
112    sys.exit(1)
113  temp_dir = Path(argv[2])
114  if not temp_dir.exists() or not temp_dir.is_dir():
115    print("{} is not a valid source dir".format(temp_dir), file=sys.stderr)
116    sys.exit(1)
117  classes_dir = Path(argv[3])
118  if not classes_dir.exists() or not classes_dir.is_dir():
119    print("{} is not a valid classes directory".format(classes_dir), file=sys.stderr)
120    sys.exit(1)
121  expected_txt = Path(argv[4])
122  mainclass, all_files = base.create_all_test_files()
123
124  with expected_txt.open('w') as out:
125    print(mainclass.get_expected(), file=out)
126
127  Compiler(all_files, javac_exec, temp_dir, classes_dir).execute()
128
129if __name__ == '__main__':
130  main(sys.argv)
131