This is meant to be a guide to people working on the standard library. It covers coding standards, code organization, best practices, internal annotations, and provides a guide to standard library internals. This document is inspired by LLVM's excellent programmer's manual and coding standards.
TODO: Should this subsume or link to StdlibRationales.rst?
TODO: Should this subsume or link to AccessControlInStdlib.rst
@available
, etc.@_inlineable
, @_versioned
, etc@inline(__always)
and @inline(never)
@semantics(...)
...
ranges work?Optionals can be unwrapped with !
, which triggers a trap on nil. Alternatively, they can be .unsafelyUnwrapped()
, which will check and trap in debug builds of user code. Internal to the standard library is ._unsafelyUnwrappedUnchecked()
which will only check and trap in debug builds of the standard library itself. These correspond directly with _precondition
, _debugPrecondition
, and _sanityCheck
. See that section for details.
_fastPath
and _slowPath
(also, _branchHint
)_fastPath
returns its argument, wrapped in a Builtin.expect. This informs the optimizer that the vast majority of the time, the branch will be taken (i.e. the then branch is “hot”). The SIL optimizer may alter heuristics for anything dominated by the then branch. But the real performance impact comes from the fact that the SIL optimizer will consider anything dominated by the else branch to be infrequently executed (i.e. “cold”). This means that transformations that may increase code size have very conservative heuristics to keep the rarely executed code small.
The probabilities are passed through to LLVM as branch weight metadata, to leverage LLVM’s use of GCC style builtin_expect knowledge (e.g. for code layout and low-level inlining).
_fastPath
probabilities are compounding, see the example below. For this reason, it can actually degrade performance in non-intuitive ways as it marks all other code (including subsequent _fastPath
s) as being cold. Consider _fastPath
as basically spraying the rest of the code with a Mr. Freeze-style ice gun.
_slowPath
is the same as _fastPath
, just with the branches swapped. Both are just wrappers around _branchHint
, which is otherwise never called directly.
Example:
if _fastPath(...) { // 90% of the time we execute this: aggressive inlining ... return } // 10% of the time we execute this: very conservative inlining ... if _fastPath(...) { // 9% of the time we execute this: very conservative inlining ... return } // 1% of the time we execute this: very conservative inlining ... return
NOTE: these are due for a rename and possibly a redesign. They conflate multiple notions that don’t match the average standard library programmer’s intuition.
_onFastPath
This should be rarely used. It informs the SIL optimizer that any code dominated by it should be treated as the innermost loop of a performance critical section of code. It cranks optimizer heuristics to 11. Injudicious use of this will degrade performance and bloat binary size.
_precondition
, _debugPrecondition
, and _sanityCheck
These three functions are assertions that will trigger a run time trap if violated.
_precondition
executes in all build configurations. Use this for invariant enforcement in all user code build configurations_debugPrecondition
will execute when user code is built with assertions enabled. Use this for invariant enforcement that's useful while debugging, but might be prohibitively expensive when user code is configured without assertions._sanityCheck
will execute when standard library code is built with assertions enabled. Use this for internal only invariant checks that useful for debugging the standard library itself._fixLifetime
A call to _fixLifetime
is considered a use of its argument, meaning that the argument is guaranteed to live at least up until the call. It is otherwise a nop. This is useful for guaranteeing the lifetime of a value while inspecting its physical layout. Without a call to _fixLifetime
, the last formal use may occur while the value's bits are still being munged.
Example:
var x = ... defer { _fixLifetime(x) } // Guarantee at least lexical lifetime for x let theBits = unsafeBitCast(&x, ...) ... // use of theBits in ways that may outlive x if it weren't for the _fixLifetime call
@_transparent
Should only be used if necessary. This has the effect of forcing inlining to occur before any dataflow analyses take place. Unless you specifically need this behavior, use @_inline(__always)
or some other mechanism. Its primary purpose is to force the compiler's static checks to peer into the body for diagnostic purposes.
Use of this attribute imposes limitations on what can be in the body. For more details, refer to the documentation.
@unsafe_no_objc_tagged_pointer
This is currently used in the standard library as an additional annotation applied to @objc protocols signifying that any objects which conform to this protocol are not tagged. This means that (on Darwin platforms) such references, unlike AnyObject, have spare bits available from things like restricted memory spaces or alignment.
@_silgen_name
This attribute specifies the name that a declaration will have at link time. It is used for two purposes, the second of which is currently considered bad practice and should be replaced with shims:
@_silgen_name
to call Swift from Swift-aware CRather than hard-code Swift mangling into C code, @_silgen_name
is used to provide a stable and known symbol name for linking. Note that C code still must understand and use the Swift calling convention (available in swift-clang) for such Swift functions (if they use Swift's CC). Example:
@_silgen_name("_swift_stdlib_destroyTLS") internal func _destroyTLS(_ ptr: UnsafeMutableRawPointer?) { // ... implementation ... }
__attribute__((__swiftcall__)) extern "C" void _swift_stdlib_destroyTLS(void *); // ... C code can now call _swift_stdlib_destroyTLS on a void * ...
@_silgen_name
to call C from SwiftThe standard library cannot import the Darwin module (much less an ICU module), yet it needs access to these C functions that it otherwise wouldn‘t have a decl for. For that, we use shims. But, @_silgen_name
can also be used on a body-less Swift function declaration to denote that it’s an external C function whose symbol will be available at link time, even if not available at compile time. This usage is discouraged.
_FixedArray
The standard library has internal fixed size arrays of some limited sizes. This provides fast random access into contiguous (usually stack-allocated) memory. These are metaprogrammed based on size, so if you need a new size not currently defined, add it to the sizes
gyb variable. See FixedArray.swift.gyb for implementation.
The standard library utilizes thread local storage (TLS) to cache expensive computations or operations in a thread-safe fashion. This is currently used for tracking some ICU state for Strings. Adding new things to this struct is a little more involved, as Swift lacks some of the features required for it to be expressed elegantly (e.g. move-only structs):
_ThreadLocalStorage
and a static getMyNewMember
method to access it. getMyNewMember
should be implemented using getPointer
._initializeThreadLocalStorage
and _ThreadLocalStorage.init
._destroyTLS
to properly destroy the value.See ThreadLocalStorage.swift for more details.