about summary refs log blame commit diff
path: root/third_party/bazel/rules_haskell/haskell/toolchain.bzl
blob: b05d51ea3ba65a4b89bebc5e6ff7d30a74903b37 (plain) (tree)













































































































































































































































































































































                                                                                                                                                                                            
"""Rules for defining toolchains"""

load("@bazel_skylib//lib:paths.bzl", "paths")
load(":ghc_bindist.bzl", "haskell_register_ghc_bindists")
load(
    ":private/actions/compile.bzl",
    "compile_binary",
    "compile_library",
)
load(
    ":private/actions/link.bzl",
    "link_binary",
    "link_library_dynamic",
    "link_library_static",
)
load(":private/actions/package.bzl", "package")

_GHC_BINARIES = ["ghc", "ghc-pkg", "hsc2hs", "haddock", "ghci", "runghc", "hpc"]

def _run_ghc(hs, cc, inputs, outputs, mnemonic, arguments, params_file = None, env = None, progress_message = None, input_manifests = None):
    if not env:
        env = hs.env

    args = hs.actions.args()
    args.add(hs.tools.ghc)

    # Do not use Bazel's CC toolchain on Windows, as it leads to linker and librarty compatibility issues.
    # XXX: We should also tether Bazel's CC toolchain to GHC's, so that we can properly mix Bazel-compiled
    # C libraries with Haskell targets.
    if not hs.toolchain.is_windows:
        args.add_all([
            # GHC uses C compiler for assemly, linking and preprocessing as well.
            "-pgma",
            cc.tools.cc,
            "-pgmc",
            cc.tools.cc,
            "-pgml",
            cc.tools.cc,
            "-pgmP",
            cc.tools.cc,
            # Setting -pgm* flags explicitly has the unfortunate side effect
            # of resetting any program flags in the GHC settings file. So we
            # restore them here. See
            # https://ghc.haskell.org/trac/ghc/ticket/7929.
            "-optc-fno-stack-protector",
            "-optP-E",
            "-optP-undef",
            "-optP-traditional",
        ])

    compile_flags_file = hs.actions.declare_file("compile_flags_%s_%s" % (hs.name, mnemonic))
    extra_args_file = hs.actions.declare_file("extra_args_%s_%s" % (hs.name, mnemonic))

    args.set_param_file_format("multiline")
    arguments.set_param_file_format("multiline")
    hs.actions.write(compile_flags_file, args)
    hs.actions.write(extra_args_file, arguments)

    extra_inputs = [
        hs.tools.ghc,
        # Depend on the version file of the Haskell toolchain,
        # to ensure the version comparison check is run first.
        hs.toolchain.version_file,
        compile_flags_file,
        extra_args_file,
    ] + cc.files

    if params_file:
        params_file_src = params_file.path
        extra_inputs.append(params_file)
    else:
        params_file_src = "<(:)"  # a temporary file with no contents

    script = """
export PATH=${PATH:-} # otherwise GCC fails on Windows

# this is equivalent to 'readarray'. We do not use 'readarray' in order to
# support older bash versions.
while IFS= read -r line; do compile_flags+=("$line"); done < %s
while IFS= read -r line; do extra_args+=("$line"); done < %s
while IFS= read -r line; do param_file_args+=("$line"); done < %s

"${compile_flags[@]}" "${extra_args[@]}" ${param_file_args+"${param_file_args[@]}"}
""" % (compile_flags_file.path, extra_args_file.path, params_file_src)

    ghc_wrapper_name = "ghc_wrapper_%s_%s" % (hs.name, mnemonic)
    ghc_wrapper = hs.actions.declare_file(ghc_wrapper_name)
    hs.actions.write(ghc_wrapper, script, is_executable = True)
    extra_inputs.append(ghc_wrapper)

    if type(inputs) == type(depset()):
        inputs = depset(extra_inputs, transitive = [inputs])
    else:
        inputs += extra_inputs

    hs.actions.run_shell(
        inputs = inputs,
        input_manifests = input_manifests,
        outputs = outputs,
        command = ghc_wrapper.path,
        mnemonic = mnemonic,
        progress_message = progress_message,
        env = env,
        arguments = [],
    )

    return args

