Вы можете реорганизовать условно скомпилированный код для динамического чтения значений из интерфейса HAL. Например:
#ifdef TARGET_FORCE_HWC_FOR_VIRTUAL_DISPLAYS //some code fragment #endif
 Затем код фреймворка может вызвать соответствующую служебную функцию, определенную в <configstore/Utils.h> в зависимости от ее типа.
Пример ConfigStore
 В этом примере показано чтение TARGET_FORCE_HWC_FOR_VIRTUAL_DISPLAYS , определенного в ConfigStore HAL как forceHwcForVirtualDisplays() с возвращаемым типом OptionalBool :
#include <configstore/Utils.h>
using namespace android::hardware::configstore;
using namespace android::hardware::configstore::V1_0;
static bool vsyncPhaseOffsetNs = getBool<ISurfaceFlingerConfigs,
        ISurfaceFlingerConfigs::forceHwcForVirtualDisplays>(false);
 Вспомогательная функция ( getBool в приведенном выше примере) связывается со службой configstore , чтобы получить дескриптор прокси функции интерфейса, а затем извлекает значение, вызывая дескриптор через HIDL/hwbinder.
Вспомогательные функции
 <configstore/Utils.h> ( configstore/1.0/include/configstore/Utils.h ) предоставляет служебные функции для каждого примитивного возвращаемого типа, включая Optional[Bool|String|Int32|UInt32|Int64|UInt64] , как указано ниже:
| Тип | Функция (параметры шаблона опущены) | 
|---|---|
 OptionalBool |  bool getBool(const bool defValue) | 
 OptionalInt32 |  int32_t getInt32(const int32_t defValue) | 
 OptionalUInt32 |  uint32_t getUInt32(const uint32_t defValue) | 
 OptionalInt64 |  int64_t getInt64(const int64_t defValue) | 
 OptionalUInt64 |  uint64_t getUInt64(const uint64_t defValue) | 
 OptionalString |  std::string getString(const std::string &defValue) | 
 defValue — это значение по умолчанию, возвращаемое, когда реализация HAL не указывает значение для элемента конфигурации. Каждая функция принимает два параметра шаблона:
-  
I— имя класса интерфейса. -  
Func— это указатель функции-члена для получения элемента конфигурации. 
Поскольку значение конфигурации доступно только для чтения и не изменяется, служебная функция внутренне кэширует значение конфигурации. Последующие вызовы обслуживаются более эффективно с использованием кэшированного значения в том же блоке связывания.
Использование configstore-utils
 HAL ConfigStore предназначен для прямой совместимости с обновлениями второстепенных версий, а это означает, что, когда HAL пересматривается и некоторый код платформы использует недавно введенные элементы, служба ConfigStore с более старой второстепенной версией в /vendor все еще может использоваться.
Для прямой совместимости убедитесь, что ваша реализация соответствует следующим рекомендациям:
-  Новые элементы используют значение по умолчанию, когда доступна только служба старой версии. Пример:
service = V1_1::IConfig::getService(); // null if V1_0 is installed value = DEFAULT_VALUE; if(service) { value = service->v1_1API(DEFAULT_VALUE); } -  Клиент использует первый интерфейс, содержащий элемент ConfigStore. Пример:
V1_1::IConfig::getService()->v1_0API(); // NOT ALLOWED V1_0::IConfig::getService()->v1_0API(); // OK
 -  Служба новой версии может быть получена для интерфейса старой версии. В следующем примере, если установлена версия v1_1, служба v1_1 должна быть возвращена для 
getService():V1_0::IConfig::getService()->v1_0API();
 
 Когда функции доступа в библиотеке configstore-utils используются для доступа к элементу ConfigStore, #1 гарантируется реализацией, а #2 гарантируется ошибками компилятора. По этим причинам мы настоятельно рекомендуем использовать configstore-utils везде, где это возможно.