|  | #!/usr/bin/python2.4 | 
|  | # | 
|  | # | 
|  | # Copyright 2009, 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. | 
|  |  | 
|  | """In memory representation of Android.mk file. | 
|  |  | 
|  | Specifications for Android.mk can be found at | 
|  | development/ndk/docs/ANDROID-MK.txt | 
|  | """ | 
|  |  | 
|  | import os | 
|  | import re | 
|  | from sets import Set | 
|  |  | 
|  | import logger | 
|  |  | 
|  | class AndroidMK(object): | 
|  | """In memory representation of Android.mk file.""" | 
|  |  | 
|  | _RE_INCLUDE = re.compile(r'include\s+\$\((.+)\)') | 
|  | _RE_VARIABLE_REF = re.compile(r'\$\((.+)\)') | 
|  | _VAR_DELIMITER = ":=" | 
|  | FILENAME = "Android.mk" | 
|  | CERTIFICATE = "LOCAL_CERTIFICATE" | 
|  | PACKAGE_NAME = "LOCAL_PACKAGE_NAME" | 
|  |  | 
|  | def __init__(self): | 
|  | self._includes = Set() # variables included in makefile | 
|  | self._variables = {} # variables defined in makefile | 
|  | self._has_gtestlib = False | 
|  |  | 
|  | def _ProcessMKLine(self, line): | 
|  | """Add a variable definition or include. | 
|  |  | 
|  | Ignores unrecognized lines. | 
|  |  | 
|  | Args: | 
|  | line: line of text from makefile | 
|  | """ | 
|  | m = self._RE_INCLUDE.match(line) | 
|  | if m: | 
|  | self._includes.add(m.group(1)) | 
|  | else: | 
|  | parts = line.split(self._VAR_DELIMITER) | 
|  | if len(parts) > 1: | 
|  | self._variables[parts[0].strip()] = parts[1].strip() | 
|  | # hack, look for explicit mention of libgtest_main | 
|  | if line.find('libgtest_main') != -1: | 
|  | self._has_gtestlib = True | 
|  |  | 
|  | def GetVariable(self, identifier): | 
|  | """Retrieve makefile variable. | 
|  |  | 
|  | Args: | 
|  | identifier: name of variable to retrieve | 
|  | Returns: | 
|  | value of specified identifier, None if identifier not found in makefile | 
|  | """ | 
|  | # use dict.get(x) rather than dict[x] to avoid KeyError exception, | 
|  | # so None is returned if identifier not found | 
|  | return self._variables.get(identifier, None) | 
|  |  | 
|  | def GetExpandedVariable(self, identifier): | 
|  | """Retrieve makefile variable. | 
|  |  | 
|  | If variable value refers to another variable, recursively expand it to | 
|  | find its literal value | 
|  |  | 
|  | Args: | 
|  | identifier: name of variable to retrieve | 
|  | Returns: | 
|  | value of specified identifier, None if identifier not found in makefile | 
|  | """ | 
|  | # use dict.get(x) rather than dict[x] to avoid KeyError exception, | 
|  | # so None is returned if identifier not found | 
|  | return self.__RecursiveGetVariable(identifier, Set()) | 
|  |  | 
|  | def __RecursiveGetVariable(self, identifier, visited_variables): | 
|  | variable_value = self.GetVariable(identifier) | 
|  | if not variable_value: | 
|  | return None | 
|  | if variable_value in visited_variables: | 
|  | raise RuntimeError('recursive loop found for makefile variable %s' | 
|  | % variable_value) | 
|  | m = self._RE_VARIABLE_REF.match(variable_value) | 
|  | if m: | 
|  | logger.SilentLog('Found variable ref %s for identifier %s' | 
|  | % (variable_value, identifier)) | 
|  | variable_ref = m.group(1) | 
|  | visited_variables.add(variable_ref) | 
|  | return self.__RecursiveGetVariable(variable_ref, visited_variables) | 
|  | else: | 
|  | return variable_value | 
|  |  | 
|  | def HasInclude(self, identifier): | 
|  | """Check variable is included in makefile. | 
|  |  | 
|  | Args: | 
|  | identifer: name of variable to check | 
|  | Returns: | 
|  | True if identifer is included in makefile, otherwise False | 
|  | """ | 
|  | return identifier in self._includes | 
|  |  | 
|  | def IncludesMakefilesUnder(self): | 
|  | """Check if makefile has a 'include makefiles under here' rule""" | 
|  | return self.HasInclude('call all-makefiles-under,$(LOCAL_PATH)') | 
|  |  | 
|  | def HasJavaLibrary(self, library_name): | 
|  | """Check if library is specified as a local java library in makefile. | 
|  |  | 
|  | Args: | 
|  | library_name: name of library to check | 
|  | Returns: | 
|  | True if library_name is included in makefile, otherwise False | 
|  | """ | 
|  | java_lib_string = self.GetExpandedVariable('LOCAL_JAVA_LIBRARIES') | 
|  | if java_lib_string: | 
|  | java_libs = java_lib_string.split(' ') | 
|  | return library_name in java_libs | 
|  | return False | 
|  |  | 
|  | def HasGTest(self): | 
|  | """Check if makefile includes rule to build a native gtest. | 
|  |  | 
|  | Returns: | 
|  | True if rule to build native test is in makefile, otherwise False | 
|  | """ | 
|  | return self._has_gtestlib or self.HasInclude('BUILD_NATIVE_TEST') | 
|  |  | 
|  | def _ParseMK(self, mk_path): | 
|  | """Parse Android.mk at the specified path. | 
|  |  | 
|  | Args: | 
|  | mk_path: path to Android.mk | 
|  | Raises: | 
|  | IOError: Android.mk cannot be found at given path, or cannot be opened | 
|  | for reading | 
|  | """ | 
|  | mk = open(mk_path) | 
|  | for line in mk: | 
|  | self._ProcessMKLine(line) | 
|  | mk.close() | 
|  |  | 
|  |  | 
|  | def CreateAndroidMK(path, filename=AndroidMK.FILENAME): | 
|  | """Factory method for creating a AndroidMK. | 
|  |  | 
|  | Args: | 
|  | path: the directory of the make file | 
|  | filename: the filename of the makefile | 
|  |  | 
|  | Return: | 
|  | the AndroidMK or None if there was no file present | 
|  | """ | 
|  | mk_path = os.path.join(path, filename) | 
|  | if os.path.isfile(mk_path): | 
|  | mk = AndroidMK() | 
|  | mk._ParseMK(mk_path) | 
|  | return mk | 
|  | else: | 
|  | return None |