Skip to content
This repository has been archived by the owner on Oct 25, 2021. It is now read-only.

Commit

Permalink
VS 2019 Refresh (#739)
Browse files Browse the repository at this point in the history
* Setup Azure DevOps for CI
  * Install Android NDK Bundle on Windows build agents
  * Workaround for `JAVA_HOME` on Windows build agents
* Support for Android NDK r19 and r20
* Updated Cake scripts to support VS 2019 on Windows
* Pin NuGet download URL to v4.7.3
* Bump to xamarin/xamarin-android-tools/master@294f447
  * Changes: dotnet/android-tools@4c00c22...294f447
* Fix for `mono-support.h` on Windows

Fixes:

    c:\program files (x86)\mono\include\mono-2.0\mono\utils\mono-publib.h(94): error C2371: 'mono_unichar2': redefinition; different basic types [D:\a\1\s\tests\common\mk\common.Tests.vcxproj]
  • Loading branch information
jonathanpeppers authored Sep 9, 2019
1 parent c0f68d0 commit 9cd5d13
Show file tree
Hide file tree
Showing 10 changed files with 156 additions and 34 deletions.
42 changes: 42 additions & 0 deletions azure-pipelines.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
# https://aka.ms/yaml

name: $(BuildID)

trigger:
- master

variables:
Mono.Windows.Url: https://download.mono-project.com/archive/6.0.0/windows-installer/mono-6.0.0.319-gtksharp-2.12.45-win32-0.msi
SdkManager.Windows: 'C:\Program Files (x86)\Android\android-sdk\tools\bin\sdkmanager'
DOTNET_SKIP_FIRST_TIME_EXPERIENCE: true

jobs:

- job: windows
pool:
name: Hosted VS2017
demands: msbuild
steps:
- powershell: |
echo y | & "$(SdkManager.Windows)" ndk-bundle
displayName: install Android NDK
- powershell: |
Invoke-WebRequest $(Mono.Windows.Url) -OutFile mono.msi -Verbose
Start-Process msiexec -Wait -ArgumentList '/i mono.msi /quiet /l*v mono.log'
Get-Content mono.log
displayName: install mono
- powershell: |
.\build.ps1 -t Jenkins -v diagnostic
displayName: build and test
- job: mac
pool:
name: Hosted macOS
demands: msbuild
steps:
- bash: |
source build/SetupCI.sh
displayName: setup ci
- bash: |
./build.sh -t Travis -v diagnostic
displayName: build and test
23 changes: 17 additions & 6 deletions binder/Compilation.cs
Original file line number Diff line number Diff line change
Expand Up @@ -687,15 +687,12 @@ bool CompileNDK(IEnumerable<string> files)
var libName = $"lib{name}.so";
var ndkPath = XamarinAndroid.AndroidSdk.AndroidNdkPath;

foreach (var abi in new[] { "armeabi", "armeabi-v7a", "arm64-v8a", "x86", "x86_64" })
foreach (var abi in new[] { "armeabi-v7a", "arm64-v8a", "x86", "x86_64" })
{
string extra = string.Empty;
AndroidTargetArch targetArch;
switch (abi)
{
case "armeabi":
targetArch = AndroidTargetArch.Arm;
break;
case "armeabi-v7a":
targetArch = AndroidTargetArch.Arm;
extra = " -march=armv7-a -mfloat-abi=softfp -mfpu=vfpv3-d16";
Expand All @@ -715,8 +712,17 @@ bool CompileNDK(IEnumerable<string> files)
throw new NotImplementedException();
}

bool isLLVM = false;
var clangBin = NdkUtil.GetNdkClangBin(Path.Combine(ndkPath, "toolchains"), targetArch);
var systemInclude = NdkUtil.GetNdkPlatformIncludePath(ndkPath, targetArch, XamarinAndroid.ApiLevel);
if (string.IsNullOrEmpty(clangBin)) {
clangBin = NdkUtil.GetNdkClangBin(Path.Combine(ndkPath, "toolchains", "llvm"), targetArch);
isLLVM = true;
}
if (string.IsNullOrEmpty (clangBin))
{
throw new Exception($"Unable to find NDK toolchain for {abi}!");
}
var systemInclude = NdkUtil.GetNdkPlatformIncludePath(ndkPath, targetArch, XamarinAndroid.ApiLevel, isLLVM);
var monoDroidPath = Path.Combine(XamarinAndroid.LibraryPath, abi);
var abiDir = Path.Combine(Options.OutputDir, "android", "jni", abi);
var outputPath = Path.Combine(abiDir, libName);
Expand All @@ -731,9 +737,14 @@ bool CompileNDK(IEnumerable<string> files)
$"-I\"{monoPath}\"",
$"-L\"{monoDroidPath}\" -lmonosgen-2.0 -lmono-android.release",
string.Join(" ", files.ToList()),
"--std=c99",
"--std=c11",
"-fPIC",
$"-shared -o {outputPath}",
};
if (isLLVM)
{
args.Add($"--target={NdkUtil.GetLlvmToolchainTarget(targetArch, XamarinAndroid.ApiLevel)}");
}

var invocation = string.Join(" ", args);
var output = Invoke(clangBin, invocation);
Expand Down
54 changes: 45 additions & 9 deletions binder/Utils/NdkUtils.cs
Original file line number Diff line number Diff line change
Expand Up @@ -48,9 +48,19 @@ public static string GetNdkTool(string androidNdkPath, AndroidTargetArch arch, s
throw new Exception($"C compiler for target {arch} was not found. Tried paths: \"{string.Join("; ", toolPaths)}\"");
}

public static string GetNdkPlatformIncludePath(string androidNdkPath, AndroidTargetArch arch, int level)
public static string GetNdkPlatformIncludePath(string androidNdkPath, AndroidTargetArch arch, int level, bool isLLVM)
{
string path = Path.Combine(androidNdkPath, "platforms", "android-" + level, "arch-" + GetPlatformArch(arch));
string path;
if (isLLVM)
{
var toolchainDir = Path.Combine(androidNdkPath, "toolchains", "llvm", "prebuilt");
var machineDir = Directory.GetDirectories(Path.Combine(toolchainDir)).First();
path = Path.Combine(machineDir, "sysroot");
}
else
{
path = Path.Combine(androidNdkPath, "platforms", "android-" + level, "arch-" + GetPlatformArch(arch));
}
if (!Directory.Exists(path))
throw new InvalidOperationException(String.Format("Platform header files for target {0} and API Level {1} was not found. Expected path is \"{2}\"", arch, level, path));
return path;
Expand Down Expand Up @@ -86,6 +96,10 @@ static string[] GetNdkToolchainDirectories(string toolchainsPath, AndroidTargetA
{
if (!Directory.Exists(toolchainsPath))
throw new Exception($"Missing Android NDK toolchains directory '{toolchainsPath}'. Please install the Android NDK.");
if (string.Compare(Path.GetFileName(toolchainsPath), "llvm", StringComparison.OrdinalIgnoreCase) == 0)
{
return new[] { toolchainsPath };
}
switch (arch)
{
case AndroidTargetArch.Arm:
Expand Down Expand Up @@ -125,17 +139,39 @@ static string GetNdkToolchainPrefix(AndroidTargetArch arch)
}
}

public static string GetLlvmToolchainTarget (AndroidTargetArch arch, int apiLevel)
{
switch (arch)
{
case AndroidTargetArch.Arm:
return $"armv7a-linux-androideabi{apiLevel}";
case AndroidTargetArch.Arm64:
return $"aarch64-linux-android{apiLevel}";
case AndroidTargetArch.X86:
return $"i686-linux-android{apiLevel}";
case AndroidTargetArch.X86_64:
return $"x86_64-linux-android{apiLevel}";
default:
return string.Empty;
}
}

public static string GetNdkClangBin(string toolchainsPath, AndroidTargetArch arch)
{
var toolChainDir = GetNdkToolchainDirectories(toolchainsPath, arch).First();
var machineDir = Directory.GetDirectories(Path.Combine(toolChainDir, "prebuilt")).First();
foreach (var toolChainDir in GetNdkToolchainDirectories(toolchainsPath, arch))
{
var machineDir = Directory.GetDirectories(Path.Combine(toolChainDir, "prebuilt")).First();

var executableSuffix = Platform.IsWindows ? ".exe" : string.Empty;
var gcc = Path.Combine(machineDir, "bin", GetNdkToolchainPrefix(arch) + "gcc" + executableSuffix);
if (File.Exists(gcc))
return gcc;
var executableSuffix = Platform.IsWindows ? ".exe" : string.Empty;
var gcc = Path.Combine(machineDir, "bin", GetNdkToolchainPrefix(arch) + "gcc" + executableSuffix);
if (File.Exists(gcc))
return gcc;

throw new Exception($"Unable to find NDK toolchain for {arch}!");
var clang = Path.Combine(machineDir, "bin", "clang" + executableSuffix);
if (File.Exists(clang))
return clang;
}
return null;
}

static bool GetNdkToolchainRelease(string androidNdkPath, out string version)
Expand Down
2 changes: 1 addition & 1 deletion build.cake
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ Task("Build-Binder")
.IsDependentOn("NuGet-Restore")
.Does(() =>
{
MSBuild("./build/projects/Embeddinator-4000.csproj", settings => settings.SetConfiguration(configuration));
MSBuild("./build/projects/Embeddinator-4000.csproj", MSBuildSettings());
});

Task("Generate-Project-Files")
Expand Down
2 changes: 1 addition & 1 deletion build.ps1
Original file line number Diff line number Diff line change
Expand Up @@ -89,7 +89,7 @@ if(!$PSScriptRoot){
$TOOLS_DIR = Join-Path $PSScriptRoot ".cake"
$NUGET_EXE = Join-Path $PSScriptRoot ".cake/nuget.exe"
$CAKE_EXE = Join-Path $TOOLS_DIR "Cake/Cake.exe"
$NUGET_URL = "https://dist.nuget.org/win-x86-commandline/latest/nuget.exe"
$NUGET_URL = "https://dist.nuget.org/win-x86-commandline/v4.7.3/nuget.exe"
$PACKAGES_CONFIG = Join-Path $TOOLS_DIR "packages.config"
$PACKAGES_CONFIG_MD5 = Join-Path $TOOLS_DIR "packages.config.md5sum"

Expand Down
2 changes: 1 addition & 1 deletion build.sh
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,7 @@ fi
# Download NuGet if it does not exist.
if [ ! -f "$NUGET_EXE" ]; then
echo "Downloading NuGet..."
curl -Lsfo "$NUGET_EXE" https://dist.nuget.org/win-x86-commandline/latest/nuget.exe
curl -Lsfo "$NUGET_EXE" https://dist.nuget.org/win-x86-commandline/v4.7.3/nuget.exe
if [ $? -ne 0 ]; then
echo "An error occured while downloading nuget.exe."
exit 1
Expand Down
23 changes: 8 additions & 15 deletions build/Tests.cake
Original file line number Diff line number Diff line change
Expand Up @@ -11,44 +11,39 @@ Task("Build-Managed")
.IsDependentOn("NuGet-Restore")
.Does(() =>
{
MSBuild("./tests/managed/generic/managed-generic.csproj", settings =>
settings.SetConfiguration(configuration));
MSBuild("./tests/managed/generic/managed-generic.csproj", MSBuildSettings());
});

Task("Build-Android")
.IsDependentOn("Clean")
.IsDependentOn("NuGet-Restore")
.Does(() =>
{
MSBuild("./tests/managed/android/managed-android.csproj", settings =>
settings.SetConfiguration(configuration).SetPlatformTarget(PlatformTarget.MSIL));
MSBuild("./tests/managed/android/managed-android.csproj", MSBuildSettings().SetPlatformTarget(PlatformTarget.MSIL));
});

Task("Build-FSharp-Android")
.IsDependentOn("Clean")
.IsDependentOn("NuGet-Restore")
.Does(() =>
{
MSBuild("./tests/managed/fsharp-android/fsharp-android.fsproj", settings =>
settings.SetConfiguration(configuration).SetPlatformTarget(PlatformTarget.MSIL));
MSBuild("./tests/managed/fsharp-android/fsharp-android.fsproj", MSBuildSettings().SetPlatformTarget(PlatformTarget.MSIL));
});

Task("Build-FSharp-Generic")
.IsDependentOn("Clean")
.IsDependentOn("NuGet-Restore")
.Does(()=>
{
MSBuild("./tests/managed/fsharp-generic/fsharp-generic.fsproj", settings =>
settings.SetConfiguration(configuration).SetPlatformTarget(PlatformTarget.MSIL));
MSBuild("./tests/managed/fsharp-generic/fsharp-generic.fsproj", MSBuildSettings().SetPlatformTarget(PlatformTarget.MSIL));
});

Task("Build-PCL")
.IsDependentOn("Clean")
.IsDependentOn("NuGet-Restore")
.Does(() =>
{
MSBuild("./tests/managed/pcl/managed-pcl.csproj", settings =>
settings.SetConfiguration(configuration).SetPlatformTarget(PlatformTarget.MSIL));
MSBuild("./tests/managed/pcl/managed-pcl.csproj", MSBuildSettings().SetPlatformTarget(PlatformTarget.MSIL));
});

Task("Build-NetStandard")
Expand All @@ -58,15 +53,15 @@ Task("Build-NetStandard")
{
var project = "./tests/managed/netstandard/managed-netstandard.csproj";
DotNetCoreRestore(project);
MSBuild(project, settings => settings.SetConfiguration(configuration).SetPlatformTarget(PlatformTarget.MSIL));
MSBuild(project, MSBuildSettings().SetPlatformTarget(PlatformTarget.MSIL));
});

Task("Build-CSharp-Tests")
.IsDependentOn("Build-Binder")
.IsDependentOn("Build-Managed")
.Does(() =>
{
MSBuild("./tests/MonoEmbeddinator4000.Tests/MonoEmbeddinator4000.Tests.csproj", settings => settings.SetConfiguration(configuration));
MSBuild("./tests/MonoEmbeddinator4000.Tests/MonoEmbeddinator4000.Tests.csproj", MSBuildSettings());
});

Task("Run-CSharp-Tests")
Expand Down Expand Up @@ -108,9 +103,7 @@ Task("Build-C-Tests")

// Execute the build files.
if (IsRunningOnWindows())
MSBuild(mkDir + File("mk.sln"), settings =>
settings.SetConfiguration(configuration)
.SetPlatformTarget(PlatformTarget.Win32));
MSBuild(mkDir + File("mk.sln"), MSBuildSettings().SetPlatformTarget(PlatformTarget.Win32));
else
{
var envVars = new Dictionary<string, string> ();
Expand Down
38 changes: 38 additions & 0 deletions build/Utils.cake
Original file line number Diff line number Diff line change
Expand Up @@ -59,3 +59,41 @@ void Premake(string file, string args, string action)
File("premake5.exe") : IsRunningOnMacOS() ? File("premake5-osx") : File("premake5-linux-64"));
Exec(premakePath, $"--file={file} {args} {action}");
}

MSBuildSettings MSBuildSettings()
{
var settings = new MSBuildSettings { Configuration = configuration };

if (IsRunningOnWindows())
{
// Find MSBuild for Visual Studio 2019 and newer
DirectoryPath vsLatest = VSWhereLatest();
FilePath msBuildPath = vsLatest?.CombineWithFilePath("./MSBuild/Current/Bin/MSBuild.exe");

// Find MSBuild for Visual Studio 2017
if (msBuildPath != null && !FileExists(msBuildPath))
msBuildPath = vsLatest.CombineWithFilePath("./MSBuild/15.0/Bin/MSBuild.exe");

// Have we found MSBuild yet?
if (!FileExists(msBuildPath))
{
throw new Exception($"Failed to find MSBuild: {msBuildPath}");
}

Information("Building using MSBuild at " + msBuildPath);
settings.ToolPath = msBuildPath;

var java_home = Environment.GetEnvironmentVariable("JAVA_HOME_8_X64");
if (!string.IsNullOrEmpty(java_home))
{
Information("JAVA_HOME_8_X64 set: " + java_home);
settings = settings.WithProperty("JavaSdkDirectory", java_home);
}
}
else
{
settings.ToolPath = Context.Tools.Resolve("msbuild");
}

return settings;
}
2 changes: 1 addition & 1 deletion external/Xamarin.Android.Tools
2 changes: 2 additions & 0 deletions support/mono-support.h
Original file line number Diff line number Diff line change
Expand Up @@ -89,7 +89,9 @@ typedef struct _GString GString;

/* utils/mono-publib.h */
typedef int32_t mono_bool;
#ifndef _WIN32
typedef uint16_t mono_unichar2;
#endif

/* metadata/image.h */
typedef struct _MonoAssembly MonoAssembly;
Expand Down

0 comments on commit 9cd5d13

Please sign in to comment.