1 /** Universal Build System. 2 * 3 * TODO: Support `$0 REPO_URL`. Try for instance git@github.com:rui314/mold.git 4 * or https://github.com/rui314/mold.git. Supports matching repo with registered 5 * repos where repo name is stripped ending slash before matching. 6 * 7 * TODO: Delete installPrefix prior to installation 8 * 9 * TODO: Support `xy REPO_URL`. Try for instance git@github.com:rui314/mold.git 10 * or https://github.com/rui314/mold.git. Supports matching repo with registered 11 * repos where repo name is stripped ending slash before matching. 12 * 13 */ 14 module xdag; 15 16 // version = main; 17 18 debug import std.stdio : writeln; 19 import std.process : Config, spawnProcess, wait, Pid; 20 21 import nxt.path; 22 import nxt.file : homeDir; 23 import nxt.git : RepoURL; 24 25 private alias ProcessEnvironment = string[string]; 26 27 version (main) { 28 int main(string[] args) { 29 return buildN(args[1 .. $]); 30 } 31 } 32 33 /// Spec Name. 34 struct Name { string value; alias value this; } 35 36 /// D `version` symbol. 37 alias DlangVersionName = string; 38 39 struct RemoteSpec { 40 RepoURL url; 41 string remoteName; 42 string[] repoBranches; 43 } 44 45 void ewriteln(S...)(S args) @trusted { 46 import std.stdio : writeln, stderr; 47 return stderr.writeln(args); 48 } 49 50 /** Build the packages `packages`. 51 */ 52 int buildN(scope const string[] packages) { 53 const bool echo = true; // std.getopt: TODO: getopt flag being negation of make-like flag -s, --silent, --quiet 54 const BuildOption[] bos; 55 auto builds = makeSpecs(echo); 56 foreach (const name; packages) 57 if (Spec* buildPtr = Name(name) in builds) { 58 (*buildPtr).go(bos, echo); 59 } 60 else { 61 ewriteln("No spec named ", Name(name), " found"); 62 } 63 return 0; /+ TODO: propagate exit code +/ 64 } 65 66 const DlangVersionName[] versionNames = ["cli"]; 67 const string[] dflagsLDCRelease = ["-m64", "-O3", "-release", "-boundscheck=off", "-enable-inlining", "-flto=full"]; 68 69 /++ Build option. 70 +/ 71 enum BuildOption { 72 debugMode, /// Compile in debug mode (enables contracts, -debug) 73 releaseMode, /// Compile in release mode (disables assertions and bounds checks, -release) 74 ltoMode, /// Compile using LTO 75 pgoMode, /// Compile using PGO 76 coverage, /// Enable code coverage analysis (-cov) 77 debugInfo, /// Enable symbolic debug information (-g) 78 debugInfoC, /// Enable symbolic debug information in C compatible form (-gc) 79 alwaysStackFrame, /// Always generate a stack frame (-gs) 80 stackStomping, /// Perform stack stomping (-gx) 81 inline, /// Perform function inlining (-inline) 82 optimize, /// Enable optimizations (-O) 83 profile, /// Emit profiling code (-profile) 84 unittests, /// Compile unit tests (-unittest) 85 verbose, /// Verbose compiler output (-v) 86 ignoreUnknownPragmas, /// Ignores unknown pragmas during compilation (-ignore) 87 syntaxOnly, /// Don't generate object files (-o-) 88 warnings, /// Enable warnings (-wi) 89 warningsAsErrors, /// Treat warnings as errors (-w) 90 ignoreDeprecations, /// Do not warn about using deprecated features (-d) 91 deprecationWarnings, /// Warn about using deprecated features (-dw) 92 deprecationErrors, /// Stop compilation upon usage of deprecated features (-de) 93 property, /// DEPRECATED: Enforce property syntax (-property) 94 profileGC, /// Profile runtime allocations 95 pic, /// Generate position independent code 96 betterC, /// Compile in betterC mode (-betterC) 97 lowmem, /// Compile in lowmem mode (-lowmem) 98 } 99 100 alias BuildFlagsByOption = string[][BuildOption]; 101 102 Spec[Name] makeSpecs(bool echo) { 103 typeof(return) specs; 104 { 105 const name = "vox"; 106 Spec spec = 107 { homePage : URL("https://github.com/MrSmith33/"~name~"/"), 108 name : Name(name), 109 url : RepoURL("https://github.com/MrSmith33/"~name~".git"), 110 compiler : FilePath("ldc2"), 111 versionNames : ["cli"], 112 sourceFilePaths : [Path("main.d")], 113 outFilePath : Path(name~".out"), 114 }; 115 specs[spec.name] = spec; 116 } 117 { 118 const name = "dmd"; 119 Spec spec = 120 { homePage : URL("http://dlang.org/"), 121 name : Name(name), 122 url : RepoURL("https://github.com/dlang/"~name~".git"), 123 // nordlow/selective-unittest-2 124 // nordlow/check-self-assign 125 // nordlow/check-unused 126 extraRemoteSpecs : [RemoteSpec(RepoURL("https://github.com/nordlow/dmd.git"), "nordlow", 127 ["relax-alias-assign", 128 "traits-isarray-take2", 129 "diagnose-padding", 130 "aliasseq-bench", 131 "add-traits-hasAliasing", 132 "unique-qualifier"]), 133 // RemoteSpec(RepoURL("https://github.com/UplinkCoder/dmd.git"), "skoch", ["newCTFE_upstream"]), 134 ], 135 // patches : [Patch(Path("new-ctfe.patch"), 1, echo)], 136 buildFlagsDefault : ["PIC=1"], 137 buildFlagsByOption : [BuildOption.debugMode : ["ENABLE_DEBUG=1"], 138 BuildOption.releaseMode : ["ENABLE_RELASE=1"]], 139 }; 140 specs[spec.name] = spec; 141 } 142 { 143 const name = "phobos"; 144 Spec spec = 145 { homePage : RepoURL("http://dlang.org/"), 146 name : Name(name), 147 url : RepoURL("https://github.com/dlang/"~name~".git"), 148 extraRemoteSpecs : [RemoteSpec(RepoURL("https://github.com/nordlow/phobos.git"), "nordlow", ["faster-formatValue"])], 149 }; 150 specs[spec.name] = spec; 151 } 152 { 153 const name = "dub"; 154 Spec spec = 155 { homePage : RepoURL("http://dlang.org/"), 156 name : Name(name), 157 url : RepoURL("https://github.com/dlang/"~name~".git"), 158 }; 159 specs[spec.name] = spec; 160 } 161 { 162 const name = "DustMite"; 163 Spec spec = 164 { homePage : RepoURL("https://github.com/CyberShadow/"~name~"/wiki"), 165 name : Name(name), 166 compiler : FilePath("ldmd2"), 167 sources : ["dustmite.d", "polyhash.d", "splitter.d"], 168 url : RepoURL("https://github.com/CyberShadow/"~name~".git"), 169 }; 170 specs[spec.name] = spec; 171 } 172 { 173 const name = "DCD"; 174 Spec spec = 175 { name : Name(name), 176 url : RepoURL("https://github.com/dlang-community/"~name~".git"), 177 buildFlagsDefault : ["ldc"], 178 buildFlagsByOption : [BuildOption.debugMode : ["debug"], 179 BuildOption.releaseMode : ["ldc"]], 180 }; 181 specs[spec.name] = spec; 182 } 183 { 184 const name = "mold"; 185 Spec spec = 186 { name : Name(name), 187 url : RepoURL("https://github.com/rui314/"~name~".git"), 188 }; 189 specs[spec.name] = spec; 190 } 191 return specs; 192 } 193 194 /** Launch/Execution specification. 195 */ 196 @safe struct Spec { 197 import std.exception : enforce; 198 import std.algorithm : canFind, joiner; 199 import std.file : chdir, exists, mkdir; 200 import nxt.patching : Patch; 201 202 URL homePage; 203 Name name; 204 205 // Git sources 206 RepoURL url; 207 string gitRemote = "origin"; 208 string gitBranch; 209 RemoteSpec[] extraRemoteSpecs; 210 Patch[] patches; 211 212 FilePath compiler; 213 string[] sources; 214 string[] buildFlagsDefault; 215 BuildFlagsByOption buildFlagsByOption; 216 DlangVersionName[] versionNames; 217 Path[] sourceFilePaths; 218 Path outFilePath; 219 uint jobCount; 220 bool recurseSubModulesFlag = true; 221 222 void go(scope ref const BuildOption[] bos, in bool echo) scope @trusted { // TODO: -dip1000 without @trusted 223 const dlRoot = homeDir.buildPath(DirPath(".cache/repos")); 224 const dlDir = dlRoot.buildPath(DirName(name)); 225 writeln(); 226 /+ TODO: create build DAG and link these steps: +/ 227 fetch(bos, dlDir, echo); 228 prebuild(bos, dlDir, echo); 229 build(bos, dlDir, echo); 230 } 231 232 void fetch(scope ref const BuildOption[] bos, in DirPath dlDir, in bool echo) scope @trusted { // TODO: -dip1000 without @trusted 233 import nxt.git : RepoAndDir; 234 import core.thread : Thread; 235 auto rad = RepoAndDir(url, dlDir, echo); 236 /+ TODO: use waitAllInSequence(); +/ 237 enforce(!rad.cloneOrResetHard(recurseSubModulesFlag, gitRemote, gitBranch).wait()); 238 enforce(!rad.clean().wait()); 239 string[] remoteNames; 240 foreach (const spec; extraRemoteSpecs) 241 { 242 remoteNames ~= spec.remoteName; 243 const statusIgnored = rad.remoteRemove(spec.remoteName).wait(); // ok if already removed 244 enforce(!rad.remoteAdd(spec.url, spec.remoteName).wait()); 245 } 246 enforce(!rad.fetch(remoteNames).wait()); 247 foreach (const spec; extraRemoteSpecs) 248 foreach (const gitBranch; spec.repoBranches) 249 enforce(!rad.merge([spec.remoteName~"/"~gitBranch], ["--no-edit", "-Xignore-all-space"]).wait()); 250 foreach (ref patch; patches) 251 enforce(!patch.applyIn(dlDir).wait()); 252 } 253 254 string[] bosFlags(scope ref const BuildOption[] bos) const scope pure nothrow { 255 typeof(return) result; 256 foreach (bo; bos) 257 result ~= buildFlagsByOption[bo]; 258 return result; 259 } 260 261 string[] allBuildFlags(scope ref const BuildOption[] bos) const scope pure nothrow { 262 return buildFlagsDefault ~ bosFlags(bos) ~ jobFlags() ~ sources; 263 } 264 265 void prebuild(scope ref const BuildOption[] bos, in DirPath workDir, in bool echo) const scope { 266 if (exists("CMakeLists.txt")) { 267 return prebuildCmake(bos, workDir, echo); 268 } 269 } 270 271 void prebuildCmake(scope ref const BuildOption[] bos, in DirPath workDir, in bool echo) const scope @trusted { // TODO: -dip1000 without @trusted 272 Config config = Config.none; 273 string[] args; 274 ProcessEnvironment env_; 275 276 chdir(workDir.str); /+ TODO: replace with absolute paths in exists or dir.exists +/ 277 278 args = ["cmake"]; 279 const ninjaAvailable = true; /+ TODO: check if ninja is available +/ 280 if (ninjaAvailable) { 281 args ~= ["-G", "Ninja"]; 282 } 283 args ~= ["."]; /+ TODO: support out-of-source (OOS) build +/ 284 writeln("Pre-Building as `", args.joiner(" "), "` ..."); 285 scope Pid pid = spawnProcess(args, env_, config, workDir.str); 286 const status = pid.wait(); 287 if (status != 0) 288 writeln("Compilation failed:\n"); 289 if (status != 0) 290 writeln("Compilation successful:\n"); 291 } 292 293 void build(scope ref const BuildOption[] bos, in DirPath workDir, in bool echo) const scope @trusted { // TODO: -dip1000 without @trusted 294 Config config = Config.none; 295 296 chdir(workDir.str); /+ TODO: replace with absolute paths in exists or dir.exists +/ 297 298 /+ TODO: create build DAG and link these steps: +/ 299 300 string[] args; 301 ProcessEnvironment env_; 302 303 /+ TODO: when serveral of this exists ask user +/ 304 if (exists("Makefile")) 305 args = ["make", "-f", "Makefile"] ~ allBuildFlags(bos); 306 else if (exists("makefile")) 307 args = ["make", "-f", "makefile"] ~ allBuildFlags(bos); 308 else if (exists("posix.mak")) 309 args = ["make", "-f", "posix.mak"] ~ allBuildFlags(bos); 310 else if (exists("dub.sdl") || 311 exists("dub.json")) 312 args = ["dub", "build"]; 313 else 314 { 315 if (!compiler) { 316 if (bos.canFind(BuildOption.debugMode)) 317 args = [FilePath("dmd").str]; 318 else if (bos.canFind(BuildOption.releaseMode)) 319 args = [FilePath("ldc2").str]; 320 else 321 args = [FilePath("dmd").str]; // default to faster dmd 322 } else { 323 args = [compiler.str]; 324 } 325 if (bos.canFind(BuildOption.releaseMode) && 326 (compiler.str.canFind("ldc2") || 327 compiler.str.canFind("ldmd2"))) 328 args ~= dflagsLDCRelease; /+ TODO: make declarative +/ 329 foreach (const ver; versionNames) 330 args ~= ("-d-version=" ~ ver); 331 args ~= sources; 332 } 333 /* ${DC} -release -boundscheck=off dustmite.d splitter.d polyhash.d */ 334 /* mkdir -p ${DMD_EXEC_PREFIX} */ 335 /* mv dustmite ${DMD_EXEC_PREFIX} */ 336 337 enforce(args.length, "Could not deduce build command"); 338 339 /+ TODO: Use: import nxt.cmd : spawn; +/ 340 writeln("Building as `", args.joiner(" "), "` ..."); 341 /* TODO: add a wrapper spawnProcess1 called by Patch.applyIn, 342 * Repository.spawn and here that sets stdout, stderr based on bool 343 * echo. */ 344 scope Pid pid = spawnProcess(args, env_, config, workDir.str); 345 const status = pid.wait(); 346 if (status != 0) 347 writeln("Compilation failed:\n"); 348 if (status != 0) 349 writeln("Compilation successful:\n"); 350 } 351 352 string[] jobFlags() const scope pure nothrow { 353 import std.conv : to; 354 return jobCount ? ["-j"~jobCount.to!string] : []; 355 } 356 } 357 358 int[] waitAllInSequence(scope Pid[] pids...) { 359 typeof(return) statuses; 360 foreach (pid; pids) 361 statuses ~= pid.wait(); 362 return statuses; 363 } 364 365 int[] waitAllInParallel(scope Pid[] pids...) { 366 assert(0, "TODO: implement and use in place of waitAllInSequence"); 367 }