Home | History | Annotate | Download | only in ant
      1 <?xml version="1.0" encoding="UTF-8"?>
      2 <project name="android_rules" default="debug">
      3 
      4     <!--
      5         This build file is imported by the project build file. It contains
      6         all the targets and tasks necessary to build Android projects, be they
      7         regular projects, library projects, or test projects.
      8 
      9         At the beginning of the file is a list of properties that can be overridden
     10         by adding them to your ant.properties (properties are immutable, so their
     11         first definition sticks and is never changed).
     12 
     13         Follows:
     14         - custom task definitions,
     15         - more properties (do not override those unless the whole build system is modified).
     16         - macros used throughout the build,
     17         - base build targets,
     18         - debug-specific build targets,
     19         - release-specific build targets,
     20         - instrument-specific build targets,
     21         - test project-specific build targets,
     22         - install targets,
     23         - help target
     24     -->
     25 
     26     <!-- ******************************************************* -->
     27     <!-- **************** Overridable Properties *************** -->
     28     <!-- ******************************************************* -->
     29 
     30     <!-- You can override these values in your build.xml or ant.properties.
     31          Overriding any other properties may result in broken build. -->
     32 
     33     <!-- Tells adb which device to target. You can change this from the command line
     34          by invoking "ant -Dadb.device.arg=-d" for device "ant -Dadb.device.arg=-e" for
     35          the emulator. -->
     36     <property name="adb.device.arg" value="" />
     37 
     38     <!-- fileset exclude patterns (space separated) to prevent
     39          files inside src/ from being packaged. -->
     40     <property name="android.package.excludes" value="" />
     41 
     42     <!-- set some properties used for filtering/override. If those weren't defined
     43          before, then this will create them with empty values, which are then ignored
     44          by the custom tasks receiving them. -->
     45     <property name="version.code" value="" />
     46     <property name="version.name" value="" />
     47     <property name="aapt.resource.filter" value="" />
     48     <!-- 'aapt.ignore.assets' is the list of file patterns to ignore under /res and /assets.
     49          Default is "!.svn:!.git:.*:<dir>_*:!CVS:!thumbs.db:!picasa.ini:!*.scc:*~"
     50 
     51          Overall patterns syntax is:
     52            [!][<dir>|<file>][*suffix-match|prefix-match*|full-match]:more:patterns...
     53 
     54          - The first character flag ! avoids printing a warning.
     55          - Pattern can have the flag "<dir>" to match only directories
     56            or "<file>" to match only files. Default is to match both.
     57          - Match is not case-sensitive.
     58     -->
     59     <property name="aapt.ignore.assets" value="" />
     60 
     61     <!-- dex force jumbo options, to be used when dex merging fails with
     62          UNEXPECTED TOP-LEVEL EXCEPTION: com.android.dx.util.DexException: Cannot handle conversion to jumbo index!
     63            at com.android.dx.merge.InstructionTransformer.jumboCheck(InstructionTransformer.java:103)
     64            ...
     65     -->
     66     <property name="dex.force.jumbo" value="false" />
     67     <property name="dex.disable.merger" value="false" />
     68 
     69     <!-- compilation options -->
     70     <property name="java.encoding" value="UTF-8" />
     71     <property name="java.target" value="1.5" />
     72     <property name="java.source" value="1.5" />
     73     <property name="java.compilerargs" value="" />
     74     <property name="java.compiler.classpath" value="" />
     75 
     76     <!-- Renderscript options -->
     77     <property name="renderscript.debug.opt.level" value="O0" />
     78     <property name="renderscript.release.opt.level" value="O3" />
     79 
     80     <!-- manifest merger default value -->
     81     <property name="manifestmerger.enabled" value="false" />
     82 
     83     <!-- instrumentation options -->
     84     <property name="emma.filter" value="" />
     85 
     86     <!-- Verbosity -->
     87     <property name="verbose" value="false" />
     88 
     89     <!-- Output location of the HTML report for the "lint" target.
     90          Ideally this would be specified as
     91             value="${out.dir}/lint-results.html"
     92          but we can't make a forward reference to the definition for
     93          ${out.dir}, and it is not a configurable property (yet).
     94      -->
     95     <property name="lint.out.html" value="bin/lint-results.html" />
     96 
     97     <!-- Output location of the XML report for the "lint" target -->
     98     <property name="lint.out.xml" value="bin/lint-results.xml" />
     99 
    100     <!-- ******************************************************* -->
    101     <!-- ********************* Custom Tasks ******************** -->
    102     <!-- ******************************************************* -->
    103 
    104     <!-- jar file from where the tasks are loaded -->
    105     <path id="android.antlibs">
    106         <pathelement path="${sdk.dir}/tools/lib/ant-tasks.jar" />
    107     </path>
    108 
    109     <!-- Custom tasks -->
    110     <taskdef resource="anttasks.properties" classpathref="android.antlibs" />
    111 
    112     <!-- Emma configuration -->
    113     <property name="emma.dir" value="${sdk.dir}/tools/lib" />
    114     <path id="emma.lib">
    115         <pathelement location="${emma.dir}/emma.jar" />
    116         <pathelement location="${emma.dir}/emma_ant.jar" />
    117     </path>
    118     <taskdef resource="emma_ant.properties" classpathref="emma.lib" />
    119     <!-- End of emma configuration -->
    120 
    121 
    122     <!-- ******************************************************* -->
    123     <!-- ******************* Other Properties ****************** -->
    124     <!-- ******************************************************* -->
    125     <!-- overriding these properties may break the build
    126          unless the whole file is updated -->
    127 
    128     <!-- Input directories -->
    129     <property name="source.dir" value="src" />
    130     <property name="source.absolute.dir" location="${source.dir}" />
    131     <property name="gen.absolute.dir" location="gen" />
    132     <property name="resource.absolute.dir" location="res" />
    133     <property name="asset.dir" value="assets" />
    134     <property name="asset.absolute.dir" location="${asset.dir}" />
    135     <property name="jar.libs.dir" value="libs" />
    136     <property name="jar.libs.absolute.dir" location="${jar.libs.dir}" />
    137     <property name="native.libs.absolute.dir" location="libs" />
    138 
    139     <property name="manifest.file" value="AndroidManifest.xml" />
    140     <property name="manifest.abs.file" location="${manifest.file}" />
    141 
    142     <!-- Output directories -->
    143     <property name="out.dir" value="bin" />
    144     <property name="out.absolute.dir" location="${out.dir}" />
    145     <property name="out.classes.absolute.dir" location="${out.dir}/classes" />
    146     <property name="out.res.absolute.dir" location="${out.dir}/res" />
    147     <property name="out.aidl.absolute.dir" location="${out.dir}/aidl" />
    148     <property name="out.dexed.absolute.dir" location="${out.dir}/dexedLibs" />
    149     <property name="out.manifest.abs.file" location="${out.dir}/AndroidManifest.xml" />
    150 
    151     <!-- tools location -->
    152     <property name="android.tools.dir" location="${sdk.dir}/tools" />
    153     <property name="android.platform.tools.dir" location="${sdk.dir}/platform-tools" />
    154     <condition property="exe" value=".exe" else=""><os family="windows" /></condition>
    155     <condition property="bat" value=".bat" else=""><os family="windows" /></condition>
    156     <property name="adb" location="${android.platform.tools.dir}/adb${exe}" />
    157     <property name="zipalign" location="${android.tools.dir}/zipalign${exe}" />
    158     <property name="lint" location="${android.tools.dir}/lint${bat}" />
    159 
    160     <!-- Intermediate files -->
    161     <property name="dex.file.name" value="classes.dex" />
    162     <property name="intermediate.dex.file" location="${out.absolute.dir}/${dex.file.name}" />
    163     <property name="resource.package.file.name" value="${ant.project.name}.ap_" />
    164 
    165     <!-- Build property file -->
    166     <property name="out.build.prop.file" location="${out.absolute.dir}/build.prop" />
    167 
    168 
    169     <!-- This is needed by emma as it uses multilevel verbosity instead of simple 'true' or 'false'
    170          The property 'verbosity' is not user configurable and depends exclusively on 'verbose'
    171          value.-->
    172     <condition property="verbosity" value="verbose" else="quiet">
    173         <istrue value="${verbose}" />
    174     </condition>
    175 
    176     <!-- properties for signing in release mode -->
    177     <condition property="has.keystore">
    178         <and>
    179             <isset property="key.store" />
    180             <length string="${key.store}" when="greater" length="0" />
    181             <isset property="key.alias" />
    182         </and>
    183     </condition>
    184     <condition property="has.password">
    185         <and>
    186             <isset property="has.keystore" />
    187             <isset property="key.store.password" />
    188             <isset property="key.alias.password" />
    189         </and>
    190     </condition>
    191 
    192     <!-- properties for packaging -->
    193     <property name="build.packaging.nocrunch" value="true" />
    194 
    195     <!-- whether we need to fork javac.
    196          This is only needed on Windows when running Java < 7 -->
    197     <condition else="false" property="need.javac.fork">
    198         <and>
    199             <matches pattern="1\.[56]" string="${java.specification.version}"/>
    200             <not>
    201                 <os family="unix"/>
    202             </not>
    203         </and>
    204     </condition>
    205 
    206     <!-- ******************************************************* -->
    207     <!-- ************************ Macros *********************** -->
    208     <!-- ******************************************************* -->
    209 
    210     <!-- macro to do a task on if project.is.library is false.
    211          elseText attribute is displayed otherwise -->
    212     <macrodef name="do-only-if-not-library">
    213         <attribute name="elseText" />
    214         <element name="task-to-do" implicit="yes" />
    215         <sequential>
    216         <if condition="${project.is.library}">
    217             <else>
    218                 <task-to-do />
    219             </else>
    220             <then>
    221                 <echo level="info">@{elseText}</echo>
    222             </then>
    223         </if>
    224         </sequential>
    225     </macrodef>
    226 
    227     <!-- macro to do a task on if manifest.hasCode is true.
    228          elseText attribute is displayed otherwise -->
    229     <macrodef name="do-only-if-manifest-hasCode">
    230         <attribute name="elseText" default=""/>
    231         <element name="task-to-do" implicit="yes" />
    232         <sequential>
    233         <if condition="${manifest.hasCode}">
    234             <then>
    235                 <task-to-do />
    236             </then>
    237             <else>
    238                 <if>
    239                     <condition>
    240                         <length string="@{elseText}" trim="true" when="greater" length="0" />
    241                     </condition>
    242                     <then>
    243                         <echo level="info">@{elseText}</echo>
    244                     </then>
    245                 </if>
    246             </else>
    247         </if>
    248         </sequential>
    249     </macrodef>
    250 
    251 
    252     <!-- Configurable macro, which allows to pass as parameters output directory,
    253          output dex filename and external libraries to dex (optional) -->
    254     <macrodef name="dex-helper">
    255         <element name="external-libs" optional="yes" />
    256         <attribute name="nolocals" default="false" />
    257         <sequential>
    258             <!-- sets the primary input for dex. If a pre-dex task sets it to
    259                  something else this has no effect -->
    260             <property name="out.dex.input.absolute.dir" value="${out.classes.absolute.dir}" />
    261 
    262             <!-- set the secondary dx input: the project (and library) jar files
    263                  If a pre-dex task sets it to something else this has no effect -->
    264             <if>
    265                 <condition>
    266                     <isreference refid="out.dex.jar.input.ref" />
    267                 </condition>
    268                 <else>
    269                     <path id="out.dex.jar.input.ref">
    270                         <path refid="project.all.jars.path" />
    271                     </path>
    272                 </else>
    273             </if>
    274 
    275             <dex executable="${dx}"
    276                     output="${intermediate.dex.file}"
    277                     dexedlibs="${out.dexed.absolute.dir}"
    278                     nolocals="@{nolocals}"
    279                     forceJumbo="${dex.force.jumbo}"
    280                     disableDexMerger="${dex.disable.merger}"
    281                     verbose="${verbose}">
    282                 <path path="${out.dex.input.absolute.dir}"/>
    283                 <path refid="out.dex.jar.input.ref" />
    284                 <external-libs />
    285             </dex>
    286         </sequential>
    287     </macrodef>
    288 
    289     <!-- This is macro that enable passing variable list of external jar files to ApkBuilder
    290          Example of use:
    291          <package-helper>
    292              <extra-jars>
    293                 <jarfolder path="my_jars" />
    294                 <jarfile path="foo/bar.jar" />
    295                 <jarfolder path="your_jars" />
    296              </extra-jars>
    297          </package-helper> -->
    298     <macrodef name="package-helper">
    299         <element name="extra-jars" optional="yes" />
    300         <sequential>
    301             <apkbuilder
    302                     outfolder="${out.absolute.dir}"
    303                     resourcefile="${resource.package.file.name}"
    304                     apkfilepath="${out.packaged.file}"
    305                     debugpackaging="${build.is.packaging.debug}"
    306                     debugsigning="${build.is.signing.debug}"
    307                     verbose="${verbose}"
    308                     hascode="${manifest.hasCode}"
    309                     previousBuildType="${build.last.is.packaging.debug}/${build.last.is.signing.debug}"
    310                     buildType="${build.is.packaging.debug}/${build.is.signing.debug}">
    311                 <dex path="${intermediate.dex.file}"/>
    312                 <sourcefolder path="${source.absolute.dir}"/>
    313                 <jarfile refid="project.all.jars.path" />
    314                 <nativefolder path="${native.libs.absolute.dir}" />
    315                 <nativefolder refid="project.library.native.folder.path" />
    316                 <extra-jars/>
    317             </apkbuilder>
    318         </sequential>
    319     </macrodef>
    320 
    321     <!-- This is macro which zipaligns in.package and outputs it to out.package. Used by targets
    322          debug, -debug-with-emma and release.-->
    323     <macrodef name="zipalign-helper">
    324         <attribute name="in.package" />
    325         <attribute name="out.package" />
    326         <sequential>
    327             <zipalign
    328                     executable="${zipalign}"
    329                     input="@{in.package}"
    330                     output="@{out.package}"
    331                     verbose="${verbose}" />
    332         </sequential>
    333     </macrodef>
    334 
    335     <macrodef name="run-tests-helper">
    336         <attribute name="emma.enabled" default="false" />
    337         <element name="extra-instrument-args" optional="yes" />
    338         <sequential>
    339             <echo level="info">Running tests ...</echo>
    340             <exec executable="${adb}" failonerror="true">
    341                 <arg line="${adb.device.arg}" />
    342                 <arg value="shell" />
    343                 <arg value="am" />
    344                 <arg value="instrument" />
    345                 <arg value="-w" />
    346                 <arg value="-e" />
    347                 <arg value="coverage" />
    348                 <arg value="@{emma.enabled}" />
    349                 <extra-instrument-args />
    350                 <arg value="${project.app.package}/${test.runner}" />
    351             </exec>
    352         </sequential>
    353     </macrodef>
    354 
    355     <macrodef name="record-build-key">
    356         <attribute name="key" default="false" />
    357         <attribute name="value" default="false" />
    358         <sequential>
    359             <propertyfile file="${out.build.prop.file}" comment="Last build type">
    360                 <entry key="@{key}" value="@{value}"/>
    361             </propertyfile>
    362         </sequential>
    363     </macrodef>
    364 
    365     <macrodef name="record-build-info">
    366         <sequential>
    367             <record-build-key key="build.last.target" value="${build.target}" />
    368             <record-build-key key="build.last.is.instrumented" value="${build.is.instrumented}" />
    369             <record-build-key key="build.last.is.packaging.debug" value="${build.is.packaging.debug}" />
    370             <record-build-key key="build.last.is.signing.debug" value="${build.is.signing.debug}" />
    371         </sequential>
    372     </macrodef>
    373 
    374     <macrodef name="uninstall-helper">
    375         <attribute name="app.package" default="false" />
    376         <sequential>
    377             <echo level="info">Uninstalling @{app.package} from the default emulator or device...</echo>
    378             <exec executable="${adb}" failonerror="true">
    379                 <arg line="${adb.device.arg}" />
    380                 <arg value="uninstall" />
    381                 <arg value="@{app.package}" />
    382             </exec>
    383         </sequential>
    384     </macrodef>
    385 
    386     <!-- ******************************************************* -->
    387     <!-- ******************** Build Targets ******************** -->
    388     <!-- ******************************************************* -->
    389 
    390     <!-- Basic Ant + SDK check -->
    391     <target name="-check-env">
    392         <checkenv />
    393     </target>
    394 
    395     <!-- target to disable building dependencies -->
    396     <target name="nodeps">
    397         <property name="dont.do.deps" value="true" />
    398     </target>
    399 
    400     <!-- generic setup -->
    401     <target name="-setup" depends="-check-env">
    402         <echo level="info">Project Name: ${ant.project.name}</echo>
    403         <gettype projectTypeOut="project.type" />
    404 
    405         <!-- sets a few boolean based on project.type
    406              to make the if task easier -->
    407         <condition property="project.is.library" value="true" else="false">
    408             <equals arg1="${project.type}" arg2="library" />
    409         </condition>
    410         <condition property="project.is.test" value="true" else="false">
    411             <equals arg1="${project.type}" arg2="test" />
    412         </condition>
    413         <condition property="project.is.testapp" value="true" else="false">
    414             <equals arg1="${project.type}" arg2="test-app" />
    415         </condition>
    416 
    417         <!-- If a test project, resolve absolute path to tested project. -->
    418         <if condition="${project.is.test}">
    419             <then>
    420                 <property name="tested.project.absolute.dir" location="${tested.project.dir}" />
    421             </then>
    422         </if>
    423 
    424         <!-- get the project manifest package -->
    425         <xpath input="${manifest.abs.file}"
    426                 expression="/manifest/@package" output="project.app.package" />
    427 
    428     </target>
    429 
    430     <!-- empty default pre-clean target. Create a similar target in
    431          your build.xml and it'll be called instead of this one. -->
    432     <target name="-pre-clean"/>
    433 
    434     <!-- clean target -->
    435     <target name="clean" depends="-setup, -pre-clean"
    436             description="Removes output files created by other targets.">
    437         <delete dir="${out.absolute.dir}" verbose="${verbose}" />
    438         <delete dir="${gen.absolute.dir}" verbose="${verbose}" />
    439 
    440         <!-- if we know about a tested project or libraries, we clean them too. -->
    441         <if condition="${project.is.test}">
    442             <then>
    443                 <property name="tested.project.absolute.dir" location="${tested.project.dir}" />
    444                 <subant failonerror="true">
    445                     <fileset dir="${tested.project.absolute.dir}" includes="build.xml" />
    446                     <target name="clean" />
    447                 </subant>
    448             </then>
    449         </if>
    450 
    451         <!-- get all the libraries -->
    452         <if>
    453             <condition><not><isset property="dont.do.deps" /></not></condition>
    454             <then>
    455                 <getlibpath libraryFolderPathOut="project.library.folder.path" />
    456                 <if>
    457                     <condition>
    458                         <isreference refid="project.library.folder.path" />
    459                     </condition>
    460                     <then>
    461                         <!-- clean the libraries with nodeps since we already
    462                              know about all the libraries even the indirect one -->
    463                         <subant
    464                                 buildpathref="project.library.folder.path"
    465                                 antfile="build.xml"
    466                                 failonerror="true">
    467                             <target name="nodeps" />
    468                             <target name="clean" />
    469                         </subant>
    470                     </then>
    471                 </if>
    472             </then>
    473         </if>
    474     </target>
    475 
    476     <!-- Pre build setup -->
    477     <target name="-build-setup" depends="-setup">
    478         <!-- find location of build tools -->
    479         <getbuildtools name="android.build.tools.dir" />
    480         <property name="aidl" location="${android.build.tools.dir}/aidl${exe}" />
    481         <property name="aapt" location="${android.build.tools.dir}/aapt${exe}" />
    482         <property name="dx" location="${android.build.tools.dir}/dx${bat}" />
    483         <property name="renderscript" location="${android.build.tools.dir}/llvm-rs-cc${exe}"/>
    484 
    485         <!-- Renderscript include Path -->
    486         <path id="android.renderscript.include.path">
    487             <pathelement location="${android.platform.tools.dir}/renderscript/include" />
    488             <pathelement location="${android.platform.tools.dir}/renderscript/clang-include" />
    489         </path>
    490 
    491         <!-- read the previous build mode -->
    492         <property file="${out.build.prop.file}" />
    493         <!-- if empty the props won't be set, meaning it's a new build.
    494              To force a build, set the prop to empty values. -->
    495         <property name="build.last.target" value="" />
    496         <property name="build.last.is.instrumented" value="" />
    497         <property name="build.last.is.packaging.debug" value="" />
    498         <property name="build.last.is.signing.debug" value="" />
    499 
    500         <!-- If the "debug" build type changed, clear out the compiled code.
    501              This is to make sure the new BuildConfig.DEBUG value is picked up
    502              as javac can't deal with this type of change in its dependency computation. -->
    503         <if>
    504             <condition>
    505                 <and>
    506                     <length string="${build.last.is.packaging.debug}" trim="true" when="greater" length="0" />
    507                     <not><equals
    508                             arg1="${build.is.packaging.debug}"
    509                             arg2="${build.last.is.packaging.debug}" /></not>
    510                 </and>
    511             </condition>
    512             <then>
    513                 <echo level="info">Switching between debug and non debug build: Deleting previous compilation output...</echo>
    514                 <delete dir="${out.classes.absolute.dir}" verbose="${verbose}" />
    515             </then>
    516             <else>
    517                 <!-- Else, we may still need to clean the code, for another reason.
    518                      special case for instrumented: if the previous build was
    519                      instrumented but not this one, clear out the compiled code -->
    520                 <if>
    521                     <condition>
    522                         <and>
    523                             <istrue value="${build.last.is.instrumented}" />
    524                             <isfalse value="${build.is.instrumented}" />
    525                         </and>
    526                     </condition>
    527                     <then>
    528                         <echo level="info">Switching from instrumented to non-instrumented build: Deleting previous compilation output...</echo>
    529                         <delete dir="${out.classes.absolute.dir}" verbose="${verbose}" />
    530                     </then>
    531                 </if>
    532             </else>
    533         </if>
    534 
    535         <echo level="info">Resolving Build Target for ${ant.project.name}...</echo>
    536         <!-- load project properties, resolve Android target, library dependencies
    537              and set some properties with the results.
    538              All property names are passed as parameters ending in -Out -->
    539         <gettarget
    540                 androidJarFileOut="project.target.android.jar"
    541                 androidAidlFileOut="project.target.framework.aidl"
    542                 bootClassPathOut="project.target.class.path"
    543                 targetApiOut="project.target.apilevel"
    544                 minSdkVersionOut="project.minSdkVersion" />
    545 
    546         <!-- Value of the hasCode attribute (Application node) extracted from manifest file -->
    547         <xpath input="${manifest.abs.file}" expression="/manifest/application/@android:hasCode"
    548                     output="manifest.hasCode" default="true"/>
    549 
    550         <echo level="info">----------</echo>
    551         <echo level="info">Creating output directories if needed...</echo>
    552         <mkdir dir="${resource.absolute.dir}" />
    553         <mkdir dir="${jar.libs.absolute.dir}" />
    554         <mkdir dir="${out.absolute.dir}" />
    555         <mkdir dir="${out.res.absolute.dir}" />
    556         <do-only-if-manifest-hasCode>
    557             <mkdir dir="${gen.absolute.dir}" />
    558             <mkdir dir="${out.classes.absolute.dir}" />
    559             <mkdir dir="${out.dexed.absolute.dir}" />
    560         </do-only-if-manifest-hasCode>
    561 
    562         <echo level="info">----------</echo>
    563         <echo level="info">Resolving Dependencies for ${ant.project.name}...</echo>
    564         <dependency
    565                 libraryFolderPathOut="project.library.folder.path"
    566                 libraryPackagesOut="project.library.packages"
    567                 libraryManifestFilePathOut="project.library.manifest.file.path"
    568                 libraryResFolderPathOut="project.library.res.folder.path"
    569                 libraryBinAidlFolderPathOut="project.library.bin.aidl.folder.path"
    570                 libraryRFilePathOut="project.library.bin.r.file.path"
    571                 libraryNativeFolderPathOut="project.library.native.folder.path"
    572                 jarLibraryPathOut="project.all.jars.path"
    573                 targetApi="${project.target.apilevel}"
    574                 verbose="${verbose}" />
    575 
    576         <!-- compile the libraries if any -->
    577         <if>
    578             <condition>
    579                 <and>
    580                     <isreference refid="project.library.folder.path" />
    581                     <not><isset property="dont.do.deps" /></not>
    582                 </and>
    583             </condition>
    584             <then>
    585                 <!-- figure out which target must be used to build the library projects.
    586                      If emma is enabled, then use 'instrument' otherwise, use 'debug' -->
    587                 <condition property="project.libraries.target" value="instrument" else="${build.target}">
    588                     <istrue value="${build.is.instrumented}" />
    589                 </condition>
    590 
    591                 <echo level="info">----------</echo>
    592                 <echo level="info">Building Libraries with '${project.libraries.target}'...</echo>
    593 
    594                 <!-- no need to build the deps as we have already
    595                      the full list of libraries -->
    596                 <subant failonerror="true"
    597                         buildpathref="project.library.folder.path"
    598                         antfile="build.xml">
    599                     <target name="nodeps" />
    600                     <target name="${project.libraries.target}" />
    601                     <property name="emma.coverage.absolute.file" location="${out.absolute.dir}/coverage.em" />
    602                 </subant>
    603             </then>
    604         </if>
    605 
    606         <!-- compile the main project if this is a test project -->
    607         <if condition="${project.is.test}">
    608             <then>
    609                 <!-- figure out which target must be used to build the tested project.
    610                      If emma is enabled, then use 'instrument' otherwise, use 'debug' -->
    611                 <condition property="tested.project.target" value="instrument" else="debug">
    612                     <isset property="emma.enabled" />
    613                 </condition>
    614 
    615                 <echo level="info">----------</echo>
    616                 <echo level="info">Building tested project at ${tested.project.absolute.dir} with '${tested.project.target}'...</echo>
    617                 <subant target="${tested.project.target}" failonerror="true">
    618                     <fileset dir="${tested.project.absolute.dir}" includes="build.xml" />
    619                 </subant>
    620 
    621                 <!-- get the tested project full classpath to be able to build
    622                      the test project -->
    623                 <testedprojectclasspath
    624                         projectLocation="${tested.project.absolute.dir}"
    625                         projectClassPathOut="tested.project.classpath"/>
    626             </then>
    627             <else>
    628                 <!-- no tested project, make an empty Path object so that javac doesn't
    629                      complain -->
    630                 <path id="tested.project.classpath" />
    631             </else>
    632         </if>
    633     </target>
    634 
    635     <!-- empty default pre-build target. Create a similar target in
    636          your build.xml and it'll be called instead of this one. -->
    637     <target name="-pre-build"/>
    638 
    639     <!-- Code Generation: compile resources (aapt -> R.java), aidl, renderscript -->
    640     <target name="-code-gen">
    641         <!-- always merge manifest -->
    642         <mergemanifest
    643                 appManifest="${manifest.abs.file}"
    644                 outManifest="${out.manifest.abs.file}"
    645                 enabled="${manifestmerger.enabled}">
    646             <library refid="project.library.manifest.file.path" />
    647         </mergemanifest>
    648 
    649         <do-only-if-manifest-hasCode
    650                 elseText="hasCode = false. Skipping aidl/renderscript/R.java">
    651             <echo level="info">Handling aidl files...</echo>
    652             <aidl executable="${aidl}"
    653                     framework="${project.target.framework.aidl}"
    654                     libraryBinAidlFolderPathRefid="project.library.bin.aidl.folder.path"
    655                     genFolder="${gen.absolute.dir}"
    656                     aidlOutFolder="${out.aidl.absolute.dir}">
    657                 <source path="${source.absolute.dir}"/>
    658             </aidl>
    659 
    660             <!-- renderscript generates resources so it must be called before aapt -->
    661             <echo level="info">----------</echo>
    662             <echo level="info">Handling RenderScript files...</echo>
    663             <!-- set the rs target prop in case it hasn't been set. -->
    664             <property name="renderscript.target" value="${project.minSdkVersion}" />
    665             <renderscript executable="${renderscript}"
    666                     includePathRefId="android.renderscript.include.path"
    667                     genFolder="${gen.absolute.dir}"
    668                     resFolder="${out.res.absolute.dir}/raw"
    669                     targetApi="${renderscript.target}"
    670                     optLevel="${renderscript.opt.level}"
    671                     buildType="${build.is.packaging.debug}"
    672                     previousBuildType="${build.last.is.packaging.debug}">
    673                 <source path="${source.absolute.dir}"/>
    674             </renderscript>
    675 
    676             <echo level="info">----------</echo>
    677             <echo level="info">Handling Resources...</echo>
    678             <aapt executable="${aapt}"
    679                     command="package"
    680                     verbose="${verbose}"
    681                     manifest="${out.manifest.abs.file}"
    682                     originalManifestPackage="${project.app.package}"
    683                     androidjar="${project.target.android.jar}"
    684                     rfolder="${gen.absolute.dir}"
    685                     nonConstantId="${android.library}"
    686                     libraryResFolderPathRefid="project.library.res.folder.path"
    687                     libraryPackagesRefid="project.library.packages"
    688                     libraryRFileRefid="project.library.bin.r.file.path"
    689                     ignoreAssets="${aapt.ignore.assets}"
    690                     binFolder="${out.absolute.dir}"
    691                     proguardFile="${out.absolute.dir}/proguard.txt">
    692                 <res path="${out.res.absolute.dir}" />
    693                 <res path="${resource.absolute.dir}" />
    694             </aapt>
    695 
    696             <echo level="info">----------</echo>
    697             <echo level="info">Handling BuildConfig class...</echo>
    698             <buildconfig
    699                     genFolder="${gen.absolute.dir}"
    700                     package="${project.app.package}"
    701                     buildType="${build.is.packaging.debug}"
    702                     previousBuildType="${build.last.is.packaging.debug}"/>
    703 
    704         </do-only-if-manifest-hasCode>
    705     </target>
    706 
    707     <!-- empty default pre-compile target. Create a similar target in
    708          your build.xml and it'll be called instead of this one. -->
    709     <target name="-pre-compile"/>
    710 
    711     <!-- Compiles this project's .java files into .class files. -->
    712     <target name="-compile" depends="-pre-build, -build-setup, -code-gen, -pre-compile">
    713         <do-only-if-manifest-hasCode elseText="hasCode = false. Skipping...">
    714             <!-- merge the project's own classpath and the tested project's classpath -->
    715             <path id="project.javac.classpath">
    716                 <path refid="project.all.jars.path" />
    717                 <path refid="tested.project.classpath" />
    718                 <path path="${java.compiler.classpath}" />
    719             </path>
    720             <javac encoding="${java.encoding}"
    721                     source="${java.source}" target="${java.target}"
    722                     debug="true" extdirs="" includeantruntime="false"
    723                     destdir="${out.classes.absolute.dir}"
    724                     bootclasspathref="project.target.class.path"
    725                     verbose="${verbose}"
    726                     classpathref="project.javac.classpath"
    727                     fork="${need.javac.fork}">
    728                 <src path="${source.absolute.dir}" />
    729                 <src path="${gen.absolute.dir}" />
    730                 <compilerarg line="${java.compilerargs}" />
    731             </javac>
    732 
    733             <!-- if the project is instrumented, intrument the classes -->
    734             <if condition="${build.is.instrumented}">
    735                 <then>
    736                     <echo level="info">Instrumenting classes from ${out.absolute.dir}/classes...</echo>
    737 
    738                     <!-- build the filter to remove R, Manifest, BuildConfig -->
    739                     <getemmafilter
    740                             appPackage="${project.app.package}"
    741                             libraryPackagesRefId="project.library.packages"
    742                             filterOut="emma.default.filter"/>
    743 
    744                     <!-- define where the .em file is going. This may have been
    745                          setup already if this is a library -->
    746                     <property name="emma.coverage.absolute.file" location="${out.absolute.dir}/coverage.em" />
    747 
    748                     <!-- It only instruments class files, not any external libs -->
    749                     <emma enabled="true">
    750                         <instr verbosity="${verbosity}"
    751                                mode="overwrite"
    752                                instrpath="${out.absolute.dir}/classes"
    753                                outdir="${out.absolute.dir}/classes"
    754                                metadatafile="${emma.coverage.absolute.file}">
    755                             <filter excludes="${emma.default.filter}" />
    756                             <filter value="${emma.filter}" />
    757                         </instr>
    758                     </emma>
    759                 </then>
    760             </if>
    761 
    762             <!-- if the project is a library then we generate a jar file -->
    763             <if condition="${project.is.library}">
    764                 <then>
    765                     <echo level="info">Creating library output jar file...</echo>
    766                     <property name="out.library.jar.file" location="${out.absolute.dir}/classes.jar" />
    767                     <if>
    768                         <condition>
    769                             <length string="${android.package.excludes}" trim="true" when="greater" length="0" />
    770                         </condition>
    771                         <then>
    772                             <echo level="info">Custom jar packaging exclusion: ${android.package.excludes}</echo>
    773                         </then>
    774                     </if>
    775 
    776                     <propertybyreplace name="project.app.package.path" input="${project.app.package}" replace="." with="/" />
    777 
    778                     <jar destfile="${out.library.jar.file}">
    779                         <fileset dir="${out.classes.absolute.dir}"
    780                                 includes="**/*.class"
    781                                 excludes="${project.app.package.path}/R.class ${project.app.package.path}/R$*.class ${project.app.package.path}/BuildConfig.class"/>
    782                         <fileset dir="${source.absolute.dir}" excludes="**/*.java ${android.package.excludes}" />
    783                     </jar>
    784                 </then>
    785             </if>
    786 
    787         </do-only-if-manifest-hasCode>
    788     </target>
    789 
    790     <!-- empty default post-compile target. Create a similar target in
    791          your build.xml and it'll be called instead of this one. -->
    792     <target name="-post-compile"/>
    793 
    794     <!-- Obfuscate target
    795         This is only active in release builds when proguard.config is defined
    796         in default.properties.
    797 
    798         To replace Proguard with a different obfuscation engine:
    799         Override the following targets in your build.xml, before the call to <setup>
    800             -release-obfuscation-check
    801                 Check whether obfuscation should happen, and put the result in a property.
    802             -debug-obfuscation-check
    803                 Obfuscation should not happen. Set the same property to false.
    804             -obfuscate
    805                 check if the property set in -debug/release-obfuscation-check is set to true.
    806                 If true:
    807                     Perform obfuscation
    808                     Set property out.dex.input.absolute.dir to be the output of the obfuscation
    809     -->
    810     <target name="-obfuscate">
    811         <if condition="${proguard.enabled}">
    812             <then>
    813                 <property name="obfuscate.absolute.dir" location="${out.absolute.dir}/proguard" />
    814                 <property name="preobfuscate.jar.file" value="${obfuscate.absolute.dir}/original.jar" />
    815                 <property name="obfuscated.jar.file" value="${obfuscate.absolute.dir}/obfuscated.jar" />
    816                 <!-- input for dex will be proguard's output -->
    817                 <property name="out.dex.input.absolute.dir" value="${obfuscated.jar.file}" />
    818 
    819                 <!-- Add Proguard Tasks -->
    820                 <property name="proguard.jar" location="${android.tools.dir}/proguard/lib/proguard.jar" />
    821                 <taskdef name="proguard" classname="proguard.ant.ProGuardTask" classpath="${proguard.jar}" />
    822 
    823                 <!-- Set the android classpath Path object into a single property. It'll be
    824                      all the jar files separated by a platform path-separator.
    825                      Each path must be quoted if it contains spaces.
    826                 -->
    827                 <pathconvert property="project.target.classpath.value" refid="project.target.class.path">
    828                     <firstmatchmapper>
    829                         <regexpmapper from='^([^ ]*)( .*)$$' to='"\1\2"'/>
    830                         <identitymapper/>
    831                     </firstmatchmapper>
    832                 </pathconvert>
    833 
    834                 <!-- Build a path object with all the jar files that must be obfuscated.
    835                      This include the project compiled source code and any 3rd party jar
    836                      files. -->
    837                 <path id="project.all.classes.path">
    838                     <pathelement location="${preobfuscate.jar.file}" />
    839                     <path refid="project.all.jars.path" />
    840                 </path>
    841                 <!-- Set the project jar files Path object into a single property. It'll be
    842                      all the jar files separated by a platform path-separator.
    843                      Each path must be quoted if it contains spaces.
    844                 -->
    845                 <pathconvert property="project.all.classes.value" refid="project.all.classes.path">
    846                     <firstmatchmapper>
    847                         <regexpmapper from='^([^ ]*)( .*)$$' to='"\1\2"'/>
    848                         <identitymapper/>
    849                     </firstmatchmapper>
    850                 </pathconvert>
    851 
    852                 <!-- Turn the path property ${proguard.config} from an A:B:C property
    853                      into a series of includes: -include A -include B -include C
    854                      suitable for processing by the ProGuard task. Note - this does
    855                      not include the leading '-include "' or the closing '"'; those
    856                      are added under the <proguard> call below.
    857                 -->
    858                 <path id="proguard.configpath">
    859                     <pathelement path="${proguard.config}"/>
    860                 </path>
    861                 <pathconvert pathsep='" -include "' property="proguard.configcmd" refid="proguard.configpath"/>
    862 
    863                 <mkdir   dir="${obfuscate.absolute.dir}" />
    864                 <delete file="${preobfuscate.jar.file}"/>
    865                 <delete file="${obfuscated.jar.file}"/>
    866                 <jar basedir="${out.classes.absolute.dir}"
    867                     destfile="${preobfuscate.jar.file}" />
    868                 <proguard>
    869                     -include      "${proguard.configcmd}"
    870                     -include      "${out.absolute.dir}/proguard.txt"
    871                     -injars       ${project.all.classes.value}
    872                     -outjars      "${obfuscated.jar.file}"
    873                     -libraryjars  ${project.target.classpath.value}
    874                     -dump         "${obfuscate.absolute.dir}/dump.txt"
    875                     -printseeds   "${obfuscate.absolute.dir}/seeds.txt"
    876                     -printusage   "${obfuscate.absolute.dir}/usage.txt"
    877                     -printmapping "${obfuscate.absolute.dir}/mapping.txt"
    878                 </proguard>
    879             </then>
    880         </if>
    881     </target>
    882 
    883     <!-- Converts this project's .class files into .dex files -->
    884     <target name="-dex" depends="-compile, -post-compile, -obfuscate">
    885         <do-only-if-manifest-hasCode elseText="hasCode = false. Skipping...">
    886             <!-- only convert to dalvik bytecode is *not* a library -->
    887             <do-only-if-not-library elseText="Library project: do not convert bytecode..." >
    888                 <!-- special case for instrumented builds: need to use no-locals and need
    889                      to pass in the emma jar. -->
    890                 <if condition="${build.is.instrumented}">
    891                     <then>
    892                         <dex-helper nolocals="true">
    893                             <external-libs>
    894                                 <fileset file="${emma.dir}/emma_device.jar" />
    895                             </external-libs>
    896                         </dex-helper>
    897                     </then>
    898                     <else>
    899                         <dex-helper />
    900                     </else>
    901                 </if>
    902             </do-only-if-not-library>
    903         </do-only-if-manifest-hasCode>
    904     </target>
    905 
    906 <!-- Updates the pre-processed PNG cache -->
    907     <target name="-crunch">
    908         <exec executable="${aapt}" taskName="crunch">
    909             <arg value="crunch" />
    910             <arg value="-v" />
    911             <arg value="-S" />
    912             <arg path="${resource.absolute.dir}" />
    913             <arg value="-C" />
    914             <arg path="${out.res.absolute.dir}" />
    915         </exec>
    916     </target>
    917 
    918     <!-- Puts the project's resources into the output package file
    919          This actually can create multiple resource package in case
    920          Some custom apk with specific configuration have been
    921          declared in default.properties.
    922          -->
    923     <target name="-package-resources" depends="-crunch">
    924         <!-- only package resources if *not* a library project -->
    925         <do-only-if-not-library elseText="Library project: do not package resources..." >
    926             <aapt executable="${aapt}"
    927                     command="package"
    928                     versioncode="${version.code}"
    929                     versionname="${version.name}"
    930                     debug="${build.is.packaging.debug}"
    931                     manifest="${out.manifest.abs.file}"
    932                     assets="${asset.absolute.dir}"
    933                     androidjar="${project.target.android.jar}"
    934                     apkfolder="${out.absolute.dir}"
    935                     nocrunch="${build.packaging.nocrunch}"
    936                     resourcefilename="${resource.package.file.name}"
    937                     resourcefilter="${aapt.resource.filter}"
    938                     libraryResFolderPathRefid="project.library.res.folder.path"
    939                     libraryPackagesRefid="project.library.packages"
    940                     libraryRFileRefid="project.library.bin.r.file.path"
    941                     previousBuildType="${build.last.target}"
    942                     buildType="${build.target}"
    943                     ignoreAssets="${aapt.ignore.assets}">
    944                 <res path="${out.res.absolute.dir}" />
    945                 <res path="${resource.absolute.dir}" />
    946                 <!-- <nocompress /> forces no compression on any files in assets or res/raw -->
    947                 <!-- <nocompress extension="xml" /> forces no compression on specific file extensions in assets and res/raw -->
    948             </aapt>
    949         </do-only-if-not-library>
    950     </target>
    951 
    952     <!-- Packages the application. -->
    953     <target name="-package" depends="-dex, -package-resources">
    954         <!-- only package apk if *not* a library project -->
    955         <do-only-if-not-library elseText="Library project: do not package apk..." >
    956             <if condition="${build.is.instrumented}">
    957                 <then>
    958                     <package-helper>
    959                         <extra-jars>
    960                             <!-- Injected from external file -->
    961                             <jarfile path="${emma.dir}/emma_device.jar" />
    962                         </extra-jars>
    963                     </package-helper>
    964                 </then>
    965                 <else>
    966                     <package-helper />
    967                 </else>
    968             </if>
    969         </do-only-if-not-library>
    970     </target>
    971 
    972     <target name="-post-package" />
    973     <target name="-post-build" />
    974 
    975     <target name="-set-mode-check">
    976         <fail if="build.is.mode.set"
    977                 message="Cannot run two different modes at the same time. If you are running more than one debug/release/instrument type targets, call them from different Ant calls." />
    978     </target>
    979 
    980     <!-- ******************************************************* -->
    981     <!-- **************** Debug specific targets *************** -->
    982     <!-- ******************************************************* -->
    983 
    984     <target name="-set-debug-files" depends="-set-mode-check">
    985 
    986         <property name="out.packaged.file" location="${out.absolute.dir}/${ant.project.name}-debug-unaligned.apk" />
    987         <property name="out.final.file" location="${out.absolute.dir}/${ant.project.name}-debug.apk" />
    988         <property name="build.is.mode.set" value="true" />
    989     </target>
    990 
    991 
    992     <target name="-set-debug-mode" depends="-setup">
    993         <!-- record the current build target -->
    994         <property name="build.target" value="debug" />
    995 
    996         <if>
    997             <condition>
    998                 <and>
    999                     <istrue value="${project.is.testapp}" />
   1000                     <istrue value="${emma.enabled}" />
   1001                 </and>
   1002             </condition>
   1003             <then>
   1004                 <property name="build.is.instrumented" value="true" />
   1005             </then>
   1006             <else>
   1007                 <property name="build.is.instrumented" value="false" />
   1008             </else>
   1009         </if>
   1010 
   1011         <!-- whether the build is a debug build. always set. -->
   1012         <property name="build.is.packaging.debug" value="true" />
   1013 
   1014         <!-- signing mode: debug -->
   1015         <property name="build.is.signing.debug" value="true" />
   1016 
   1017         <!-- Renderscript optimization level: none -->
   1018         <property name="renderscript.opt.level" value="${renderscript.debug.opt.level}" />
   1019 
   1020     </target>
   1021 
   1022     <target name="-debug-obfuscation-check">
   1023         <!-- proguard is never enabled in debug mode -->
   1024         <property name="proguard.enabled" value="false"/>
   1025     </target>
   1026 
   1027     <!-- Builds debug output package -->
   1028     <target name="-do-debug" depends="-set-debug-mode, -debug-obfuscation-check, -package, -post-package">
   1029         <!-- only create apk if *not* a library project -->
   1030         <do-only-if-not-library elseText="Library project: do not create apk..." >
   1031             <sequential>
   1032                 <zipalign-helper in.package="${out.packaged.file}" out.package="${out.final.file}" />
   1033                 <echo level="info">Debug Package: ${out.final.file}</echo>
   1034             </sequential>
   1035         </do-only-if-not-library>
   1036         <record-build-info />
   1037     </target>
   1038 
   1039     <!-- Builds debug output package -->
   1040     <target name="debug" depends="-set-debug-files, -do-debug, -post-build"
   1041                 description="Builds the application and signs it with a debug key.">
   1042     </target>
   1043 
   1044 
   1045     <!-- ******************************************************* -->
   1046     <!-- *************** Release specific targets ************** -->
   1047     <!-- ******************************************************* -->
   1048 
   1049     <!-- called through target 'release'. Only executed if the keystore and
   1050          key alias are known but not their password. -->
   1051     <target name="-release-prompt-for-password" if="has.keystore" unless="has.password">
   1052         <!-- Gets passwords -->
   1053         <input
   1054                 message="Please enter keystore password (store:${key.store}):"
   1055                 addproperty="key.store.password" />
   1056         <input
   1057                 message="Please enter password for alias '${key.alias}':"
   1058                 addproperty="key.alias.password" />
   1059     </target>
   1060 
   1061     <!-- called through target 'release'. Only executed if there's no
   1062          keystore/key alias set -->
   1063     <target name="-release-nosign" unless="has.keystore">
   1064         <!-- no release builds for library project -->
   1065         <do-only-if-not-library elseText="" >
   1066             <sequential>
   1067                 <echo level="info">No key.store and key.alias properties found in build.properties.</echo>
   1068                 <echo level="info">Please sign ${out.packaged.file} manually</echo>
   1069                 <echo level="info">and run zipalign from the Android SDK tools.</echo>
   1070             </sequential>
   1071         </do-only-if-not-library>
   1072         <record-build-info />
   1073     </target>
   1074 
   1075     <target name="-release-obfuscation-check">
   1076         <echo level="info">proguard.config is ${proguard.config}</echo>
   1077         <condition property="proguard.enabled" value="true" else="false">
   1078             <and>
   1079                 <isset property="build.is.mode.release" />
   1080                 <isset property="proguard.config" />
   1081             </and>
   1082         </condition>
   1083         <if condition="${proguard.enabled}">
   1084             <then>
   1085                 <echo level="info">Proguard.config is enabled</echo>
   1086                 <!-- Secondary dx input (jar files) is empty since all the
   1087                      jar files will be in the obfuscated jar -->
   1088                 <path id="out.dex.jar.input.ref" />
   1089             </then>
   1090         </if>
   1091     </target>
   1092 
   1093     <target name="-set-release-mode" depends="-set-mode-check">
   1094         <property name="out.packaged.file" location="${out.absolute.dir}/${ant.project.name}-release-unsigned.apk" />
   1095         <property name="out.final.file" location="${out.absolute.dir}/${ant.project.name}-release.apk" />
   1096         <property name="build.is.mode.set" value="true" />
   1097 
   1098         <!-- record the current build target -->
   1099         <property name="build.target" value="release" />
   1100 
   1101         <property name="build.is.instrumented" value="false" />
   1102 
   1103         <!-- release mode is only valid if the manifest does not explicitly
   1104              set debuggable to true. default is false. -->
   1105         <xpath input="${manifest.abs.file}" expression="/manifest/application/@android:debuggable"
   1106                 output="build.is.packaging.debug" default="false"/>
   1107 
   1108         <!-- signing mode: release -->
   1109         <property name="build.is.signing.debug" value="false" />
   1110 
   1111         <!-- Renderscript optimization level: aggressive -->
   1112         <property name="renderscript.opt.level" value="${renderscript.release.opt.level}" />
   1113 
   1114         <if condition="${build.is.packaging.debug}">
   1115             <then>
   1116                 <echo>*************************************************</echo>
   1117                 <echo>****  Android Manifest has debuggable=true   ****</echo>
   1118                 <echo>**** Doing DEBUG packaging with RELEASE keys ****</echo>
   1119                 <echo>*************************************************</echo>
   1120             </then>
   1121             <else>
   1122                 <!-- property only set in release mode.
   1123                      Useful for if/unless attributes in target node
   1124                      when using Ant before 1.8 -->
   1125                 <property name="build.is.mode.release" value="true"/>
   1126             </else>
   1127         </if>
   1128     </target>
   1129 
   1130     <target name="-release-sign" if="has.keystore" >
   1131         <!-- only create apk if *not* a library project -->
   1132         <do-only-if-not-library elseText="Library project: do not create apk..." >
   1133             <sequential>
   1134                 <property name="out.unaligned.file" location="${out.absolute.dir}/${ant.project.name}-release-unaligned.apk" />
   1135 
   1136                 <!-- Signs the APK -->
   1137                 <echo level="info">Signing final apk...</echo>
   1138                 <signapk
   1139                         input="${out.packaged.file}"
   1140                         output="${out.unaligned.file}"
   1141                         keystore="${key.store}"
   1142                         storepass="${key.store.password}"
   1143                         alias="${key.alias}"
   1144                         keypass="${key.alias.password}"/>
   1145 
   1146                 <!-- Zip aligns the APK -->
   1147                 <zipalign-helper
   1148                         in.package="${out.unaligned.file}"
   1149                         out.package="${out.final.file}" />
   1150                 <echo level="info">Release Package: ${out.final.file}</echo>
   1151             </sequential>
   1152         </do-only-if-not-library>
   1153         <record-build-info />
   1154     </target>
   1155 
   1156     <!-- This runs -package-release and -release-nosign first and then runs
   1157          only if release-sign is true (set in -release-check,
   1158          called by -release-no-sign)-->
   1159     <target name="release"
   1160                 depends="-set-release-mode, -release-obfuscation-check, -package, -post-package, -release-prompt-for-password, -release-nosign, -release-sign, -post-build"
   1161                 description="Builds the application in release mode.">
   1162     </target>
   1163 
   1164     <!-- ******************************************************* -->
   1165     <!-- ************ Instrumented specific targets ************ -->
   1166     <!-- ******************************************************* -->
   1167 
   1168     <!-- These targets are specific for the project under test when it
   1169          gets compiled by the test projects in a way that will make it
   1170          support emma code coverage -->
   1171 
   1172     <target name="-set-instrumented-mode" depends="-set-mode-check">
   1173         <property name="out.packaged.file" location="${out.absolute.dir}/${ant.project.name}-instrumented-unaligned.apk" />
   1174         <property name="out.final.file" location="${out.absolute.dir}/${ant.project.name}-instrumented.apk" />
   1175         <property name="build.is.mode.set" value="true" />
   1176 
   1177         <!-- whether the build is an instrumented build. -->
   1178         <property name="build.is.instrumented" value="true" />
   1179     </target>
   1180 
   1181     <!-- Builds instrumented output package -->
   1182     <target name="instrument" depends="-set-instrumented-mode, -do-debug"
   1183                 description="Builds an instrumented packaged.">
   1184         <!-- only create apk if *not* a library project -->
   1185         <do-only-if-not-library elseText="Library project: do not create apk..." >
   1186             <sequential>
   1187                 <zipalign-helper in.package="${out.packaged.file}" out.package="${out.final.file}" />
   1188                 <echo level="info">Instrumented Package: ${out.final.file}</echo>
   1189             </sequential>
   1190         </do-only-if-not-library>
   1191         <record-build-info />
   1192     </target>
   1193 
   1194     <!-- ******************************************************* -->
   1195     <!-- ************ Test project specific targets ************ -->
   1196     <!-- ******************************************************* -->
   1197 
   1198     <!-- enable code coverage -->
   1199     <target name="emma">
   1200         <property name="emma.enabled" value="true" />
   1201     </target>
   1202 
   1203     <!-- fails if the project is not a test project -->
   1204     <target name="-test-project-check" depends="-setup">
   1205         <if>
   1206             <condition>
   1207                 <and>
   1208                     <isfalse value="${project.is.test}" />
   1209                     <isfalse value="${project.is.testapp}" />
   1210                 </and>
   1211             </condition>
   1212             <then>
   1213                 <fail message="Project is not a test project." />
   1214             </then>
   1215         </if>
   1216     </target>
   1217 
   1218     <target name="test" depends="-test-project-check"
   1219                 description="Runs tests from the package defined in test.package property">
   1220         <property name="test.runner" value="android.test.InstrumentationTestRunner" />
   1221 
   1222         <if condition="${project.is.test}">
   1223         <then>
   1224             <property name="tested.project.absolute.dir" location="${tested.project.dir}" />
   1225 
   1226             <!-- Application package of the tested project extracted from its manifest file -->
   1227             <xpath input="${tested.project.absolute.dir}/AndroidManifest.xml"
   1228                     expression="/manifest/@package" output="tested.project.app.package" />
   1229 
   1230             <if condition="${emma.enabled}">
   1231                 <then>
   1232                     <getprojectpaths projectPath="${tested.project.absolute.dir}"
   1233                             binOut="tested.project.out.absolute.dir"
   1234                             srcOut="tested.project.source.absolute.dir" />
   1235 
   1236                     <getlibpath projectPath="${tested.project.absolute.dir}"
   1237                             libraryFolderPathOut="tested.project.lib.source.path"
   1238                             leaf="@{source.dir}" />
   1239 
   1240                 </then>
   1241             </if>
   1242 
   1243         </then>
   1244         <else>
   1245             <!-- this is a test app, the tested package is the app's own package -->
   1246             <property name="tested.project.app.package" value="${project.app.package}" />
   1247 
   1248             <if condition="${emma.enabled}">
   1249                 <then>
   1250                     <property name="tested.project.out.absolute.dir" value="${out.absolute.dir}" />
   1251                     <property name="tested.project.source.absolute.dir" value="${source.absolute.dir}" />
   1252 
   1253                     <getlibpath
   1254                             libraryFolderPathOut="tested.project.lib.source.path"
   1255                             leaf="@{source.dir}" />
   1256 
   1257                 </then>
   1258             </if>
   1259 
   1260         </else>
   1261         </if>
   1262 
   1263         <property name="emma.dump.file"
   1264                 value="/data/data/${tested.project.app.package}/coverage.ec" />
   1265 
   1266         <if condition="${emma.enabled}">
   1267             <then>
   1268                 <echo>Running tests...</echo>
   1269                 <run-tests-helper emma.enabled="true">
   1270                     <extra-instrument-args>
   1271                         <arg value="-e" />
   1272                            <arg value="coverageFile" />
   1273                            <arg value="${emma.dump.file}" />
   1274                     </extra-instrument-args>
   1275                 </run-tests-helper>
   1276 
   1277                 <echo level="info">Setting permission to download the coverage file...</echo>
   1278                 <exec executable="${adb}" failonerror="true">
   1279                     <arg line="${adb.device.arg}" />
   1280                     <arg value="shell" />
   1281                     <arg value="run-as" />
   1282                     <arg value="${tested.project.app.package}" />
   1283                     <arg value="chmod" />
   1284                     <arg value="644" />
   1285                     <arg value="${emma.dump.file}" />
   1286                 </exec>
   1287                 <echo level="info">Downloading coverage file into project directory...</echo>
   1288                 <exec executable="${adb}" failonerror="true">
   1289                     <arg line="${adb.device.arg}" />
   1290                     <arg value="pull" />
   1291                     <arg value="${emma.dump.file}" />
   1292                     <arg path="${out.absolute.dir}/coverage.ec" />
   1293                 </exec>
   1294 
   1295                 <pathconvert property="tested.project.lib.source.path.value" refid="tested.project.lib.source.path">
   1296                     <firstmatchmapper>
   1297                         <regexpmapper from='^([^ ]*)( .*)$$' to='"\1\2"'/>
   1298                         <identitymapper/>
   1299                     </firstmatchmapper>
   1300                 </pathconvert>
   1301 
   1302                 <echo level="info">Extracting coverage report...</echo>
   1303                 <emma>
   1304                     <property name="report.html.out.encoding" value="UTF-8" />
   1305                     <report sourcepath="${tested.project.source.absolute.dir}:${tested.project.lib.source.path.value}"
   1306                             verbosity="${verbosity}">
   1307                         <!-- TODO: report.dir or something like should be introduced if necessary -->
   1308                         <infileset file="${out.absolute.dir}/coverage.ec" />
   1309                         <infileset file="${tested.project.out.absolute.dir}/coverage.em" />
   1310                         <!-- TODO: reports in other, indicated by user formats -->
   1311                         <html outfile="${out.absolute.dir}/coverage.html" />
   1312                         <txt outfile="${out.absolute.dir}/coverage.txt" />
   1313                         <xml outfile="${out.absolute.dir}/coverage.xml" />
   1314                    </report>
   1315                 </emma>
   1316                 <echo level="info">Cleaning up temporary files...</echo>
   1317                 <delete file="${out.absolute.dir}/coverage.ec" />
   1318                 <delete file="${tested.project.out.absolute.dir}/coverage.em" />
   1319                 <exec executable="${adb}" failonerror="true">
   1320                     <arg line="${adb.device.arg}" />
   1321                     <arg value="shell" />
   1322                     <arg value="run-as" />
   1323                     <arg value="${tested.project.app.package}" />
   1324                     <arg value="rm" />
   1325                     <arg value="${emma.dump.file}" />
   1326                 </exec>
   1327                 <echo level="info">Saving the coverage reports in ${out.absolute.dir}</echo>
   1328             </then>
   1329             <else>
   1330                 <run-tests-helper />
   1331             </else>
   1332         </if>
   1333     </target>
   1334 
   1335     <!-- ******************************************************* -->
   1336     <!-- **********        Run Lint on the project     ********* -->
   1337     <!-- ******************************************************* -->
   1338 
   1339     <target name="lint"
   1340             description="Runs lint on the project to look for potential bugs" >
   1341         <lint executable="${lint}"
   1342               html="${lint.out.html}"
   1343               xml="${lint.out.xml}"
   1344               src="${source.absolute.dir}:${gen.absolute.dir}"
   1345               classpath="${out.classes.absolute.dir}" />
   1346     </target>
   1347 
   1348     <!-- ******************************************************* -->
   1349     <!-- ********** Install/uninstall specific targets ********* -->
   1350     <!-- ******************************************************* -->
   1351 
   1352     <target name="install"
   1353                 description="Installs the newly build package. Must be used in conjunction with a build target
   1354                             (debug/release/instrument). If the application was previously installed, the application
   1355                             is reinstalled if the signature matches." >
   1356         <!-- only do install if *not* a library project -->
   1357         <do-only-if-not-library elseText="Library project: nothing to install!" >
   1358             <if>
   1359                 <condition>
   1360                     <isset property="out.final.file" />
   1361                 </condition>
   1362                 <then>
   1363                     <if>
   1364                         <condition>
   1365                             <resourceexists>
   1366                                 <file file="${out.final.file}"/>
   1367                             </resourceexists>
   1368                         </condition>
   1369                         <then>
   1370                             <echo level="info">Installing ${out.final.file} onto default emulator or device...</echo>
   1371                             <exec executable="${adb}" failonerror="true">
   1372                                 <arg line="${adb.device.arg}" />
   1373                                 <arg value="install" />
   1374                                 <arg value="-r" />
   1375                                 <arg path="${out.final.file}" />
   1376                             </exec>
   1377 
   1378                             <!-- now install the tested project if applicable -->
   1379                             <!-- can't use project.is.test since the setup target might not have run -->
   1380                             <if>
   1381                                 <condition>
   1382                                     <and>
   1383                                         <isset property="tested.project.dir" />
   1384                                         <not>
   1385                                             <isset property="dont.do.deps" />
   1386                                         </not>
   1387                                     </and>
   1388                                 </condition>
   1389                                 <then>
   1390                                     <property name="tested.project.absolute.dir" location="${tested.project.dir}" />
   1391 
   1392                                     <!-- figure out which tested package to install based on emma.enabled -->
   1393                                     <condition property="tested.project.install.target" value="installi" else="installd">
   1394                                         <isset property="emma.enabled" />
   1395                                     </condition>
   1396                                     <subant target="${tested.project.install.target}" failonerror="true">
   1397                                         <fileset dir="${tested.project.absolute.dir}" includes="build.xml" />
   1398                                     </subant>
   1399                                 </then>
   1400                             </if>
   1401                         </then>
   1402                         <else>
   1403                             <fail message="File ${out.final.file} does not exist." />
   1404                         </else>
   1405                     </if>
   1406                 </then>
   1407                 <else>
   1408                     <echo>Install file not specified.</echo>
   1409                     <echo></echo>
   1410                     <echo>'ant install' now requires the build target to be specified as well.</echo>
   1411                     <echo></echo>
   1412                     <echo></echo>
   1413                     <echo>    ant debug install</echo>
   1414                     <echo>    ant release install</echo>
   1415                     <echo>    ant instrument install</echo>
   1416                     <echo>This will build the given package and install it.</echo>
   1417                     <echo></echo>
   1418                     <echo>Alternatively, you can use</echo>
   1419                     <echo>    ant installd</echo>
   1420                     <echo>    ant installr</echo>
   1421                     <echo>    ant installi</echo>
   1422                     <echo>    ant installt</echo>
   1423                     <echo>to only install an existing package (this will not rebuild the package.)</echo>
   1424                     <fail />
   1425                 </else>
   1426             </if>
   1427         </do-only-if-not-library>
   1428     </target>
   1429 
   1430     <target name="installd" depends="-set-debug-files, install"
   1431             description="Installs (only) the debug package." />
   1432     <target name="installr" depends="-set-release-mode, install"
   1433             description="Installs (only) the release package." />
   1434     <target name="installi" depends="-set-instrumented-mode, install"
   1435             description="Installs (only) the instrumented package." />
   1436     <target name="installt" depends="-test-project-check, installd"
   1437             description="Installs (only) the test and tested packages." />
   1438 
   1439 
   1440     <!-- Uninstalls the package from the default emulator/device -->
   1441     <target name="uninstall" depends="-setup"
   1442                 description="Uninstalls the application from a running emulator or device.">
   1443         <if>
   1444             <condition>
   1445                 <isset property="project.app.package" />
   1446             </condition>
   1447             <then>
   1448                 <uninstall-helper app.package="${project.app.package}" />
   1449             </then>
   1450             <else>
   1451                 <fail message="Could not find application package in manifest. Cannot run 'adb uninstall'." />
   1452             </else>
   1453         </if>
   1454 
   1455         <!-- Now uninstall the tested project, if applicable -->
   1456         <if>
   1457             <condition>
   1458                 <and>
   1459                     <istrue value="${project.is.test}" />
   1460                     <not>
   1461                         <isset property="dont.do.deps" />
   1462                     </not>
   1463                 </and>
   1464             </condition>
   1465             <then>
   1466                 <property name="tested.project.absolute.dir" location="${tested.project.dir}" />
   1467 
   1468                 <!-- Application package of the tested project extracted from its manifest file -->
   1469                 <xpath input="${tested.project.absolute.dir}/AndroidManifest.xml"
   1470                     expression="/manifest/@package" output="tested.project.app.package" />
   1471                 <if>
   1472                     <condition>
   1473                         <isset property="tested.project.app.package" />
   1474                     </condition>
   1475                     <then>
   1476                         <uninstall-helper app.package="${tested.project.app.package}" />
   1477                     </then>
   1478                     <else>
   1479                         <fail message="Could not find tested application package in manifest. Cannot run 'adb uninstall'." />
   1480                     </else>
   1481                 </if>
   1482             </then>
   1483         </if>
   1484 
   1485     </target>
   1486 
   1487 
   1488     <!-- ******************************************************* -->
   1489     <!-- ************************* Help ************************ -->
   1490     <!-- ******************************************************* -->
   1491 
   1492     <target name="help">
   1493         <!-- displays starts at col 13
   1494               |13                                                              80| -->
   1495         <echo>Android Ant Build. Available targets:</echo>
   1496         <echo>   help:      Displays this help.</echo>
   1497         <echo>   clean:     Removes output files created by other targets.</echo>
   1498         <echo>              This calls the same target on all dependent projects.</echo>
   1499         <echo>              Use 'ant nodeps clean' to only clean the local project</echo>
   1500         <echo>   debug:     Builds the application and signs it with a debug key.</echo>
   1501         <echo>              The 'nodeps' target can be used to only build the</echo>
   1502         <echo>              current project and ignore the libraries using:</echo>
   1503         <echo>              'ant nodeps debug'</echo>
   1504         <echo>   release:   Builds the application. The generated apk file must be</echo>
   1505         <echo>              signed before it is published.</echo>
   1506         <echo>              The 'nodeps' target can be used to only build the</echo>
   1507         <echo>              current project and ignore the libraries using:</echo>
   1508         <echo>              'ant nodeps release'</echo>
   1509         <echo>   instrument:Builds an instrumented package and signs it with a</echo>
   1510         <echo>              debug key.</echo>
   1511         <echo>   test:      Runs the tests. Project must be a test project and</echo>
   1512         <echo>              must have been built. Typical usage would be:</echo>
   1513         <echo>                  ant [emma] debug install test</echo>
   1514         <echo>   emma:      Transiently enables code coverage for subsequent</echo>
   1515         <echo>              targets.</echo>
   1516         <echo>   install:   Installs the newly build package. Must either be used</echo>
   1517         <echo>              in conjunction with a build target (debug/release/</echo>
   1518         <echo>              instrument) or with the proper suffix indicating</echo>
   1519         <echo>              which package to install (see below).</echo>
   1520         <echo>              If the application was previously installed, the</echo>
   1521         <echo>              application is reinstalled if the signature matches.</echo>
   1522         <echo>   installd:  Installs (only) the debug package.</echo>
   1523         <echo>   installr:  Installs (only) the release package.</echo>
   1524         <echo>   installi:  Installs (only) the instrumented package.</echo>
   1525         <echo>   installt:  Installs (only) the test and tested packages (unless</echo>
   1526         <echo>              nodeps is used as well.</echo>
   1527         <echo>   uninstall: Uninstalls the application from a running emulator or</echo>
   1528         <echo>              device. Also uninstall tested package if applicable</echo>
   1529         <echo>              unless 'nodeps' is used as well.</echo>
   1530     </target>
   1531 </project>
   1532