| #!/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 |