Helium is a library of base abstractions to reduce the amount of required code
needed to successfully implement the ANARI API. While using helium is not
required to implement ANARI, it does provide base functionality that is very
common for devices to implement. Specifically, libhelium
provides the
following functionality:
- Parameter management (
anariSetParameter
+anariUnsetParameter
) - Object commit management
- Deferred object commit buffering
- Top-down commit reordering
- Object lifetime management
- Interface base classes to handle array and frame specfic API calls
- Array change tracking
- Status callback messaging infrastructure
The tradeoff implementors make is that by choosing to use helium, you necessarily buy into implementation choices made. Thus implementations which want to tightly control the above set of features (or simply make different choices) ought to instead directly implement the ANARI API interface.
The helide device is a minimal implementation which combines helium and Embree to do very basic CPU rendering with ray tracing. Potential helium library users can use helide as a guide for how helium abstractions are intended to be used.
The name helium is a pun on the gallium library found in the Mesa3D graphics library -- both libraries are aimed at reducing the amount of implementation required to implement their respective APIs, but doing so for ANARI is a much smaller problem and therefore requires a much smaller library than gallium. Thus an element lighter than gallium was chosen as the name...helium!
Helium is a small static library shipped by the ANARI SDK, which can be linked as a CMake target. This may look like:
find_package(anari)
add_library(anari_device_myDevice SHARED)
target_link_libraries(anari_device_myDevice PRIVATE anari::helium)
# ...
The following sections will outline various design choices relevent to implementors using helium.
The concept of a TimeStamp is just a unique integer for
every time helium::newTimeStamp()
is called -- this makes it easy to track
when two events occured relative to each other in time, but the precise clock
time does not matter. This concept/abstraction is primarily used to track if
object commits need to occur or can be skipped.
The helium::BaseDevice class implements a subset of the main API
class that the C API forward to
(anari::DeviceImpl).
Implementations are mostly responsible for implementing all the new*()
methods
of anari::DeviceImpl
as helium doesn't know how to create concrete objects.
Some implementations may desire to still "intercept" all base implemented API
calls, which can easily be done by overriding the method found in
helium::BaseDevice
and then call it when appropriately. This may look like:
void MyDevice::setParameter(
ANARIObject object, const char *name, ANARIDataType type, const void *mem)
{
// Insert 'MyDevice' specific code (ex: setting an active GPU device)
//
// Then call base helium library
helium::BaseDevice::setParameter(object, name, type, mem);
}
NOTE: For helium::BaseDevice
to function correctly, all objects passed through
the API must derive from BaseObject
, BaseArray
, and BaseFrame
respectively!
Helium based devices use helium::BaseObject to generically
represent API calls involving ANARIObject
. This includes setting/unsetting
parameters, property queries, and committing parameters. It also includes
information for comparing when parameter changes have occured to when the object
has committed those parameters.
All parameter handling is done through
helium::ParameterizedObject. Helium uses a
'pull' based model for handling parameters -- all parameter values are
generically stored in the object and are expected to be read on
helium::BaseObject::commit()
. See comments on ParameterizedObject
methods
for a further explanation.
Object commits are deferred until the device chooses to flush the DefferedCommitBuffer that lives in the global device state instance on the device itself. It is entirely up to the implementation to choose when this buffer should be flushed -- most commonly this will be done at the beginning of rendering a frame and when some property values are queried.
Finally, objects can use helium::BaseObject::reportMessage()
to generically
report status messages through the application provided callbacks (setup and
managed by helium::BaseDevice
).
helium::BaseArray and helium::BaseFrame are both
classes which add additional interface to helium::BaseObject
for handling
ANARI API calls specific to those objects.
helium::BaseArray
also maitains a list of objects which opt-in to being
notified when an array changes -- this relationship is often best managed when
an object containing the array is committed. Note that array classes inheriting
from helium::BaseArray
are responsible for notifying objects observing them
as helium can't know when that is appropriate for every implementation. This
is commonly done when arrays are mapped/unmapped.
BaseArray
also implements the concept of "privatizing" an array -- when the
public ref count of an object goes to 0 and the internal ref count is non-zero,
a shared array may need to make a copy of the array data from the application
to continue functioning correctly because the application may free that memory.
If any array is released and the above ref count case is encountered, the
BaseArray::privatize()
method is invoked so the implementation can respond
accordingly based on what the implementation may require. Note that using
helium::IntrusivePtr
by default will only modify internal ref counts, so
exclusively using it will cleanly divide application ref count changes vs.
internal ref counts.
helium::BaseGlobalDeviceState is a struct containing
data that both devices and objects will share, but breaks the need for objects
to need to include the declaration of the device itself. While not required,
implementations are expected to inherit from BaseGlobalDeviceState
to append
to this storage whatever implementation-specific data is needed.
Implementations must construct the protected BaseDevice::m_state
variable so
that status callback output can function properly. Devices which extend
BaseGlobalDeviceState
instead should initialze m_state
with their own
derived type which the downstream implementation can safely cast to.