def _haskell_toolchain_impl(ctx):
    # Store the binaries of interest in ghc_binaries.
    ghc_binaries = {}
    for tool in _GHC_BINARIES:
        for file in ctx.files.tools:
            if tool in ghc_binaries:
                continue

            basename_no_ext = paths.split_extension(file.basename)[0]
            if tool == basename_no_ext:
                ghc_binaries[tool] = file
            elif "%s-%s" % (tool, ctx.attr.version) == basename_no_ext:
                ghc_binaries[tool] = file
        if not tool in ghc_binaries:
            fail("Cannot find {} in {}".format(tool, ctx.attr.tools.label))

    # Run a version check on the compiler.
    version_file = ctx.actions.declare_file("ghc-version")
    ghc = ghc_binaries["ghc"]
    ctx.actions.run_shell(
        inputs = [ghc],
        outputs = [version_file],
        mnemonic = "HaskellVersionCheck",
        command = """
{ghc} --numeric-version > {version_file}
if [[ "{expected_version}" != "$(< {version_file})" ]]
then
    echo ERROR: GHC version does not match expected version.
    echo Your haskell_toolchain specifies {expected_version},
    echo but you have $(< {version_file}) in your environment.
exit 1
fi
        """.format(
            ghc = ghc.path,
            version_file = version_file.path,
            expected_version = ctx.attr.version,
        ),
    )

    # Get the versions of every prebuilt package.
    ghc_pkg = ghc_binaries["ghc-pkg"]
    pkgdb_file = ctx.actions.declare_file("ghc-global-pkgdb")
    ctx.actions.run_shell(
        inputs = [ghc_pkg],
        outputs = [pkgdb_file],
        mnemonic = "HaskellPackageDatabaseDump",
        command = "{ghc_pkg} dump --global > {output}".format(
            ghc_pkg = ghc_pkg.path,
            output = pkgdb_file.path,
        ),
    )

    tools_struct_args = {
        name.replace("-", "_"): file
        for name, file in ghc_binaries.items()
    }

    locale_archive = None

    if ctx.attr.locale_archive != None:
        locale_archive = ctx.file.locale_archive

    return [
        platform_common.ToolchainInfo(
            name = ctx.label.name,
            tools = struct(**tools_struct_args),
            compiler_flags = ctx.attr.compiler_flags,
            repl_ghci_args = ctx.attr.repl_ghci_args,
            haddock_flags = ctx.attr.haddock_flags,
            locale = ctx.attr.locale,
            locale_archive = locale_archive,
            osx_cc_wrapper_tpl = ctx.file._osx_cc_wrapper_tpl,
            mode = ctx.var["COMPILATION_MODE"],
            actions = struct(
                compile_binary = compile_binary,
                compile_library = compile_library,
                link_binary = link_binary,
                link_library_dynamic = link_library_dynamic,
                link_library_static = link_library_static,
                package = package,
                run_ghc = _run_ghc,
            ),
            is_darwin = ctx.attr.is_darwin,
            is_windows = ctx.attr.is_windows,
            version = ctx.attr.version,
            # Pass through the version_file, that it can be required as
            # input in _run_ghc, to make every call to GHC depend on a
            # successful version check.
            version_file = version_file,
            global_pkg_db = pkgdb_file,
        ),
    ]

