| author | wenzelm | 
| Thu, 22 Dec 2016 11:20:59 +0100 | |
| changeset 64656 | 65c8a7780538 | 
| parent 64505 | 545a7ab3c35f | 
| child 64909 | 8007f10195af | 
| permissions | -rw-r--r-- | 
| 64483 | 1 | /* Title: Pure/Admin/build_polyml.scala | 
| 2 | Author: Makarius | |
| 3 | ||
| 4 | Build Poly/ML from sources. | |
| 5 | */ | |
| 6 | ||
| 7 | package isabelle | |
| 8 | ||
| 9 | ||
| 10 | object Build_PolyML | |
| 11 | {
 | |
| 64496 | 12 | /** build_polyml **/ | 
| 13 | ||
| 64483 | 14 | sealed case class Platform_Info( | 
| 15 | options: List[String] = Nil, | |
| 16 | options_multilib: List[String] = Nil, | |
| 64487 | 17 | setup: String = "", | 
| 64483 | 18 | copy_files: List[String] = Nil) | 
| 19 | ||
| 20 | private val platform_info = Map( | |
| 21 | "x86-linux" -> | |
| 22 | Platform_Info( | |
| 23 | options_multilib = | |
| 64484 | 24 |           List("--build=i386", "CFLAGS=-m32 -O3", "CXXFLAGS=-m32 -O3", "CCASFLAGS=-m32")),
 | 
| 64483 | 25 | "x86_64-linux" -> Platform_Info(), | 
| 26 | "x86-darwin" -> | |
| 27 | Platform_Info( | |
| 28 | options = | |
| 64484 | 29 |           List("--build=i686-darwin", "CFLAGS=-arch i686 -O3 -I../libffi/include",
 | 
| 30 | "CXXFLAGS=-arch i686 -O3 -I../libffi/include", "CCASFLAGS=-arch i686 -O3", | |
| 64503 
365021be3c5b
clarified setup: avoid alternative C compiler tools, e.g. from Homebrew or MacPorts;
 wenzelm parents: 
64502diff
changeset | 31 | "LDFLAGS=-segprot POLY rwx rwx"), | 
| 
365021be3c5b
clarified setup: avoid alternative C compiler tools, e.g. from Homebrew or MacPorts;
 wenzelm parents: 
64502diff
changeset | 32 | setup = "PATH=/usr/bin:/bin:/usr/sbin:/sbin"), | 
| 64483 | 33 | "x86_64-darwin" -> | 
| 34 | Platform_Info( | |
| 35 | options = | |
| 64484 | 36 |           List("--build=x86_64-darwin", "CFLAGS=-arch x86_64 -O3 -I../libffi/include",
 | 
| 37 | "CXXFLAGS=-arch x86_64 -O3 -I../libffi/include", "CCASFLAGS=-arch x86_64", | |
| 64503 
365021be3c5b
clarified setup: avoid alternative C compiler tools, e.g. from Homebrew or MacPorts;
 wenzelm parents: 
64502diff
changeset | 38 | "LDFLAGS=-segprot POLY rwx rwx"), | 
| 
365021be3c5b
clarified setup: avoid alternative C compiler tools, e.g. from Homebrew or MacPorts;
 wenzelm parents: 
64502diff
changeset | 39 | setup = "PATH=/usr/bin:/bin:/usr/sbin:/sbin"), | 
| 64483 | 40 | "x86-windows" -> | 
| 41 | Platform_Info( | |
| 42 | options = | |
| 64484 | 43 |           List("--host=i686-w32-mingw32", "CPPFLAGS=-I/mingw32/include", "--disable-windows-gui"),
 | 
| 64494 | 44 | setup = | 
| 45 | """PATH=/usr/bin:/bin:/mingw32/bin | |
| 46 | export CONFIG_SITE=/etc/config.site""", | |
| 64483 | 47 | copy_files = | 
| 64504 | 48 |           List("$MSYS/mingw32/bin/libgcc_s_dw2-1.dll",
 | 
| 49 | "$MSYS/mingw32/bin/libgmp-10.dll", | |
| 50 | "$MSYS/mingw32/bin/libstdc++-6.dll")), | |
| 64483 | 51 | "x86_64-windows" -> | 
| 52 | Platform_Info( | |
| 53 | options = | |
| 64484 | 54 |           List("--host=x86_64-w64-mingw32", "CPPFLAGS=-I/mingw64/include", "--disable-windows-gui"),
 | 
| 64494 | 55 | setup = | 
| 56 | """PATH=/usr/bin:/bin:/mingw64/bin | |
| 57 | export CONFIG_SITE=/etc/config.site""", | |
| 64483 | 58 | copy_files = | 
| 64504 | 59 |           List("$MSYS/mingw64/bin/libgcc_s_seh-1.dll",
 | 
| 60 | "$MSYS/mingw64/bin/libgmp-10.dll", | |
| 61 | "$MSYS/mingw64/bin/libstdc++-6.dll"))) | |
| 64483 | 62 | |
| 63 | def build_polyml( | |
| 64489 | 64 | root: Path, | 
| 64495 | 65 | sha1_root: Option[Path] = None, | 
| 64483 | 66 | progress: Progress = Ignore_Progress, | 
| 64493 | 67 | arch_64: Boolean = false, | 
| 64485 
e996c0a5eca9
support other bash executable (notably for msys on Windows);
 wenzelm parents: 
64484diff
changeset | 68 | options: List[String] = Nil, | 
| 64505 | 69 | msys_root: Option[Path] = None, | 
| 70 | component_root: Option[Path] = None) | |
| 64483 | 71 |   {
 | 
| 64489 | 72 |     if (!((root + Path.explode("configure")).is_file && (root + Path.explode("PolyML")).is_dir))
 | 
| 73 |       error("Bad Poly/ML root directory: " + root)
 | |
| 64483 | 74 | |
| 64493 | 75 | val platform = | 
| 76 | (if (arch_64) "x86_64" else "x86") + | |
| 77 | (if (Platform.is_windows) "-windows" else if (Platform.is_macos) "-darwin" else "-linux") | |
| 78 | ||
| 64483 | 79 | val info = | 
| 80 | platform_info.get(platform) getOrElse | |
| 81 |         error("Bad platform identifier: " + quote(platform))
 | |
| 82 | ||
| 64504 | 83 | val settings = | 
| 84 |       msys_root match {
 | |
| 85 | case None if Platform.is_windows => | |
| 86 |           error("Windows requires specification of msys root directory")
 | |
| 87 | case None => Isabelle_System.settings() | |
| 88 |         case Some(msys) => Isabelle_System.settings() + ("MSYS" -> msys.expand.implode)
 | |
| 89 | } | |
| 64492 | 90 | |
| 64491 | 91 | |
| 64495 | 92 | /* bash */ | 
| 93 | ||
| 64504 | 94 | def bash( | 
| 95 | cwd: Path, script: String, redirect: Boolean = false, echo: Boolean = false): Process_Result = | |
| 96 |     {
 | |
| 97 | val script1 = | |
| 98 |         msys_root match {
 | |
| 99 | case None => script | |
| 100 | case Some(msys) => | |
| 101 |             File.bash_path(msys + Path.explode("usr/bin/bash")) + " -c " + Bash.string(script)
 | |
| 102 | } | |
| 103 | progress.bash(script1, cwd = cwd.file, redirect = redirect, echo = echo) | |
| 104 | } | |
| 64495 | 105 | |
| 106 | ||
| 64505 | 107 | /* component setup */ | 
| 108 | ||
| 109 |     component_root match {
 | |
| 110 | case None => | |
| 111 | case Some(component) => | |
| 112 |         val etc = component + Path.explode("etc")
 | |
| 113 | Isabelle_System.mkdirs(etc) | |
| 114 |         File.copy(Path.explode("~~/Admin/polyml/settings"), etc)
 | |
| 115 |         File.copy(Path.explode("~~/Admin/polyml/README"), component)
 | |
| 116 | } | |
| 117 | ||
| 118 | ||
| 64491 | 119 | /* configure and make */ | 
| 120 | ||
| 64483 | 121 | val configure_options = | 
| 64493 | 122 |       (if (!arch_64 && Isabelle_System.getenv("ISABELLE_PLATFORM64") == "x86_64-linux")
 | 
| 123 | info.options_multilib | |
| 124 |        else info.options) ::: List("--enable-intinf-as-int") ::: options
 | |
| 64483 | 125 | |
| 64495 | 126 | bash(root, | 
| 64487 | 127 | info.setup + "\n" + | 
| 64485 
e996c0a5eca9
support other bash executable (notably for msys on Windows);
 wenzelm parents: 
64484diff
changeset | 128 | """ | 
| 
e996c0a5eca9
support other bash executable (notably for msys on Windows);
 wenzelm parents: 
64484diff
changeset | 129 | [ -f Makefile ] && make distclean | 
| 
e996c0a5eca9
support other bash executable (notably for msys on Windows);
 wenzelm parents: 
64484diff
changeset | 130 |         {
 | 
| 
e996c0a5eca9
support other bash executable (notably for msys on Windows);
 wenzelm parents: 
64484diff
changeset | 131 | ./configure --prefix="$PWD/target" """ + Bash.strings(configure_options) + """ | 
| 
e996c0a5eca9
support other bash executable (notably for msys on Windows);
 wenzelm parents: 
64484diff
changeset | 132 | rm -rf target | 
| 
e996c0a5eca9
support other bash executable (notably for msys on Windows);
 wenzelm parents: 
64484diff
changeset | 133 | make compiler && make compiler && make install | 
| 
e996c0a5eca9
support other bash executable (notably for msys on Windows);
 wenzelm parents: 
64484diff
changeset | 134 |         } || { echo "Build failed" >&2; exit 2; }
 | 
| 64501 | 135 | """, redirect = true, echo = true).check | 
| 64485 
e996c0a5eca9
support other bash executable (notably for msys on Windows);
 wenzelm parents: 
64484diff
changeset | 136 | |
| 64495 | 137 | val ldd_files = | 
| 64493 | 138 |       if (Platform.is_linux) {
 | 
| 64495 | 139 | val libs = bash(root, "ldd target/bin/poly").check.out_lines | 
| 64491 | 140 | val Pattern = """\s*libgmp.*=>\s*(\S+).*""".r | 
| 141 | for (Pattern(lib) <- libs) yield lib | |
| 142 | } | |
| 143 | else Nil | |
| 144 | ||
| 145 | ||
| 64495 | 146 | /* sha1 library */ | 
| 147 | ||
| 148 | val sha1_files = | |
| 149 |       if (sha1_root.isDefined) {
 | |
| 150 | val dir1 = sha1_root.get | |
| 64501 | 151 | bash(dir1, "./build " + platform, redirect = true, echo = true).check | 
| 64505 | 152 | |
| 153 | if (component_root.isDefined) | |
| 154 | Mercurial.repository(dir1). | |
| 155 |             archive(File.standard_path(component_root.get + Path.explode("sha1")))
 | |
| 156 | ||
| 64495 | 157 | val dir2 = dir1 + Path.explode(platform) | 
| 158 | File.read_dir(dir2).map(entry => dir2.implode + "/" + entry) | |
| 159 | } | |
| 160 | else Nil | |
| 161 | ||
| 162 | ||
| 64491 | 163 | /* target */ | 
| 64484 | 164 | |
| 64505 | 165 | val target = component_root.getOrElse(Path.current) + Path.explode(platform) | 
| 64488 | 166 | Isabelle_System.rm_tree(target) | 
| 64483 | 167 | Isabelle_System.mkdirs(target) | 
| 168 | ||
| 169 |     for {
 | |
| 64484 | 170 |       d <- List("target/bin", "target/lib")
 | 
| 64489 | 171 | dir = root + Path.explode(d) | 
| 64483 | 172 | entry <- File.read_dir(dir) | 
| 173 | } File.move(dir + Path.explode(entry), target) | |
| 174 | ||
| 64495 | 175 | for (file <- "~~/Admin/polyml/polyi" :: info.copy_files ::: ldd_files ::: sha1_files) | 
| 64504 | 176 | File.copy(Path.explode(file).expand_env(settings), target) | 
| 64483 | 177 | } | 
| 178 | ||
| 179 | ||
| 64496 | 180 | |
| 64500 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 181 | /** Isabelle tool wrapper **/ | 
| 64483 | 182 | |
| 64500 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 183 |   val isabelle_tool = Isabelle_Tool("build_polyml", "build Poly/ML from sources", args =>
 | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 184 |     {
 | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 185 |       Command_Line.tool0 {
 | 
| 64505 | 186 | var component_root: Option[Path] = None | 
| 64504 | 187 | var msys_root: Option[Path] = None | 
| 64500 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 188 | var arch_64 = false | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 189 | var sha1_root: Option[Path] = None | 
| 64483 | 190 | |
| 64500 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 191 |         val getopts = Getopts("""
 | 
| 64489 | 192 | Usage: isabelle build_polyml [OPTIONS] ROOT [CONFIGURE_OPTIONS] | 
| 64483 | 193 | |
| 194 | Options are: | |
| 64505 | 195 | -C DIR Isabelle component root directory (for etc/settings ...) | 
| 64504 | 196 | -M DIR msys root directory (for Windows) | 
| 64493 | 197 | -m ARCH processor architecture (32=x86, 64=x86_64, default: x86) | 
| 64499 | 198 | -s DIR sha1 sources, see https://bitbucket.org/isabelle_project/sha1 | 
| 64483 | 199 | |
| 64499 | 200 | Build Poly/ML in the ROOT directory of its sources, with additional | 
| 64489 | 201 | CONFIGURE_OPTIONS (e.g. --with-gmp). | 
| 64483 | 202 | """, | 
| 64505 | 203 | "C:" -> (arg => component_root = Some(Path.explode(arg))), | 
| 64504 | 204 | "M:" -> (arg => msys_root = Some(Path.explode(arg))), | 
| 64500 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 205 | "m:" -> | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 206 |             {
 | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 207 | case "32" | "x86" => arch_64 = false | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 208 | case "64" | "x86_64" => arch_64 = true | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 209 |               case bad => error("Bad processor architecture: " + quote(bad))
 | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 210 | }, | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 211 | "s:" -> (arg => sha1_root = Some(Path.explode(arg)))) | 
| 64483 | 212 | |
| 64500 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 213 | val more_args = getopts(args) | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 214 | val (root, options) = | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 215 |           more_args match {
 | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 216 | case root :: options => (Path.explode(root), options) | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 217 | case Nil => getopts.usage() | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 218 | } | 
| 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 219 | build_polyml(root, sha1_root = sha1_root, progress = new Console_Progress, | 
| 64505 | 220 | arch_64 = arch_64, options = options, component_root = component_root, | 
| 221 | msys_root = msys_root) | |
| 64500 
159ea1055b39
back to regular Isabelle tool (reverting abc34a149690);
 wenzelm parents: 
64499diff
changeset | 222 | } | 
| 64502 | 223 | }, admin = true) | 
| 64483 | 224 | } |