#region --- License --- /* Copyright (c) 2006, 2007 Stefanos Apostolopoulos * See license.txt for license info */ #endregion #region --- Using Directives --- using System; using System.Collections.Generic; using System.Text; using System.IO; using System.Diagnostics; #endregion namespace OpenTK.Build { class Project { static string RootPath; static string SourcePath; static string ToolPath = "Build"; static string PrebuildPath = Path.Combine(ToolPath, "Prebuild.exe"); static string BinPath = "Binaries"; static string ExePath = Path.Combine(BinPath, "Exe"); static string LibPath = Path.Combine(BinPath, "Libraries"); static string ExamplePath = Path.Combine(BinPath, "Examples"); static string PrebuildXml = Path.Combine(ToolPath, "Prebuild.xml"); enum BuildMode { Default = 0, Release = 0, Debug } enum BuildTarget { Default = 0, Net = 0, Mono, VS2005, SharpDevelop, MonoDevelop, Clean, DistClean, SVNClean } static BuildMode mode = BuildMode.Default; static BuildTarget target = BuildTarget.Default; static void PrintUsage() { Console.WriteLine("Usage: Build.exe BuildMode BuildTarget"); Console.WriteLine("\tBuildMode: debug/release"); Console.WriteLine("\tBuildTarget: mono/net/monodev/sharpdev/vs2005 or clean/distclean/svnclean"); } static void Main(string[] args) { RootPath = Directory.GetCurrentDirectory(); RootPath = RootPath.Substring( 0, Directory.GetCurrentDirectory().LastIndexOf("Build")); Directory.SetCurrentDirectory(RootPath); SourcePath = Path.Combine(RootPath, "Source"); // Workaroung for nant on x64 windows (safe for other platforms too, as this affects // only the current process). Environment.SetEnvironmentVariable("CommonProgramFiles(x86)", String.Empty, EnvironmentVariableTarget.Process); Environment.SetEnvironmentVariable("ProgramFiles(x86)", String.Empty, EnvironmentVariableTarget.Process); if (args.Length == 0) { PrintUsage(); } else { foreach (string s in args) { string arg = s.ToLower(); switch (arg) { case "debug": case "d": mode = BuildMode.Debug; break; case "release": case "r": mode = BuildMode.Release; break; case "mono": target = BuildTarget.Mono; break; case "net": target = BuildTarget.Net; break; case "monodev": case "monodevelop": case "md": target = BuildTarget.MonoDevelop; break; case "sharpdev": case "sharpdevelop": case "sd": target = BuildTarget.SharpDevelop; break; case "vs2005": case "vs": target = BuildTarget.VS2005; break; case "clean": target = BuildTarget.Clean; break; case "svnclean": target = BuildTarget.SVNClean; break; case "distclean": target = BuildTarget.DistClean; break; default: Console.WriteLine("Unknown command: {0}", s); PrintUsage(); return; } } ExePath = Path.Combine( BinPath, Path.Combine(mode == BuildMode.Debug ? "Debug" : "Release", "Exe")); LibPath = Path.Combine( BinPath, Path.Combine(mode == BuildMode.Debug ? "Debug" : "Release", "Libraries")); ExamplePath = Path.Combine( BinPath, Path.Combine(mode == BuildMode.Debug ? "Debug" : "Release", "Examples")); switch (target) { case BuildTarget.Mono: Console.WriteLine("Building OpenTK using Mono."); ExecuteProcess(PrebuildPath, "/target nant /file " + PrebuildXml); Console.WriteLine(); ExecuteProcess( "nant", "-buildfile:./Build/OpenTK.build -t:mono-2.0 " + (mode == BuildMode.Debug ? "build-debug" : "build-release")); CopyBinaries(); break; case BuildTarget.Net: Console.WriteLine("Building OpenTK using .Net"); ExecuteProcess(PrebuildPath, "/target nant /file " + PrebuildXml); Console.WriteLine(); ExecuteProcess( "nant", "-buildfile:./Build/OpenTK.build -t:net-2.0 " + (mode == BuildMode.Debug ? "build-debug" : "build-release")); CopyBinaries(); break; case BuildTarget.MonoDevelop: Console.WriteLine("Creating MonoDevelop project files"); ExecuteProcess(PrebuildPath, "/target monodev /file " + PrebuildXml); break; case BuildTarget.SharpDevelop: Console.WriteLine("Creating SharpDevelop project files"); ExecuteProcess(PrebuildPath, "/target monodev /file " + PrebuildXml); break; case BuildTarget.VS2005: Console.WriteLine("Creating VS2005 project files"); ExecuteProcess(PrebuildPath, "/target vs2005 /file " + PrebuildXml); break; case BuildTarget.Clean: Console.WriteLine("Cleaning intermediate object files."); ExecuteProcess(PrebuildPath, "/clean /yes /file " + PrebuildXml); DeleteDirectories(RootPath, "obj"); break; case BuildTarget.DistClean: Console.WriteLine("Cleaning intermediate and final object files."); ExecuteProcess(PrebuildPath, "/clean /yes /file " + PrebuildXml); DeleteDirectories(RootPath, "obj"); DeleteDirectories(RootPath, "bin"); if (Directory.Exists(RootPath + "Binaries")) Directory.Delete(RootPath + "Binaries", true); break; case BuildTarget.SVNClean: Console.WriteLine("Deleting svn directories."); DeleteDirectories(RootPath, ".svn"); break; default: Console.WriteLine("Unknown target: {0}", target); PrintUsage(); return; } //Console.WriteLine("Press any key to continue..."); //Console.ReadKey(true); } } static void DeleteDirectories(string root_path, string search) { Console.WriteLine("Deleting {0} directories", search); List matches = new List(); FindDirectories(root_path, search, matches); foreach (string m in matches) { Directory.Delete(m, true); } } static void CopyBinaries() { List example_matches = new List(); List exe_matches = new List(); List dll_matches = new List(); List dll_config_matches = new List(); Directory.CreateDirectory(BinPath); Directory.CreateDirectory(ExePath); Directory.CreateDirectory(LibPath); Directory.CreateDirectory(ExamplePath); // Move the libraries and the config files. FindFiles(SourcePath, "*.dll", dll_matches); foreach (string m in dll_matches) { File.Delete(Path.Combine(LibPath, Path.GetFileName(m))); File.Copy(m, Path.Combine(LibPath, Path.GetFileName(m))); File.Delete(Path.Combine(ExamplePath, Path.GetFileName(m))); File.Copy(m, Path.Combine(ExamplePath, Path.GetFileName(m))); } FindFiles(SourcePath, "*.dll.config", dll_config_matches); foreach (string m in dll_config_matches) { File.Delete(Path.Combine(LibPath, Path.GetFileName(m))); File.Copy(m, Path.Combine(LibPath, Path.GetFileName(m))); File.Delete(Path.Combine(ExamplePath, Path.GetFileName(m))); File.Copy(m, Path.Combine(ExamplePath, Path.GetFileName(m))); } // Then the examples. FindFiles(Path.Combine(SourcePath, "Examples"), "*.exe", example_matches); foreach (string m in example_matches) { File.Delete(Path.Combine(ExamplePath, Path.GetFileName(m))); File.Move(m, Path.Combine(ExamplePath, Path.GetFileName(m))); } // Then the rest of the exes. FindFiles(SourcePath, "*.exe", exe_matches); foreach (string m in exe_matches) { File.Delete(Path.Combine(ExePath, Path.GetFileName(m))); File.Move(m, Path.Combine(ExePath, Path.GetFileName(m))); } } static void FindDirectories(string directory, string search, List matches) { try { foreach (string d in Directory.GetDirectories(directory)) { foreach (string f in Directory.GetDirectories(d, search)) { matches.Add(f); } FindDirectories(d, search, matches); } } catch (System.Exception e) { Console.WriteLine(e.Message); } } static void FindFiles(string directory, string search, List matches) { try { foreach (string f in Directory.GetFiles(directory, search, SearchOption.AllDirectories)) { matches.Add(f); } //FindFiles(d, search, matches); } catch (System.Exception e) { Console.WriteLine(e.Message); } } static void ExecuteProcess(string path, string args) { using (Process p = new Process()) { ProcessStartInfo sinfo = new ProcessStartInfo(); if (Environment.OSVersion.Platform == PlatformID.Unix && !path.ToLower().Contains("nant")) { sinfo.FileName = "mono"; sinfo.Arguments = path + " " + args; } else { sinfo.FileName = path; sinfo.Arguments = args; } sinfo.WorkingDirectory = RootPath; sinfo.CreateNoWindow = true; sinfo.RedirectStandardOutput = true; sinfo.UseShellExecute = false; p.StartInfo = sinfo; p.OutputDataReceived += new DataReceivedEventHandler(p_OutputDataReceived); p.Start(); p.BeginOutputReadLine(); //StreamReader sr = p.StandardOutput; //while (!p.HasExited) //{ // Console.WriteLine(sr.ReadLine()); // Console.Out.Flush(); //} p.WaitForExit(); } } static void p_OutputDataReceived(object sender, DataReceivedEventArgs e) { if (!String.IsNullOrEmpty(e.Data)) { // Eat the last \n, we use WriteLine instead. This way we get the same result // in both windows and linux (linux would interpret both \n and WriteLine). Console.WriteLine(e.Data.TrimEnd('\n')); } } } }