|
- /*
- ==============================================================================
-
- This file is part of the JUCE library.
- Copyright (c) 2015 - ROLI Ltd.
-
- Permission is granted to use this software under the terms of either:
- a) the GPL v2 (or any later version)
- b) the Affero GPL v3
-
- Details of these licenses can be found at: www.gnu.org/licenses
-
- JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
- WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
- A PARTICULAR PURPOSE. See the GNU General Public License for more details.
-
- ------------------------------------------------------------------------------
-
- To release a closed-source product which uses JUCE, commercial licenses are
- available: visit www.juce.com for more information.
-
- ==============================================================================
- */
-
- class AndroidAntProjectExporter : public AndroidProjectExporterBase
- {
- public:
- //==============================================================================
- bool canLaunchProject() override { return false; }
- bool launchProject() override { return false; }
- bool isAndroid() const override { return true; }
- bool usesMMFiles() const override { return false; }
- bool canCopeWithDuplicateFiles() override { return false; }
- bool supportsUserDefinedConfigurations() const override { return true; }
-
- bool isAndroidStudio() const override { return false; }
- bool isAndroidAnt() const override { return true; }
-
- bool supportsTargetType (ProjectType::Target::Type type) const override
- {
- switch (type)
- {
- case ProjectType::Target::GUIApp:
- case ProjectType::Target::StaticLibrary:
- return true;
- default:
- break;
- }
-
- return false;
- }
-
- //==============================================================================
- static const char* getName() { return "Android Ant Project"; }
- static const char* getValueTreeTypeName() { return "ANDROID"; }
-
- //==============================================================================
- Value getNDKToolchainVersionValue() { return getSetting (Ids::toolset); }
- String getNDKToolchainVersionString() const { return settings [Ids::toolset]; }
- Value getStaticLibrariesValue() { return getSetting (Ids::androidStaticLibraries); }
- String getStaticLibrariesString() const { return settings [Ids::androidStaticLibraries]; }
- Value getSharedLibrariesValue() { return getSetting (Ids::androidSharedLibraries); }
- String getSharedLibrariesString() const { return settings [Ids::androidSharedLibraries]; }
-
- //==============================================================================
- static AndroidAntProjectExporter* createForSettings (Project& project, const ValueTree& settings)
- {
- if (settings.hasType (getValueTreeTypeName()))
- return new AndroidAntProjectExporter (project, settings);
-
- return nullptr;
- }
-
- //==============================================================================
- AndroidAntProjectExporter (Project& p, const ValueTree& t)
- : AndroidProjectExporterBase (p, t)
- {
- name = getName();
-
- if (getTargetLocationString().isEmpty())
- getTargetLocationValue() = getDefaultBuildsRootFolder() + "Android";
- }
-
- //==============================================================================
- void createToolchainExporterProperties (PropertyListBuilder& props) override
- {
- props.add (new TextPropertyComponent (getNDKToolchainVersionValue(), "NDK Toolchain version", 32, false),
- "The variable NDK_TOOLCHAIN_VERSION in Application.mk - leave blank for a default value");
- }
-
- void createLibraryModuleExporterProperties (PropertyListBuilder& props) override
- {
- props.add (new TextPropertyComponent (getStaticLibrariesValue(), "Import static library modules", 8192, true),
- "Comma or whitespace delimited list of static libraries (.a) defined in NDK_MODULE_PATH.");
-
- props.add (new TextPropertyComponent (getSharedLibrariesValue(), "Import shared library modules", 8192, true),
- "Comma or whitespace delimited list of shared libraries (.so) defined in NDK_MODULE_PATH.");
- }
-
- //==============================================================================
- void create (const OwnedArray<LibraryModule>& modules) const override
- {
- AndroidProjectExporterBase::create (modules);
-
- const File target (getTargetFolder());
- const File jniFolder (target.getChildFile ("jni"));
-
- createDirectoryOrThrow (jniFolder);
- createDirectoryOrThrow (target.getChildFile ("res").getChildFile ("values"));
- createDirectoryOrThrow (target.getChildFile ("libs"));
- createDirectoryOrThrow (target.getChildFile ("bin"));
-
- {
- ScopedPointer<XmlElement> manifest (createManifestXML());
- writeXmlOrThrow (*manifest, target.getChildFile ("AndroidManifest.xml"), "utf-8", 100, true);
- }
-
- writeApplicationMk (jniFolder.getChildFile ("Application.mk"));
- writeAndroidMk (jniFolder.getChildFile ("Android.mk"));
-
- {
- ScopedPointer<XmlElement> antBuildXml (createAntBuildXML());
- writeXmlOrThrow (*antBuildXml, target.getChildFile ("build.xml"), "UTF-8", 100);
- }
-
- writeProjectPropertiesFile (target.getChildFile ("project.properties"));
- writeLocalPropertiesFile (target.getChildFile ("local.properties"));
- writeStringsFile (target.getChildFile ("res/values/strings.xml"));
- writeIcons (target.getChildFile ("res"));
- }
-
- //==============================================================================
- class AndroidBuildConfiguration : public BuildConfiguration
- {
- public:
- AndroidBuildConfiguration (Project& p, const ValueTree& settings, const ProjectExporter& e)
- : BuildConfiguration (p, settings, e)
- {
- if (getArchitectures().isEmpty())
- {
- if (isDebug())
- getArchitecturesValue() = "armeabi x86";
- else
- getArchitecturesValue() = "armeabi armeabi-v7a x86";
- }
- }
-
- Value getArchitecturesValue() { return getValue (Ids::androidArchitectures); }
- String getArchitectures() const { return config [Ids::androidArchitectures]; }
-
- var getDefaultOptimisationLevel() const override { return var ((int) (isDebug() ? gccO0 : gccO3)); }
-
- void createConfigProperties (PropertyListBuilder& props) override
- {
- addGCCOptimisationProperty (props);
-
- props.add (new TextPropertyComponent (getArchitecturesValue(), "Architectures", 256, false),
- "A list of the ARM architectures to build (for a fat binary).");
- }
- };
-
- BuildConfiguration::Ptr createBuildConfig (const ValueTree& v) const override
- {
- return new AndroidBuildConfiguration (project, v, *this);
- }
-
- private:
- //==============================================================================
- String getToolchainVersion() const
- {
- String v (getNDKToolchainVersionString());
- return v.isNotEmpty() ? v : "4.9";
- }
-
-
- //==============================================================================
- String getCppFlags() const
- {
- String flags ("-fsigned-char -fexceptions -frtti");
-
- if (! getNDKToolchainVersionString().startsWithIgnoreCase ("clang"))
- flags << " -Wno-psabi";
-
- return flags;
- }
-
- String getAppPlatform() const
- {
- int ndkVersion = androidMinimumSDK.get().getIntValue();
- if (ndkVersion == 9)
- ndkVersion = 10; // (doesn't seem to be a version '9')
-
- return "android-" + String (ndkVersion);
- }
-
- void writeApplicationMk (const File& file) const
- {
- MemoryOutputStream mo;
-
- mo << "# Automatically generated makefile, created by the Projucer" << newLine
- << "# Don't edit this file! Your changes will be overwritten when you re-save the Projucer project!" << newLine
- << newLine
- << "APP_STL := gnustl_static" << newLine
- << "APP_CPPFLAGS += " << getCppFlags() << newLine
- << "APP_PLATFORM := " << getAppPlatform() << newLine
- << "NDK_TOOLCHAIN_VERSION := " << getToolchainVersion() << newLine
- << newLine
- << "ifeq ($(NDK_DEBUG),1)" << newLine
- << " APP_ABI := " << getABIs<AndroidBuildConfiguration> (true) << newLine
- << "else" << newLine
- << " APP_ABI := " << getABIs<AndroidBuildConfiguration> (false) << newLine
- << "endif" << newLine;
-
- overwriteFileIfDifferentOrThrow (file, mo);
- }
-
- struct ShouldFileBeCompiledPredicate
- {
- bool operator() (const Project::Item& projectItem) const { return projectItem.shouldBeCompiled(); }
- };
-
- void writeAndroidMk (const File& file) const
- {
- Array<RelativePath> files;
-
- for (int i = 0; i < getAllGroups().size(); ++i)
- findAllProjectItemsWithPredicate (getAllGroups().getReference(i), files, ShouldFileBeCompiledPredicate());
-
- MemoryOutputStream mo;
- writeAndroidMk (mo, files);
-
- overwriteFileIfDifferentOrThrow (file, mo);
- }
-
- void writeAndroidMkVariableList (OutputStream& out, const String& variableName, const String& settingsValue) const
- {
- const StringArray separatedItems (getCommaOrWhitespaceSeparatedItems (settingsValue));
-
- if (separatedItems.size() > 0)
- out << newLine << variableName << " := " << separatedItems.joinIntoString (" ") << newLine;
- }
-
- void writeAndroidMk (OutputStream& out, const Array<RelativePath>& files) const
- {
- out << "# Automatically generated makefile, created by the Projucer" << newLine
- << "# Don't edit this file! Your changes will be overwritten when you re-save the Projucer project!" << newLine
- << newLine
- << "LOCAL_PATH := $(call my-dir)" << newLine
- << newLine
- << "include $(CLEAR_VARS)" << newLine
- << newLine
- << "ifeq ($(TARGET_ARCH_ABI), armeabi-v7a)" << newLine
- << " LOCAL_ARM_MODE := arm" << newLine
- << "endif" << newLine
- << newLine
- << "LOCAL_MODULE := juce_jni" << newLine
- << "LOCAL_SRC_FILES := \\" << newLine;
-
- for (int i = 0; i < files.size(); ++i)
- out << " " << (files.getReference(i).isAbsolute() ? "" : "../")
- << escapeSpaces (files.getReference(i).toUnixStyle()) << "\\" << newLine;
-
- writeAndroidMkVariableList (out, "LOCAL_STATIC_LIBRARIES", getStaticLibrariesString());
- writeAndroidMkVariableList (out, "LOCAL_SHARED_LIBRARIES", getSharedLibrariesString());
-
- out << newLine
- << "ifeq ($(NDK_DEBUG),1)" << newLine;
- writeConfigSettings (out, true);
- out << "else" << newLine;
- writeConfigSettings (out, false);
- out << "endif" << newLine
- << newLine
- << "include $(BUILD_SHARED_LIBRARY)" << newLine;
-
- StringArray importModules (getCommaOrWhitespaceSeparatedItems (getStaticLibrariesString()));
- importModules.addArray (getCommaOrWhitespaceSeparatedItems (getSharedLibrariesString()));
-
- for (int i = 0; i < importModules.size(); ++i)
- out << "$(call import-module," << importModules[i] << ")" << newLine;
- }
-
- void writeConfigSettings (OutputStream& out, bool forDebug) const
- {
- for (ConstConfigIterator config (*this); config.next();)
- {
- if (config->isDebug() == forDebug)
- {
- const AndroidBuildConfiguration& androidConfig = dynamic_cast<const AndroidBuildConfiguration&> (*config);
-
- String cppFlags;
- cppFlags << createCPPFlags (androidConfig)
- << (" " + replacePreprocessorTokens (androidConfig, getExtraCompilerFlagsString()).trim()).trimEnd()
- << newLine
- << getLDLIBS (androidConfig).trimEnd()
- << newLine;
-
- out << " LOCAL_CPPFLAGS += " << cppFlags;
- out << " LOCAL_CFLAGS += " << cppFlags;
- break;
- }
- }
- }
-
- String getLDLIBS (const AndroidBuildConfiguration& config) const
- {
- return " LOCAL_LDLIBS :=" + config.getGCCLibraryPathFlags()
- + " -llog -lGLESv2 -landroid -lEGL" + getExternalLibraryFlags (config)
- + " " + replacePreprocessorTokens (config, getExtraLinkerFlagsString());
- }
-
- String createIncludePathFlags (const BuildConfiguration& config) const
- {
- String flags;
- StringArray searchPaths (extraSearchPaths);
- searchPaths.addArray (config.getHeaderSearchPaths());
-
- searchPaths = getCleanedStringArray (searchPaths);
-
- for (int i = 0; i < searchPaths.size(); ++i)
- flags << " -I " << FileHelpers::unixStylePath (replacePreprocessorTokens (config, searchPaths[i])).quoted();
-
- return flags;
- }
-
- String createCPPFlags (const BuildConfiguration& config) const
- {
- StringPairArray defines;
- defines.set ("JUCE_ANDROID", "1");
- defines.set ("JUCE_ANDROID_API_VERSION", androidMinimumSDK.get());
- defines.set ("JUCE_ANDROID_ACTIVITY_CLASSNAME", getJNIActivityClassName().replaceCharacter ('/', '_'));
- defines.set ("JUCE_ANDROID_ACTIVITY_CLASSPATH", "\\\"" + getJNIActivityClassName() + "\\\"");
-
- String flags ("-fsigned-char -fexceptions -frtti");
-
- if (config.isDebug())
- {
- flags << " -g";
- defines.set ("DEBUG", "1");
- defines.set ("_DEBUG", "1");
- }
- else
- {
- defines.set ("NDEBUG", "1");
- }
-
- flags << createIncludePathFlags (config)
- << " -O" << config.getGCCOptimisationFlag();
-
- flags << " -std=gnu++11";
-
- defines = mergePreprocessorDefs (defines, getAllPreprocessorDefs ());
- return flags + createGCCPreprocessorFlags (defines);
- }
-
- //==============================================================================
- XmlElement* createAntBuildXML() const
- {
- XmlElement* proj = new XmlElement ("project");
- proj->setAttribute ("name", projectName);
- proj->setAttribute ("default", "debug");
-
- proj->createNewChildElement ("loadproperties")->setAttribute ("srcFile", "local.properties");
- proj->createNewChildElement ("loadproperties")->setAttribute ("srcFile", "project.properties");
-
- {
- XmlElement* target = proj->createNewChildElement ("target");
- target->setAttribute ("name", "clean");
- target->setAttribute ("depends", "android_rules.clean");
-
- target->createNewChildElement ("delete")->setAttribute ("dir", "libs");
- target->createNewChildElement ("delete")->setAttribute ("dir", "obj");
-
- XmlElement* executable = target->createNewChildElement ("exec");
- executable->setAttribute ("executable", "${ndk.dir}/ndk-build");
- executable->setAttribute ("dir", "${basedir}");
- executable->setAttribute ("failonerror", "true");
-
- executable->createNewChildElement ("arg")->setAttribute ("value", "clean");
- }
-
- {
- XmlElement* target = proj->createNewChildElement ("target");
- target->setAttribute ("name", "-pre-build");
-
- addDebugConditionClause (target, "makefileConfig", "Debug", "Release");
- addDebugConditionClause (target, "ndkDebugValue", "NDK_DEBUG=1", "NDK_DEBUG=0");
-
- String debugABIs, releaseABIs;
-
- for (ConstConfigIterator config (*this); config.next();)
- {
- const AndroidBuildConfiguration& androidConfig = dynamic_cast<const AndroidBuildConfiguration&> (*config);
-
- if (config->isDebug())
- debugABIs = androidConfig.getArchitectures();
- else
- releaseABIs = androidConfig.getArchitectures();
- }
-
- addDebugConditionClause (target, "app_abis", debugABIs, releaseABIs);
-
- XmlElement* executable = target->createNewChildElement ("exec");
- executable->setAttribute ("executable", "${ndk.dir}/ndk-build");
- executable->setAttribute ("dir", "${basedir}");
- executable->setAttribute ("failonerror", "true");
-
- executable->createNewChildElement ("arg")->setAttribute ("value", "--jobs=4");
- executable->createNewChildElement ("arg")->setAttribute ("value", "CONFIG=${makefileConfig}");
- executable->createNewChildElement ("arg")->setAttribute ("value", "${ndkDebugValue}");
- executable->createNewChildElement ("arg")->setAttribute ("value", "APP_ABI=${app_abis}");
-
- target->createNewChildElement ("delete")->setAttribute ("file", "${out.final.file}");
- target->createNewChildElement ("delete")->setAttribute ("file", "${out.packaged.file}");
- }
-
- proj->createNewChildElement ("import")->setAttribute ("file", "${sdk.dir}/tools/ant/build.xml");
-
- return proj;
- }
-
- void addDebugConditionClause (XmlElement* target, const String& property,
- const String& debugValue, const String& releaseValue) const
- {
- XmlElement* condition = target->createNewChildElement ("condition");
- condition->setAttribute ("property", property);
- condition->setAttribute ("value", debugValue);
- condition->setAttribute ("else", releaseValue);
-
- XmlElement* equals = condition->createNewChildElement ("equals");
- equals->setAttribute ("arg1", "${ant.project.invoked-targets}");
- equals->setAttribute ("arg2", "debug");
- }
-
- void writeProjectPropertiesFile (const File& file) const
- {
- MemoryOutputStream mo;
- mo << "# This file is used to override default values used by the Ant build system." << newLine
- << "# It is automatically generated - DO NOT EDIT IT or your changes will be lost!." << newLine
- << newLine
- << "target=" << getAppPlatform() << newLine
- << newLine;
-
- overwriteFileIfDifferentOrThrow (file, mo);
- }
-
- void writeLocalPropertiesFile (const File& file) const
- {
- MemoryOutputStream mo;
- mo << "# This file is used to override default values used by the Ant build system." << newLine
- << "# It is automatically generated by the Projucer - DO NOT EDIT IT or your changes will be lost!." << newLine
- << newLine
- << "sdk.dir=" << escapeSpaces (replacePreprocessorDefs (getAllPreprocessorDefs(), sdkPath.toString())) << newLine
- << "ndk.dir=" << escapeSpaces (replacePreprocessorDefs (getAllPreprocessorDefs(), ndkPath.toString())) << newLine
- << "key.store=" << androidKeyStore.get() << newLine
- << "key.alias=" << androidKeyAlias.get() << newLine
- << "key.store.password=" << androidKeyStorePass.get() << newLine
- << "key.alias.password=" << androidKeyAliasPass.get() << newLine
- << newLine;
-
- overwriteFileIfDifferentOrThrow (file, mo);
- }
-
- void writeStringsFile (const File& file) const
- {
- XmlElement strings ("resources");
- XmlElement* resourceName = strings.createNewChildElement ("string");
- resourceName->setAttribute ("name", "app_name");
- resourceName->addTextElement (projectName);
-
- writeXmlOrThrow (strings, file, "utf-8", 100);
- }
-
- //==============================================================================
- JUCE_DECLARE_NON_COPYABLE (AndroidAntProjectExporter)
- };
|