Legend:
Page
Library
Module
Module type
Parameter
Class
Class type
Source
Page
Library
Module
Module type
Parameter
Class
Class type
Source
opamSysPoll.ml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216
(**************************************************************************) (* *) (* Copyright 2017-2019 OCamlPro *) (* *) (* All rights reserved. This file is distributed under the terms of the *) (* GNU Lesser General Public License version 2.1, with the special *) (* exception on linking described in the file LICENSE. *) (* *) (**************************************************************************) open OpamStd.Option.Op let command_output c = match List.filter (fun s -> String.trim s <> "") (OpamSystem.read_command_output c) with | [""] -> None | [s] -> Some s | _ -> None | exception (OpamSystem.Process_error _ | OpamSystem.Command_not_found _) -> None let norm s = if s = "" then None else Some (String.lowercase_ascii s) let normalise_arch raw = match String.lowercase_ascii raw with | "x86" | "i386" | "i486" | "i586" | "i686" -> "x86_32" | "x86_64" | "amd64" -> "x86_64" | "powerpc" | "ppc" | "ppcle" -> "ppc32" | "ppc64" | "ppc64le" -> "ppc64" | "aarch64_be" | "aarch64" -> "arm64" | a when a = "armv8b" || a = "armv8l" || List.exists (fun prefix -> OpamStd.String.starts_with ~prefix a) ["armv5"; "armv6"; "earmv6"; "armv7"; "earmv7"] -> "arm32" | s -> s let poll_arch () = let raw = match Sys.os_type with | "Unix" | "Cygwin" -> Some (OpamStd.Sys.uname ()).machine | "Win32" -> begin match OpamStubs.getArchitecture () with | OpamStubs.AMD64 -> Some "x86_64" | ARM -> Some "arm32" | ARM64 -> Some "arm64" | IA64 -> Some "ia64" | Intel -> Some "x86_32" | Unknown -> None end | _ -> None in let normalised = match raw with | None | Some "" -> None | Some a -> Some (normalise_arch a) in match Sys.os_type with | "Unix" | "Cygwin" -> (match normalised with | Some ("x86_64" | "arm64" | "ppc64" as arch) -> (match OpamStd.Sys.get_long_bit (), arch with | Some "32", "x86_64" -> Some "x86_32" | Some "32", "arm64" -> Some "arm32" | Some "32", "ppc64" -> Some "ppc32" | _ -> normalised | exception (Unix.Unix_error _ | Sys_error _ | Not_found) -> normalised) | _ -> normalised) | _ -> normalised let arch = Lazy.from_fun poll_arch let normalise_os raw = match String.lowercase_ascii raw with | "darwin" | "osx" -> "macos" | s -> s let poll_os () = let raw = match Sys.os_type with | "Unix" -> Some (OpamStd.Sys.uname ()).sysname | s -> norm s in match raw with | None | Some "" -> None | Some s -> Some (normalise_os s) let os = Lazy.from_fun poll_os let os_release_field = let os_release_file = lazy ( List.find Sys.file_exists ["/etc/os-release"; "/usr/lib/os-release"] |> OpamProcess.read_lines |> OpamStd.List.filter_map (fun s -> try Scanf.sscanf s "%s@= %s" (fun x v -> let contents = try Scanf.sscanf v "\"%s@\"" (fun s -> s) with Scanf.Scan_failure _ | End_of_file -> v in Some (x, contents)) with Scanf.Scan_failure _ | End_of_file -> None) ) in fun f -> try Some (OpamStd.List.assoc String.equal f (Lazy.force os_release_file)) with Not_found -> None let is_android, android_release = let prop = lazy (command_output ["getprop"; "ro.build.version.release"]) in (fun () -> Lazy.force prop <> None), (fun () -> Lazy.force prop) let poll_os_distribution () = let lazy os = os in match os with | Some "macos" as macos -> if OpamSystem.resolve_command "brew" <> None then Some "homebrew" else if OpamSystem.resolve_command "port" <> None then Some "macports" else macos | Some "linux" as linux -> (if is_android () then Some "android" else os_release_field "ID" >>= norm >>+ fun () -> command_output ["lsb_release"; "-i"; "-s"] >>= norm >>+ fun () -> let release_file = List.find_opt Sys.file_exists ["/etc/redhat-release"; "/etc/centos-release"; "/etc/gentoo-release"; "/etc/issue"] in match OpamStd.Option.map OpamProcess.read_lines release_file with | None | Some [] -> linux | Some (s::_) -> try Scanf.sscanf s " %s " norm with Scanf.Scan_failure _ -> linux) | Some "win32" -> let kind = OpamStd.Sys.get_windows_executable_variant ?search_in_first:(OpamCoreConfig.(!r.cygbin)) "cygpath.exe" in begin match kind with | `Msys2 -> Some "msys2" | `Cygwin -> Some "cygwin" | `Native | `Tainted _ -> os end | os -> os let os_distribution = Lazy.from_fun poll_os_distribution let poll_os_version () = let lazy os = os in match os with | Some "linux" -> android_release () >>= norm >>+ fun () -> command_output ["lsb_release"; "-s"; "-r"] >>= norm >>+ fun () -> os_release_field "VERSION_ID" >>= norm | Some "macos" -> command_output ["sw_vers"; "-productVersion"] >>= norm | Some "win32" -> let (major, minor, build, _) = OpamStubs.getWindowsVersion () in OpamStd.Option.some @@ Printf.sprintf "%d.%d.%d" major minor build | Some "cygwin" -> (try command_output ["cmd"; "/C"; "ver"] >>= fun s -> Scanf.sscanf s "%_s@[ Version %s@]" norm with Scanf.Scan_failure _ | End_of_file -> None) | Some "freebsd" -> OpamStd.Sys.get_freebsd_version () >>= norm | _ -> norm (OpamStd.Sys.uname ()).release let os_version = Lazy.from_fun poll_os_version let poll_os_family () = let lazy os = os in match os with | Some "linux" -> (os_release_field "ID_LIKE" >>= fun s -> Scanf.sscanf s " %s" norm (* first word *)) ++ Lazy.force os_distribution | Some ("freebsd" | "openbsd" | "netbsd" | "dragonfly") -> Some "bsd" | Some ("win32" | "cygwin") -> Some "windows" | _ -> Lazy.force os_distribution let os_family = Lazy.from_fun poll_os_family let variables = List.map (fun (n, v) -> OpamVariable.of_string n, OpamCompat.Lazy.map (OpamStd.Option.map (fun v -> OpamTypes.S v)) v) [ "arch", arch; "os", os; "os-distribution", os_distribution; "os-version", os_version; "os-family", os_family; ] let cores = let v = Lazy.from_fun OpamSystem.cpu_count in fun () -> Lazy.force v (* Exported functions *) let resolve_or_poll var poll env = match OpamVariable.Full.read_from_env (OpamVariable.Full.of_string var) with | Some (S c) -> Some c | _ -> match OpamVariable.Map.find_opt (OpamVariable.of_string var) env with | Some (lazy (Some (OpamTypes.S c)), _) -> Some c | _ -> Lazy.force poll let arch = resolve_or_poll "arch" arch let os = resolve_or_poll "os" os let os_distribution = resolve_or_poll "os-distribution" os_distribution let os_version = resolve_or_poll "os-version" os_version let os_family = resolve_or_poll "os-family" os_family let to_string env = let open OpamStd.Option.Op in Printf.sprintf "arch=%s os=%s os-distribution=%s os-version=%s" (arch env +! "unknown") (os env +! "unknown") (os_distribution env +! "unknown") (os_version env +! "unknown")