_haskell_toolchain = rule(
    _haskell_toolchain_impl,
    attrs = {
        "tools": attr.label_list(
            doc = "GHC and executables that come with it. First item take precedance.",
            mandatory = True,
        ),
        "compiler_flags": attr.string_list(
            doc = "A collection of flags that will be passed to GHC on every invocation.",
        ),
        "repl_ghci_args": attr.string_list(
            doc = "A collection of flags that will be passed to GHCI on repl invocation. It extends the `compiler_flags` collection. Flags set here have precedance over `compiler_flags`.",
        ),
        "haddock_flags": attr.string_list(
            doc = "A collection of flags that will be passed to haddock.",
        ),
        "version": attr.string(
            doc = "Version of your GHC compiler. It has to match the version reported by the GHC used by bazel.",
            mandatory = True,
        ),
        "is_darwin": attr.bool(
            doc = "Whether compile on and for Darwin (macOS).",
            mandatory = True,
        ),
        "is_windows": attr.bool(
            doc = "Whether compile on and for Windows.",
            mandatory = True,
        ),
        "locale": attr.string(
            default = "en_US.UTF-8",
            doc = "Locale that will be set during compiler invocations.",
        ),
        "locale_archive": attr.label(
            allow_single_file = True,
            doc = """
Label pointing to the locale archive file to use. Mostly useful on NixOS.
""",
        ),
        "_osx_cc_wrapper_tpl": attr.label(
            allow_single_file = True,
            default = Label("@io_tweag_rules_haskell//haskell:private/osx_cc_wrapper.sh.tpl"),
        ),
    },
)

def haskell_toolchain(
        name,
        version,
        tools,
        exec_compatible_with = None,
        target_compatible_with = None,
        compiler_flags = [],
        repl_ghci_args = [],
        haddock_flags = [],
        locale_archive = None,
        **kwargs):
    """Declare a compiler toolchain.

    You need at least one of these declared somewhere in your `BUILD` files
    for the other rules to work. Once declared, you then need to *register*
    the toolchain using `register_toolchains` in your `WORKSPACE` file (see
    example below).

    Example:

      In a `BUILD` file:

      ```bzl
      haskell_toolchain(
          name = "ghc",
          version = "1.2.3",
          tools = ["@sys_ghc//:bin"],
          compiler_flags = ["-Wall"],
      )
      ```

      where `@sys_ghc` is an external repository defined in the `WORKSPACE`,
      e.g. using:

      ```bzl
      nixpkgs_package(
          name = 'sys_ghc',
          attribute_path = 'haskell.compiler.ghc822',
      )

      register_toolchains("//:ghc")
      ```
    """
    if exec_compatible_with and not target_compatible_with:
        target_compatible_with = exec_compatible_with
    elif target_compatible_with and not exec_compatible_with:
        exec_compatible_with = target_compatible_with
    impl_name = name + "-impl"
    corrected_ghci_args = repl_ghci_args + ["-no-user-package-db"]
    _haskell_toolchain(
        name = impl_name,
        version = version,
        tools = tools,
        compiler_flags = compiler_flags,
        repl_ghci_args = corrected_ghci_args,
        haddock_flags = haddock_flags,
        visibility = ["//visibility:public"],
        is_darwin = select({
            "@io_tweag_rules_haskell//haskell/platforms:darwin": True,
            "//conditions:default": False,
        }),
        is_windows = select({
            "@io_tweag_rules_haskell//haskell/platforms:mingw32": True,
            "//conditions:default": False,
        }),
        # Ignore this attribute on any platform that is not Linux. The
        # LOCALE_ARCHIVE environment variable is a Linux-specific
        # Nixpkgs hack.
        locale_archive = select({
            "@io_tweag_rules_haskell//haskell/platforms:linux": locale_archive,
            "//conditions:default": None,
        }),
        **kwargs
    )
    native.toolchain(
        name = name,
        toolchain_type = "@io_tweag_rules_haskell//haskell:toolchain",
        toolchain = ":" + impl_name,
        exec_compatible_with = exec_compatible_with,
        target_compatible_with = target_compatible_with,
    )

def haskell_register_toolchains(version):
    """Download the binary distribution of GHC for your current platform
    and register it as a toolchain. This currently has the same effect
    as just `haskell_register_ghc_bindists(version)`.
    """
    haskell_register_ghc_bindists(version)