<<../_v2_banner.md>>
The ELF runner is the runner responsible for launching components based on standard executable files (ELF format). It is a built-in runner and is available to all components.
To use the ELF runner, the component's manifest must include a program
block,
containing:
{
program: {
runner: "elf",
binary: "bin/foo",
}
}
- The
runner
field with the value set to the stringelf
. - The
binary
field with the value set to a binary output name in the component's package.
Additionally, the ELF runner accepts a set of optional fields to configure the ELF component's runtime environment.
Arguments (that is, argv
in most programming languages) can be passed to a
component's process using the args
field. This field accepts a vector of
strings (see the example below).
The arguments set will be passed in the same order as declared in the manifest.
{
program: {
runner: "elf",
binary: "bin/foo",
args: ["--verbose", "--debug"]
}
}
The stdout and stderr streams of ELF components can be routed to the
LogSink service. By default, the ELF runner doesn't route these
streams to any output sink. Therefore, any write to these streams, such as
printf
, is lost and can be considered a no-op. If your component prints
diagnostics messages to either of these streams, you should forward the streams
to the LogSink service.
To enable this feature, add the following to your manifest file:
{
include: [ "syslog/elf_stdio.shard.cml" ],
}
After including this shard, all writes to stdout are logged as INFO
messages,
and all writes to stderr are logged as WARN
messages. Messages are split
by newlines and decoded as UTF-8 strings. Invalid byte sequences are converted
to the U+FFFD replacement character, which usually looks like �
.
Note: There are known issues where messages from ZX_ASSERT_...
in C/C++
components and Error
objects returned in main
in Rust components are lost.
To prevent issues where messages are lost from ZX_ASSERT_...
in C/C++
components and Error
objects returned from main
in Rust components it is
recommended that component authors include the aforementioned shard in their
manifest.
Components have a lifecycle. Components run by the ELF runner can
integrate with the lifecycle if you add a lifecycle
attribute to your component
manifest. Currently stop
is the only method in the Lifecycle protocol.
{
program: {
runner: "elf",
binary: "bin/foo",
lifecycle: { stop_event: "notify" },
}
}
The program should take the handle to the Lifecycle channel and serve the
Lifecycle protocol on that channel. The component should exit after
receiving and processing the stop
call.
The ELF Runner monitors the process it started for the program binary of the component. If this process exits, the ELF runner will terminate the component's execution context, which includes the component's job and all subprocesses.
Note: For a complete lifecycle example, see //examples/components/lifecycle.
There are several privileged fields that are gated by an
allowlist. Only components included in this allowlist
are able to use these fields. For all fields, the policy applies to the first
process in the component. The first process is the one created by ELF runner
for the binary declared in program
block. All of the fields are booleans
that default to false
.
The main_process_critical
field may be used to mark the component's first
process as critical to component manager's job, which will
cause component manager (and all components) to be terminated if the process
exits with a non-zero code. This will force the system to trigger a hard reboot.
The ambient_mark_vmo_exec
field may be used to allow the component's first
process to use zx_vmo_replace_as_executable
with a
ZX_HANDLE_INVALID
as the second argument rather than a valid
ZX_RSRC_KIND_VMEX
.
The create_raw_processes
field may be used to allow a component to create
processes by using zx_process_create
.
For a detailed explanation of how processes are created, please see Zircon program loading and dynamic linking.
Environment variables can be set for ELF components by using the environ
attribute. This field must be a vector of strings where each string contains
the variable and value delimited by an equal sign. For example, the following
sample code declares variables FAVORITE_ANIMAL
and FAVORITE_COLOR
to cat
and red
.
{
program: {
runner: "elf",
binary: "bin/echo",
environ: [
"FAVORITE_ANIMAL=cat",
"FAVORITE_COLOR=red",
]
}
}