This repository has been archived by the owner on Jun 16, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
21 changed files
with
499 additions
and
376 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,4 @@ | ||
build/ | ||
/build/ | ||
|
||
.bsp/ | ||
target/ | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
111 changes: 111 additions & 0 deletions
111
src/main/scala/ee/hrzn/chryse/build/CommandRunner.scala
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,111 @@ | ||
/* Copyright © 2024 Asherah Connor. | ||
* | ||
* This file is part of Chryse. | ||
* | ||
* Chryse is free software: you can redistribute it and/or modify it under the | ||
* terms of the GNU Lesser General Public License as published by the Free | ||
* Software Foundation, either version 3 of the License, or (at your option) any | ||
* later version. | ||
* | ||
* Chryse 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 Lesser General Public License for more | ||
* details. | ||
* | ||
* You should have received a copy of the GNU Lesser General Public License | ||
* along with Chryse. If not, see <https://www.gnu.org/licenses/>. | ||
*/ | ||
|
||
package ee.hrzn.chryse.build | ||
|
||
import ee.hrzn.chryse.ChryseAppStepFailureException | ||
|
||
import java.io.File | ||
import scala.sys.process.Process | ||
import scala.util.matching.Regex | ||
|
||
object CommandRunner { | ||
sealed case class CmdStep(s: String) { | ||
override def toString() = s | ||
} | ||
object CmdStep { | ||
object Synthesise extends CmdStep("synthesise") | ||
object PNR extends CmdStep("place&route") | ||
object Pack extends CmdStep("pack") | ||
object Program extends CmdStep("program") | ||
object Compile extends CmdStep("compile") | ||
object Link extends CmdStep("link") | ||
object Execute extends CmdStep("execute") | ||
} | ||
|
||
private def paddedStep(step: CmdStep): String = { | ||
val r = s"($step)" | ||
val spaces = "(place&route) ".length() - r.length() | ||
r + " " * spaces | ||
} | ||
|
||
private val specialChar = "[^a-zA-Z0-9,./=+-_:@%^]".r | ||
|
||
// This isn't rigorous and it isn't meant to be — for displaying on stdout | ||
// only. | ||
private def formattedCmd(cmd: (Seq[String], Option[String])): String = { | ||
def fmtPart(part: String) = | ||
specialChar.replaceAllIn(part, Regex.quoteReplacement("\\") + "$0") | ||
cmd._2.map(dir => s"(in $dir/) ").getOrElse("") + | ||
cmd._1.map(fmtPart).mkString(" ") | ||
} | ||
|
||
sealed trait CmdAction | ||
object CmdAction { | ||
final case object Run extends CmdAction | ||
final case object Skip extends CmdAction | ||
} | ||
|
||
def reportCmd( | ||
step: CmdStep, | ||
action: CmdAction, | ||
cmd: (Seq[String], Option[String]), | ||
): Unit = { | ||
val paddedAction = action match { | ||
case CmdAction.Run => "[run] " | ||
case CmdAction.Skip => "[skip] " | ||
} | ||
println(s"${paddedStep(step)} $paddedAction ${formattedCmd(cmd)}") | ||
} | ||
|
||
def runCmd(step: CmdStep, cmd: Seq[String]) = | ||
runCmds(step, Seq((cmd, None))) | ||
|
||
def runCmds( | ||
step: CmdStep, | ||
cmds: Iterable[(Seq[String], Option[String])], | ||
): Unit = { | ||
cmds.foreach(reportCmd(step, CmdAction.Run, _)) | ||
val processes = cmds.map { cmd => | ||
val pb = Process(cmd._1, cmd._2.map(new File(_))) | ||
(cmd, pb.run()) | ||
} | ||
// TODO: consider an upper limit on concurrency. | ||
val failed = processes.collect { | ||
case (cmd, proc) if proc.exitValue() != 0 => cmd | ||
} | ||
if (!failed.isEmpty) { | ||
println("the following process(es) failed:") | ||
for { cmd <- failed } println(s" ${formattedCmd(cmd)}") | ||
throw new ChryseAppStepFailureException(step.toString()) | ||
} | ||
} | ||
|
||
def runCu(step: CmdStep, cu: CompilationUnit) = | ||
runCus(step, Seq(cu)) | ||
|
||
def runCus( | ||
step: CmdStep, | ||
cus: Iterable[CompilationUnit], | ||
): Unit = { | ||
val (skip, run) = cus.partition(_.isUpToDate()) | ||
skip.foreach(cu => reportCmd(step, CmdAction.Skip, (cu.cmd, cu.chdir))) | ||
runCmds(step, run.map(cu => (cu.cmd, cu.chdir))) | ||
run.foreach(_.markUpToDate()) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,71 @@ | ||
/* Copyright © 2024 Asherah Connor. | ||
* | ||
* This file is part of Chryse. | ||
* | ||
* Chryse is free software: you can redistribute it and/or modify it under the | ||
* terms of the GNU Lesser General Public License as published by the Free | ||
* Software Foundation, either version 3 of the License, or (at your option) any | ||
* later version. | ||
* | ||
* Chryse 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 Lesser General Public License for more | ||
* details. | ||
* | ||
* You should have received a copy of the GNU Lesser General Public License | ||
* along with Chryse. If not, see <https://www.gnu.org/licenses/>. | ||
*/ | ||
|
||
package ee.hrzn.chryse.build | ||
|
||
import java.nio.file.Files | ||
import java.nio.file.Paths | ||
import java.security.MessageDigest | ||
import java.util.HexFormat | ||
|
||
case class CompilationUnit( | ||
val primaryInPath: Option[String], | ||
val otherInPaths: Seq[String], | ||
val outPath: String, | ||
val cmd: Seq[String], | ||
val chdir: Option[String] = None, | ||
) { | ||
val digestPath = s"$outPath.dig" | ||
private val sortedInPaths = otherInPaths.appendedAll(primaryInPath).sorted | ||
|
||
private def addIntToDigest(n: Int)(implicit digest: MessageDigest): Unit = | ||
digest.update(String.format("%08x", n).getBytes("UTF-8")) | ||
|
||
private def addStringToDigest(s: String)(implicit | ||
digest: MessageDigest, | ||
): Unit = | ||
addBytesToDigest(s.getBytes("UTF-8")) | ||
|
||
private def addBytesToDigest( | ||
b: Array[Byte], | ||
)(implicit digest: MessageDigest): Unit = { | ||
addIntToDigest(b.length) | ||
digest.update(b) | ||
} | ||
|
||
private def digestInsWithCmd(): String = { | ||
implicit val digest = MessageDigest.getInstance("SHA-256") | ||
addIntToDigest(sortedInPaths.length) | ||
for { inPath <- sortedInPaths } { | ||
addStringToDigest(inPath) | ||
addBytesToDigest(Files.readAllBytes(Paths.get(inPath))) | ||
} | ||
addIntToDigest(cmd.length) | ||
for { el <- cmd } | ||
addStringToDigest(el) | ||
HexFormat.of().formatHex(digest.digest()) | ||
} | ||
|
||
def isUpToDate(): Boolean = | ||
Files.exists(Paths.get(digestPath)) && Files.readString( | ||
Paths.get(digestPath), | ||
) == digestInsWithCmd() | ||
|
||
def markUpToDate(): Unit = | ||
writePath(digestPath, digestInsWithCmd()) | ||
} |
33 changes: 33 additions & 0 deletions
33
src/main/scala/ee/hrzn/chryse/build/filesInDirWithExt.scala
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,33 @@ | ||
/* Copyright © 2024 Asherah Connor. | ||
* | ||
* This file is part of Chryse. | ||
* | ||
* Chryse is free software: you can redistribute it and/or modify it under the | ||
* terms of the GNU Lesser General Public License as published by the Free | ||
* Software Foundation, either version 3 of the License, or (at your option) any | ||
* later version. | ||
* | ||
* Chryse 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 Lesser General Public License for more | ||
* details. | ||
* | ||
* You should have received a copy of the GNU Lesser General Public License | ||
* along with Chryse. If not, see <https://www.gnu.org/licenses/>. | ||
*/ | ||
|
||
package ee.hrzn.chryse.build | ||
|
||
import java.nio.file.Files | ||
import java.nio.file.Path | ||
import scala.jdk.CollectionConverters._ | ||
|
||
object filesInDirWithExt { | ||
def apply(dir: String, ext: String): Iterator[String] = | ||
Files | ||
.walk(Path.of(dir), Integer.MAX_VALUE) | ||
.iterator | ||
.asScala | ||
.map(_.toString) | ||
.filter(_.endsWith(ext)) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,58 @@ | ||
/* Copyright © 2024 Asherah Connor. | ||
* | ||
* This file is part of Chryse. | ||
* | ||
* Chryse is free software: you can redistribute it and/or modify it under the | ||
* terms of the GNU Lesser General Public License as published by the Free | ||
* Software Foundation, either version 3 of the License, or (at your option) any | ||
* later version. | ||
* | ||
* Chryse 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 Lesser General Public License for more | ||
* details. | ||
* | ||
* You should have received a copy of the GNU Lesser General Public License | ||
* along with Chryse. If not, see <https://www.gnu.org/licenses/>. | ||
*/ | ||
|
||
package ee.hrzn.chryse.build | ||
|
||
import java.nio.file.Files | ||
import java.nio.file.Path | ||
import scala.jdk.CollectionConverters._ | ||
import scala.util.matching.Regex | ||
|
||
object logFileBetween { | ||
def apply( | ||
path: String, | ||
start: Regex, | ||
end: Regex, | ||
prefix: Option[String] = None, | ||
): Unit = { | ||
var started = false | ||
var ended = false | ||
val lines = Files.lines(Path.of(path)).iterator.asScala | ||
|
||
while (!ended && lines.hasNext) { | ||
val line = lines.next() | ||
if (!started) { | ||
started = start.matches(line) | ||
} else if (end.matches(line)) { | ||
ended = true | ||
} else { | ||
println(prefix match { | ||
case Some(prefix) => | ||
if ( | ||
line.length >= prefix.length && line | ||
.substring(0, prefix.length()) == prefix | ||
) | ||
line.substring(prefix.length()) | ||
else line | ||
case None => | ||
line | ||
}) | ||
} | ||
} | ||
} | ||
} |
Oops, something went wrong.