This is very hand-wavy and very much targeted at brainstorming. At this stage, what I suspect will be most useful will be
- Describing where
features
are not working well or where are you using--cfg
- Planning for how it would look with this proposal instead
- Evaluating how well this proposal meets your needs. What works well and what doesn't?
I do not expect to have the time to see this through the RFC process and implementation. However, I will be willing to help shepherd this effort if someone is willing to write the RFC and drive this to implementation.
- Feature Name:
global-features
- Start Date: 2023-09-20
- RFC PR: rust-lang/rfcs#0000
- Rust Issue: rust-lang/rust#0000
- Edit history
Summary
Cargo features
are targeted at direct dependents while some decisions need to be made by the top-level crate (usually a [[bin]]
).
This is currently worked around by tunneling features up or relying on environment variables.
This RFC proposes an alternative to features called globals
that are targeted
at decisions that affect the entire final artifact.
[package]
name = "git-tool"
version = "0.0.0"
set-globals = { sys = ["static"] }
[dependencies]
git2 = "0.18"
[package]
name = "git2"
version = "0.18.5"
[dependencies]
git2-sys = "0.18.5"
[package]
name = "git2-sys"
version = "0.18.5"
[globals.sys]
description = "External library link method"
values = ["static", "dynamic", "auto"]
default = "auto"
Differences from features
- Setting them applies to transitive dependencies
- Enumerations, rather than "present" / "not-present"
- No unification
We suspect this will work best for modifying behavior while features will work best for extending APIs.
Motivation
Use cases
- A crate author offers optional optimizations, like using
parking_lot
- Currently solved by using a feature like in tokio
- This requires callers to either enable it directly, re-export the feature, or have applications directly depend on
tokio
and enable it.
-sys
crates need to make the decision of whether to statically link a vendored version of the source of dynamically link against a system library.- Currently this is solved by a variety of means, usually by dynamically linking if the system library is available and then falling back to the vendored copy unless a vendored feature is enabled.
- This requires callers to either enable it directly, re-export the feature, or have applications directly depend on the
-sys
crate and enable it. - To properly represent this, we need three states: "dynamic", "static", "auto"
- See also Internals: Pre-RFC Cargo features for configuring sys crates
- Enable
alloc
orstd
features across the stack (rust-lang/cargo#2593) - Multiple backends
- flate2 has multiple backends
- Inkwell supports multiple LLVM backends
- mlua supports multiple backends and Lua versions
- async-std vs tokio runtimes
- Module-level parameters
- Inline-size control in kstring
Why are we doing this? What use cases does it support? What is the expected outcome?
See also
Guide-level explanation
TODO
Reference-level explanation
Packages may declare what globals
they subscribe to:
[package]
name = "git2-sys"
version = "0.0.0"
[globals.sys]
description = "External library link method"
values = ["static", "dynamic", "auto"]
default = "auto"
[target.'cfg(any(global::sys = "dynamic", global::sys = "auto"))'.build-dependencies]
pkg_config = "0.3.27"
[target.'cfg(any(global::sys = "static", global::sys = "auto"))'.build-dependencies]
cc = "1.0.83"
- Global names follow the same naming rules as cargo features
- Global values may be any UnicodeXID continue character
- TODO: Add deprecation support
These can then be referenced in the global
namespace in the source, like this build.rs
#[cfg(any(global::sys = "dynamic", global::sys = "auto"))]
fn dynamic() -> Option<Target> {
// ...
}
#[cfg(any(global::sys = "static", global::sys = "auto"))]
fn static() -> Option<Target> {
// ...
}
fn auto() -> Option<Target> {
#[cfg(any(global::sys = "dynamic", global::sys = "auto"))]
if let Some(target) = dynamic() {
return Some(target);
}
#[cfg(any(global::sys = "static", global::sys = "auto"))]
if let Some(target) = static() {
return Some(target);
}
None
}
- References by this crate to global names/values in
cfg
s andcargo:rustc-cfg
build script directive will be validated based on RFC 3013 (rustc and cargo)
Workspace globals may also be specified and a package may inherit them:
[workspace.globals.sys]
description = "External library link method"
values.enum = ["static", "dynamic", "auto"]
default = "auto"
[globals.sys]
workspace = true
[globals.zlib]
description = "zlib implementation"
values = ["miniz", "zlib", "zlib-ng"]
default = "auto"
- This initial design has no overridable fields when inheriting, unlike inheriting workspace dependencies
- Like workspace dependencies,
cargo new
will not automatically inheritworkspace.globals
Packages may configure globals for use when built as a root crate
[package]
set-globals = { sys = ["static"], zlib = ["miniz"] }
# may also be inherited from the workspace
# set-globals.workspace = true
- Globals are arrays in case two packages are subscribed to that global with disjoint valid values
- When building multiple root crates (
cargo check --workspace
), it is a compilation error if they disagree aboutset-globals
(i.e. no unification is happening like it does forfeatures
)
Also these may be set:
- On the command-line as
--globals sys=dynamic
- As a config setting
When resolving dependencies (the feature pass), fingerprinting of packages, an
when compiling the packages,
workspace.set-globals
will only apply to a package if the value is valid
within the schema for that package,
otherwise the default will be used if any.
When multiple values are valid, the first will be selected.
Any unused set-globals
will be a warning.
cargo <cmd> --globals
(no argument) will report the globals available for
being set within the current package along with their current value and valid values.
SemVer
SemVer Compatible
- Adding a global
- Adding a global value
Context-dependent for whether compatible or not
- Changing a global default
- Removing a global
- Removing a global value
Drawbacks
TODO
Rationale and alternatives
This couples mutually exclusive features with global features because
- Mutually exclusive features provides design insight into what we should do for global features
- Scoping mutually exclusive features to global features solves the feature-unification problem
Global features take a value, rather than allowing relying on presence like normal cfg
s
(see RFC 3013)
- Less brittle for future evolution
- Easier to unify for a distributed schema
- Removes the complexity in declaratively defining presence and value instances
package.set-globals
is a package setting, with conflict errors, rather than a
workspace setting because people will likely have binaries that serve very
different roles within the same workspace (different target platforms, host vs wasm, etc).
The loose schema is used to allow it to be declared in a distributed fashion without packages failing due to definition conflicts. Alternatively, we could make these parameters on individual packages. The downside is then we'd need a patch-like table for specifying the exact dependency in the tree to set the parameters. This scheme is brittle in handling upgrades.
Unused set-globals
is not an error so that changing dependencies is not a
breaking change.
Alternatives
Native support for controlling std
/ alloc
/ core
See Internals: Pre-Pre-RFC: making std
-dependent Cargo features a first-class concept
Native cargo support for -sys
crates
Instead of a convention around global features, cargo could have built-in flags
for controlling -sys
crates and could ship some native support that would
remove some boilerplate and ensure consistency.
Downsides
- Longer time frame: this would require more investigation and experimentation.
- Doesn't handle all use cases
See
Automatic feature activation
Too many spooky effects
See
Mutually exclusive features conflict via an identifier
Some systems call this "provides" and others a "capability".
Downsides
- No way to unify these, the decision needs to be at the top-level crate.
- Linux distributions that use this implicitly wire the packages together by dropping files in place with compatible names while Rust/Cargo need explicit wiring, requiring the choice to be bubbled up the stack. See instead "facade crates".
See
Mutually exclusive features that instantiate distinct copies a dependency
While there might be a place for a variant of this idea, most motivating cases are dealing with wanting one version of the dependency.
See also Internals: module-level generics
See also Cabal backpack support
log
/ logger split
Some cases can be modeled like log
where a trait is defined and an API is
exposed for regisering a specific implementation.
If the interface crate is able to also depend on the implementations, it could also initialize the global with a default on first use if its unset.
Benefits
- Works today
- Flexible on what state an implementation can be initialized with
Downsides
- Doesn't work for all use cases
- Imperative wiring together of APIs in the top-level crate
- Some level of overhead
- Supporting a default requires building all implementations since the choice is at runtime
Facade crates
On internals, the idea was proposed to bake-in support for the log pattern so it can be done at compile time.
Downsides
- Doesn't work for all use cases like
-sys
crates
Prior art
Gentoo USE flags
USE flags are booleans (set or unset) that are defined through a form of layered config, with some defaulted to on. Users may then enable more or disable some of the defaulted flags. This can be done globally or on a per-package basis. Packages then check what is enabled and conditionalize their builds off of them.
Bazel
Features may "provide" a feature and it is an error to "provide" a feature more than once.
Gradle
This seems work similar to Bazel but instead of "providing" a feature, you declare a capability.
See also Handling mutually exclusive dependencies
Unresolved questions
- Target-specific
set-globals
- Profile-specific
set-globals
(for custom performance / debugging experience, see Conditional compilation: Cfg option for debug vs. release Cargo(-esque) profile · Issue #14974 · rust-lang/cargo · GitHub) - Naming, especially for
package.set-globals
anddefault
- Whether
values.enum
is needed vs justvalues
- Can we make
RFC 3013
work with the root
cfg
namespace being an open set of fields whileglobal::
namespace is a closed set?
Future possibilities
Loosen error on conflicting globals
Most of the time, libs may be selected as part of the root crates (cargo check --workspace
) but effectively aren't a root crate.
Can we loosen the requirement on these so they don't need the same
set-globals
as others in the workspace, making it easier to build?
cfg_str!
Expose value set for the current package as a &'static str
let foo = cfg_str!("global::foo");
- Error for
cfg
s that have a name without a value
Additional globals.*.values
validation rules
Currently, we only support validation based on a set of possible values.
Potential fields for the globals.*.values
include:
globals.foo.values.bool = true
globals.foo.values.range = "8-23"
globals.foo.values.path = "ImplForTrait"
globals.foo.values.expr = "some_func()"
globals.foo.values.any = true
(makes values an open set, predantic warning available when it falls back to this)
(not a tagged variant to make future evolution easier)
Multi-valued globals
Extend the definition of a global to include a multi
field.
Instead of new assignments overwriting, they append.