Urho3D/rakefile
2020-10-29 08:22:44 +08:00

679 lines
20 KiB
Plaintext

#
# Copyright (c) 2008-2020 the Urho3D project.
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
# THE SOFTWARE.
#
task default: :build
desc 'Invoke CMake to configure and generate a build tree'
task :cmake => [:init] do
if ENV['CI']
system 'cmake --version' or abort 'Failed to find CMake'
end
next if ENV['PLATFORM'] == 'android' || (Dir.exist?("#{build_tree}") and not ARGV.include?('cmake'))
script = "script/cmake_#{ENV['GENERATOR']}#{ENV['OS'] ? '.bat' : '.sh'}"
build_options = /linux|macOS|win/ =~ ENV['PLATFORM'] ? '' : "-D #{ENV['PLATFORM'].upcase}=1"
File.readlines('script/.build-options').each { |var|
var.chomp!
build_options = "#{build_options} -D #{var}=#{ENV[var]}" if ENV[var]
}
system %Q{#{script} "#{build_tree}" #{build_options}} or abort
end
desc 'Clean the build tree'
task :clean => [:init] do
if ENV['PLATFORM'] == 'android'
Rake::Task[:gradle].invoke('clean')
next
end
system build_target('clean') or abort
end
desc 'Build the software'
task :build, [:target] => [:cmake] do |_, args|
system "ccache -z" if ENV['USE_CCACHE']
if ENV['PLATFORM'] == 'android'
Rake::Task[:gradle].invoke('build -x test')
system "ccache -s" if ENV['USE_CCACHE']
next
end
filter = ''
case ENV['GENERATOR']
when 'xcode'
concurrent = '' # Assume xcodebuild will do the right things without the '-jobs'
filter = '|xcpretty -c && exit ${PIPESTATUS[0]}' if system('xcpretty -v >/dev/null 2>&1')
when 'vs'
concurrent = '/maxCpuCount'
else
concurrent = "-j #{$max_jobs}"
filter = "2>#{lint_err_file}" if ENV['URHO3D_LINT']
end
system "#{build_target(args[:target])} -- #{concurrent} #{ENV['BUILD_PARAMS']} #{filter}" or abort
system "ccache -s" if ENV['USE_CCACHE']
end
desc 'Test the software'
task :test => [:init] do
if ENV['PLATFORM'] == 'android'
Rake::Task[:gradle].invoke('test')
next
elsif ENV['URHO3D_LINT'] == '1'
Rake::Task[:lint].invoke
next
elsif ENV['URHO3D_STYLE'] == '1'
Rake::Task[:style].invoke
next
end
wrapper = ENV['CI'] && ENV['PLATFORM'] == 'linux' ? 'xvfb-run' : ''
test = /xcode|vs/ =~ ENV['GENERATOR'] ? 'RUN_TESTS' : 'test'
system build_target(test, wrapper) or abort
end
desc 'Generate documentation'
task :doc => [:init] do
if ENV['PLATFORM'] == 'android'
Rake::Task[:gradle].invoke('documentationZip')
next
end
system build_target('doc') or abort
end
desc 'Install the software'
task :install, [:prefix] => [:init] do |_, args|
if ENV['PLATFORM'] == 'android'
Rake::Task[:gradle].invoke('publishToMavenLocal')
next
end
wrapper = args[:prefix] && !ENV['OS'] ? "DESTDIR=#{verify_path(args[:prefix])}" : ''
system build_target('install', wrapper) or abort
end
desc 'Package build artifact'
task :package => [:init] do
next if ENV['PLATFORM'] == 'android'
wrapper = /linux|rpi|arm/ =~ ENV['PLATFORM'] && ENV['URHO3D_64BIT'] == '0' ? 'setarch i686' : ''
system build_target('package', wrapper) or abort
end
desc 'Publish build artifact'
task :publish => [:init] do
if ENV['PLATFORM'] == 'android'
Rake::Task[:gradle].invoke('publish')
end
end
desc 'Create a new project'
task :new, [:name, :parent_dir, :use_copy] => [:init] do |_, args|
args.with_defaults(:name => 'UrhoApp', :parent_dir => '~/projects', :use_copy => false)
name = args[:name]
parent_dir = verify_path(args[:parent_dir])
dir = "#{parent_dir}/#{name}"
use_copy = args[:use_copy] || dockerized?
abort "The directory '#{dir}' already exists!" if Dir.exists?(dir)
puts "Creating a new project in #{dir}..."
func = FileUtils.method(use_copy ? :cp_r : :ln_s)
source_tree(name).split("\n").each do |it|
next if it.empty?
dirname, basename = /\// =~ it.split(/\s*<-\s*/).first ? it.match(/([^\s<]+)\/(.+)/).captures : ['.', it]
FileUtils.mkdir_p("#{dir}/#{dirname}")
if (matched = basename.match(/(?<basename>\S+)\s*<-\s*:(?<symbol>\S+)/))
File.write("#{dir}/#{dirname}/#{matched[:basename]}", method(matched[:symbol].to_sym).call(name))
elsif (matched = basename.match(/(?<basename>\S+)\s*<-\s*(?<dirname>\S+)/))
func.call(verify_path("#{matched[:dirname]}/#{matched[:basename]}"), "#{dir}/#{dirname}")
else
func.call(verify_path(it), "#{dir}/#{dirname}")
end
end
puts "Done!"
end
### Internal tasks ###
task :ci do
ENV['URHO3D_PCH'] = '0' if ENV['PLATFORM'] == 'linux-gcc' # TODO - PCH causes cache miss on initial build for Linux/GCC, why?
platform_modifier = /(.+?)-(.+)/.match(ENV['PLATFORM'])
if platform_modifier
ENV['PLATFORM'] = platform_modifier[1]
ENV['MODIFIER'] = platform_modifier[2]
end
case ENV['HOST']
when 'macOS'
ENV['GENERATOR'] = ENV['URHO3D_DEPLOYMENT_TARGET'] = 'generic' if ENV['MODIFIER'] == 'make'
ENV['BUILD_PARAMS'] = '-sdk iphonesimulator' if ENV['PLATFORM'] == 'iOS'
ENV['BUILD_PARAMS'] = '-sdk appletvsimulator' if ENV['PLATFORM'] == 'tvOS'
when 'windows'
if ENV['MODIFIER'] == 'gcc'
ENV['URHO3D_DEPLOYMENT_TARGET'] = 'generic'
ENV['GENERATOR'] = 'mingw'
end
else
ENV['URHO3D_DEPLOYMENT_TARGET'] = 'generic' if /linux|mingw/ =~ ENV['PLATFORM']
if /clang/ =~ ENV['MODIFIER']
ENV['CC'] = 'clang'
ENV['CXX'] = 'clang++'
end
end
ENV['BUILD_TREE'] = 'build/ci'
ENV['CMAKE_BUILD_TYPE'] = ENV['BUILD_TYPE'] == 'dbg' ? 'Debug' : 'Release' if /dbg|rel/ =~ ENV['BUILD_TYPE']
case ENV['GRAPHICS_API']
when 'DX11'
ENV['URHO3D_D3D11'] = '1'
when 'DX9'
ENV['URHO3D_OPENGL'] = '0' # Need to make this explicit because 'MINGW' default to use OpenGL otherwise
when 'OpenGL'
ENV['URHO3D_OPENGL'] = '1'
else
# Do nothing
end
ENV['URHO3D_LIB_TYPE'] = ENV['LIB_TYPE'].upcase if /static|shared/ =~ ENV['LIB_TYPE']
ENV['URHO3D_TESTING'] = '1' if /linux|macOS|win/ =~ ENV['PLATFORM']
ENV['URHO3D_LINT'] = '1' if ENV['MODIFIER'] == 'clang-tidy'
ENV['URHO3D_STYLE'] = '1' if ENV['MODIFIER'] == 'clang-format'
# Enable all the bells and whistles
%w[URHO3D_DATABASE_SQLITE URHO3D_EXTRAS].each { |it| ENV[it] = '1' }
end
task :source_checksum do
require 'digest'
sha256_final = Digest::SHA256.new
sha256_iter = Digest::SHA256
Dir['Source/**/*.{c,h}*'].each { |it| sha256_final << sha256_iter.file(it).hexdigest }
puts "::set-output name=hexdigest::#{sha256_final.hexdigest}"
end
task :update_dot_files do
system 'bash', '-c', %q{
perl -ne 'undef $/; print $1 if /(Build Option.*?(?=\n\n))/s' Docs/GettingStarted.dox \
|tail -n +3 |cut -d'|' -f2 |tr -d [:blank:] >script/.build-options && \
echo URHO3D_LINT >>script/.build-options && \
cat script/.build-options <(perl -ne 'while (/([A-Z_]+):.+?/g) {print "$1\n"}' .github/workflows/main.yml) \
<(perl -ne 'while (/ENV\[\x27(\w+)\x27\]/g) {print "$1\n"}' rakefile) \
<(perl -ne 'while (/System.getenv\\("(\w+)"\\)/g) {print "$1\n"}' android/urho3d-lib/build.gradle.kts) \
|sort |uniq |grep -Ev '^(HOME|PATH)$' >script/.env-file
} or abort 'Failed to update dot files'
end
task :init do
next if $max_jobs
Rake::Task[:ci].invoke if ENV['CI']
case build_host
when /linux/
$max_jobs = `grep -c processor /proc/cpuinfo`.chomp
ENV['GENERATOR'] = 'generic' unless ENV['GENERATOR']
ENV['PLATFORM'] = 'linux' unless ENV['PLATFORM']
when /darwin|macOS/
$max_jobs = `sysctl -n hw.logicalcpu`.chomp
ENV['GENERATOR'] = 'xcode' unless ENV['GENERATOR']
ENV['PLATFORM'] = 'macOS' unless ENV['PLATFORM']
when /win32|mingw|mswin|windows/
require 'win32ole'
WIN32OLE.connect('winmgmts://').ExecQuery("select NumberOfLogicalProcessors from Win32_ComputerSystem").each { |it|
$max_jobs = it.NumberOfLogicalProcessors
}
ENV['GENERATOR'] = 'vs' unless ENV['GENERATOR']
ENV['PLATFORM'] = 'win' unless ENV['PLATFORM']
else
abort "Unsupported host system: #{build_host}"
end
# The 'ARCH' env-var, when set, has higher precedence than the 'URHO3D_64BIT' env-var
ENV['URHO3D_64BIT'] = ENV['ARCH'] == '32' ? '0' : '1' if /32|64/ =~ ENV['ARCH']
end
task :gradle, [:task] do |_, args|
system "./gradlew #{args[:task]} #{ENV['CI'] ? '--console plain' : ''}" or abort
end
task :lint do
lint_err = File.read(lint_err_file)
puts lint_err
# TODO: Tighten the check by failing the job later
# abort 'Failed to pass linter checks' unless lint_err.empty?
# puts 'Passed the linter checks'
end
task :style do
system 'bash', '-c', %q{
git diff --name-only HEAD~ -- Source \
|grep -v ThirdParty \
|grep -P '\.(?:c|cpp|h|hpp)' \
|xargs clang-format -n -Werror 2>&1 \
|tee build/clang-format.out \
&& exit ${PIPESTATUS[3]}
} or abort 'Failed to pass style checks'
puts 'Passed the style checks'
end
### Internal methods ###
def build_host
ENV['HOST'] || RUBY_PLATFORM
end
def build_tree
ENV['BUILD_TREE'] || "build/#{dockerized? ? 'dockerized-' : ''}#{ENV['PLATFORM'].downcase}"
end
def build_config
/xcode|vs/ =~ ENV['GENERATOR'] ? "--config #{ENV.fetch('CONFIG', 'Release')}" : ''
end
def build_target(tgt, wrapper = '')
%Q{#{wrapper} cmake --build "#{build_tree}" #{build_config} #{tgt ? "--target #{tgt}" : ''}}
end
def lint_err_file
'build/clang-tidy.out'
end
def verify_path(path)
require 'pathname'
begin
Pathname.new(File.expand_path(path)).realdirpath.to_s
rescue
abort "The specified path '#{path}' is invalid!"
end
end
def dockerized?
File.exists?('/entrypoint.sh')
end
def source_tree(name)
<<-EOF
bin/CoreData
bin/Data/Materials/Mushroom.xml
bin/Data/Models/Mushroom.mdl
bin/Data/Music/Ninja Gods.ogg
bin/Data/Textures/Mushroom.dds
bin/Data/Textures/UrhoIcon.icns
bin/Data/Textures/UrhoIcon.png
cmake
gradle
script
src/main/cpp/#{name}.cpp <- :urho_app_cpp
src/main/cpp/#{name}.h <- :urho_app_h
src/main/java/io/urho3d/#{name.downcase}/MainActivity.kt <- :main_activity_kt
#{Dir.chdir('android/launcher-app') { (Dir['src/main/res/{drawable,mipmap}*'] + ['']).join("<- android/launcher-app/src/main/res\n") }}
src/main/res/values/strings.xml <- :strings_xml
src/main/AndroidManifest.xml <- :android_manifest_xml
build.gradle.kts <- :build_gradle_kts
CMakeLists.txt <- :cmake_lists_txt
proguard-rules.pro <- android/launcher-app
gradle.properties
gradlew
gradlew.bat
rakefile
settings.gradle.kts <- :settings_gradle_kts
.clang-format
.clang-tidy
.gitattributes <- :gitattributes
.gitignore <- :gitignore
EOF
end
def settings_gradle_kts(name)
<<-EOF
pluginManagement {
resolutionStrategy {
eachPlugin {
when {
requested.id.id.startsWith("com.android.") ->
useModule("com.android.tools.build:gradle:4.0.2")
requested.id.id.startsWith("org.jetbrains.kotlin.") ->
useVersion(embeddedKotlinVersion)
}
}
}
repositories {
gradlePluginPortal()
google()
jcenter()
}
}
rootProject.name = "#{name}"
EOF
end
def build_gradle_kts(name)
ndk_version = '21.3.6528147'
cmake_version = '3.17.3+'
build_staging_dir = '.cxx'
sdk_version = 30
min_sdk_version = 18
aar_version = '1.8-ALPHA-SNAPSHOT'
type = ENV.fetch('URHO3D_LIB_TYPE', 'STATIC').downcase
<<-EOF
plugins {
id("com.android.application")
kotlin("android")
kotlin("android.extensions")
}
android {
ndkVersion = "#{ndk_version}"
compileSdkVersion(#{sdk_version})
defaultConfig {
minSdkVersion(#{min_sdk_version})
targetSdkVersion(#{sdk_version})
applicationId = "io.urho3d.#{name}"
versionCode = 1
versionName = "1.0"
testInstrumentationRunner = "android.support.test.runner.AndroidJUnitRunner"
externalNativeBuild {
cmake {
arguments.apply {
System.getenv("ANDROID_CCACHE")?.let { add("-D ANDROID_CCACHE=$it") }
add("-D JNI_DIR=${project.file("#{build_staging_dir}")}")
// Pass along matching env-vars as CMake build options
addAll(project.file("script/.build-options")
.readLines()
.mapNotNull { variable -> System.getenv(variable)?.let { "-D $variable=$it" } }
)
}
}
}
splits {
abi {
isEnable = project.hasProperty("ANDROID_ABI")
reset()
include(
*(project.findProperty("ANDROID_ABI") as String? ?: "")
.split(',')
.toTypedArray()
)
}
}
}
buildTypes {
named("release") {
isMinifyEnabled = false
proguardFiles(getDefaultProguardFile("proguard-android-optimize.txt"), "proguard-rules.pro")
}
}
lintOptions {
isAbortOnError = false
}
externalNativeBuild {
cmake {
version = "#{cmake_version}"
path = project.file("CMakeLists.txt")
setBuildStagingDirectory("#{build_staging_dir}")
}
}
sourceSets {
named("main") {
assets.srcDir("bin")
}
}
}
repositories {
google()
jcenter()
// Remove below two repos if you are only using released AAR from JCenter
mavenLocal()
if (System.getenv("GITHUB_ACTOR") != null && System.getenv("GITHUB_TOKEN") != null) {
maven {
name = "GitHubPackages"
url = uri("https://maven.pkg.github.com/urho3d/Urho3D")
credentials {
username = System.getenv("GITHUB_ACTOR")
password = System.getenv("GITHUB_TOKEN")
}
}
}
}
val urhoReleaseImpl by configurations.creating { isCanBeResolved = true }
configurations.releaseImplementation.get().extendsFrom(urhoReleaseImpl)
val urhoDebugImpl by configurations.creating { isCanBeResolved = true }
configurations.debugImplementation.get().extendsFrom(urhoDebugImpl)
dependencies {
urhoReleaseImpl("io.urho3d:urho3d-lib-#{type}:#{aar_version}")
urhoDebugImpl("io.urho3d:urho3d-lib-#{type}-debug:#{aar_version}")
implementation(fileTree(mapOf("dir" to "libs", "include" to listOf("*.jar"))))
implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk8:$embeddedKotlinVersion")
implementation("androidx.core:core-ktx:1.3.2")
implementation("androidx.appcompat:appcompat:1.2.0")
implementation("androidx.constraintlayout:constraintlayout:2.0.2")
testImplementation("junit:junit:4.13.1")
androidTestImplementation("androidx.test:runner:1.3.0")
androidTestImplementation("androidx.test.espresso:espresso-core:3.3.0")
}
afterEvaluate {
android.buildTypes.forEach { buildType ->
val config = buildType.name.capitalize()
val unzipTaskName = "unzipJni$config"
tasks {
"generateJsonModel$config" {
dependsOn(unzipTaskName)
}
register<Copy>(unzipTaskName) {
val aar = configurations["urho${config}Impl"].resolve().first { it.name.startsWith("urho3d-lib") }
from(zipTree(aar))
include("urho3d/**")
into(android.externalNativeBuild.cmake.buildStagingDirectory)
}
}
}
}
tasks {
register<Delete>("cleanAll") {
dependsOn("clean")
delete = setOf(android.externalNativeBuild.cmake.buildStagingDirectory)
}
}
EOF
end
def strings_xml(name)
<<-EOF
<resources>
<string name="app_name">#{name}</string>
</resources>
EOF
end
def android_manifest_xml(name)
<<-EOF
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="io.urho3d.#{name.downcase}">
<application
android:allowBackup="true"
android:hardwareAccelerated="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true">
<activity
android:name=".MainActivity"
android:configChanges="keyboardHidden|orientation|screenSize"
android:screenOrientation="landscape"
android:theme="@android:style/Theme.NoTitleBar.Fullscreen">
<intent-filter>
<action android:name="android.intent.action.MAIN"/>
<category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
</application>
</manifest>
EOF
end
def main_activity_kt(name)
<<-EOF
package io.urho3d.#{name.downcase}
import io.urho3d.UrhoActivity
class MainActivity : UrhoActivity()
EOF
end
def cmake_lists_txt(name)
<<-EOF
cmake_minimum_required(VERSION 3.10.2)
project(#{name})
set(CMAKE_MODULE_PATH ${CMAKE_SOURCE_DIR}/cmake/Modules)
include(UrhoCommon)
set(TARGET_NAME #{name})
define_source_files(GLOB_CPP_PATTERNS src/main/cpp/*.cpp GLOB_H_PATTERNS src/main/cpp/*.h RECURSE GROUP)
setup_main_executable()
setup_test()
EOF
end
def urho_app_h(name)
<<-EOF
#pragma once
#include <Urho3D/Urho3DAll.h>
class #{name} : public Application
{
URHO3D_OBJECT(#{name}, Application);
public:
explicit #{name}(Context* context);
void Start() override;
private:
SharedPtr<Scene> scene_;
};
EOF
end
def urho_app_cpp(name)
<<-EOF
#include "#{name}.h"
URHO3D_DEFINE_APPLICATION_MAIN(#{name});
#{name}::#{name}(Context* context)
: Application(context)
{
}
void #{name}::Start()
{
auto* cache = GetSubsystem<ResourceCache>();
auto* graphics = GetSubsystem<Graphics>();
graphics->SetWindowIcon(cache->GetResource<Image>("Textures/UrhoIcon.png"));
graphics->SetWindowTitle("#{name}");
scene_ = new Scene(context_);
scene_->CreateComponent<Octree>();
Node* objectNode = scene_->CreateChild();
auto* object = objectNode->CreateComponent<StaticModel>();
object->SetModel(cache->GetResource<Model>("Models/Mushroom.mdl"));
object->SetMaterial(cache->GetResource<Material>("Materials/Mushroom.xml"));
auto* sound = scene_->CreateComponent<SoundSource>();
sound->SetSoundType(SOUND_MUSIC);
auto* music = cache->GetResource<Sound>("Music/Ninja Gods.ogg");
music->SetLooped(true);
sound->Play(music);
Node* lightNode = scene_->CreateChild();
auto* light = lightNode->CreateComponent<Light>();
light->SetLightType(LIGHT_DIRECTIONAL);
lightNode->SetDirection(Vector3(0.6f, -1.f, 0.8f));
Node* cameraNode = scene_->CreateChild();
auto* camera = cameraNode->CreateComponent<Camera>();
cameraNode->SetPosition(Vector3(0.f, 0.3f, -3.f));
GetSubsystem<Renderer>()->SetViewport(0, new Viewport(context_, scene_, camera));
SubscribeToEvent(E_KEYUP, [&](StringHash, VariantMap&) { engine_->Exit(); });
SubscribeToEvent(E_UPDATE, [=](StringHash, VariantMap& eventData) {
objectNode->Yaw(eventData[Update::P_TIMESTEP].GetFloat());
});
}
EOF
end
def gitattributes(_)
<<-EOF
*.h linguist-language=C++
EOF
end
def gitignore(_)
<<-EOF
# Code::Blocks project settings
/*.cbp
# Codelite project settings
/*.project
/*.workspace
# Gradle project settings
/local.properties
.gradle/
build/
.cxx/
# JetBrains IDE project settings
/.idea/
/cmake-build-*/
*.iml
# KDevelop project settings
/*.kdev?
# Qt Creator project settings
/CMakeLists.txt.user
# Visual Studio project settings
/CMakeSettings.json
/.vs/
/out/
# Misc.
*~
*.swp
.DS_Store
*.log
*.bak
Thumbs.db
.directory
EOF
end
# Load custom rake scripts
Dir['.rake/*.rake'].each { |r| load r }
# vi: set ts=2 sw=2 expandtab: