1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/drivers/include/pcf857x.h
Marian Buschsieweke e7c9451b55
drivers/pcf857x: use errno code for error reporting
This makes it easier to use common error reporting such as `strerror()`
or `tiny_strerror()` to give more insight on why something failed.

The custom error codes via `enum` have been updated to be synonymous
with the `errno` codes for backward compatibility.

In addition, `pcf857x_init()` has been updated to no longer or
together the return code, but rather abort on the first fail transaction
and return the error code as is. Otherwise (when both fail due to
different error codes) the returned error code may be garbage.
2023-11-22 10:29:33 +01:00

604 lines
24 KiB
C

/*
* Copyright (C) 2018 Gunar Schorcht
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @defgroup drivers_pcf857x PCF857X I2C I/O expanders
* @ingroup drivers_misc
* @ingroup drivers_saul
* @brief Device driver for Texas Instruments PCF857X I2C I/O expanders
*
*
* \section Device driver for Texas Instruments PCF857X I2C I/O expanders
*
* ## Overview
*
* Texas Instruments PCF857X I2C I/O expanders provide general purpose I/O
* extension via I2C bus. The driver supports the following PCF857X I2C I/O
* expander variants:
*
* <center>
* Expander | Type | Pseudomodule to be used
* :--------|:------------------------|:-----------------------
* PCF8574 | 8-bit I2C I/O expander | `pcf8574`
* PCF8574A | 8-bit I2C I/O expander | `pcf8574a`
* PCF8575 | 16-bit I2C I/O expander | `pcf8575`
* </center><br>
*
* For each of these PCF857X I2C I/O expanders variants, the driver defines
* a separate pseudomodule. Multiple PCF857X I2C I/O expanders and different
* variants can be used at the same time. Either the board definition or the
* application must specify used PCF857X I/O expander variants by a list of
* used pseudomodules. For example, to use a PCF8574A and a PCF8575 I/O
* expander in one application, the make command would be:
*
* USEMODULE="pcf8574a pcf8575" make -C tests/drivers/pcf857x BOARD=...
*
* At least one PCF857X I2C I/O expander variant has to be specified. The
* driver module `pcf857x` is then enabled implicitly.
*
* @note While PCF8575 is working in I2C fast mode with up to 400 kHz clock
* frequency, PCF8574 and PCF8574A are only specified for I2C normal mode
* with up to 100 kHz clock frequency. However, they seem also to work at
* 400 kHz clock frequency.
*
* The driver interface is kept as compatible as possible with the peripheral
* GPIO interface. The only differences are that
*
* - functions have the prefix `pcf857x_` and
* - functions require an additional parameter, the pointer to the expander
* device of type #pcf857x_t.
*
* ## Defined pseudomodules
*
* The functionality of the driver is controlled by the use of pseudomodules.
* The following pseudomodules are defined:
* <center>
* Pseudomoule | Functionality
* :--------------------|:---------------------
* `pcf8574` | support of PCF8574 enabled
* `pcf8574a` | support of PCF8574A enabled
* `pcf8575` | support of PCF8575 enabled
* `pcf857x_irq` | support of interrupts enabled with medium event priority
* `pcf857x_irq_low` | support of interrupts enabled with low event priority
* `pcf857x_irq_medium` | support of interrupts enabled with medium event priority
* `pcf857x_irq_low` | support of interrupts enabled with high event priority
* </center><br>
*
* @note At least one of the modules `pcf8574`, `pcf8574a` and `pcf8575` has to
* be used.
*
* ## Expander GPIOs
*
* The PCF857X expander devices provide a GPIO expansion over the I2C
* interface with either
*
* - 1 port with 8 quasi-parallel input/output (I/O) pins (PCF8574, PCF8574A) or
* - 2 ports with 8 quasi-parallel input/output (I/O) pins each (PCF8575).
*
* Each quasi-bidirectional expander I/O pin can be used as an input or output
* without the use of a data-direction control signal. Output pins are latched
* and have high-current drive capability for directly driving LEDs.
* The quasi-bidirectional expander I/O pins without direction control
* work as follows:
*
* - INPUT:
* Writing 1 to an expander pin configures the pin as an input, which is
* pulled up to HIGH by a very weak 100 μA pull-up. When reading the pin,
* its value then depends on the actual voltage. This corresponds to the
* behavior of the #GPIO_IN_PU mode.
* - OUTPUT:
* Writing 0 to an expander pin configures the pin as an output and
* actively drives the pin to LOW. This corresponds to the behavior of
* the #GPIO_OD_PU mode.
*
* @note Since the expander I/O pins are quasi-bidirectional without direction
* control, the only actively driven level is the output LOW. Therefore the
* driver physically supports only the modes #GPIO_IN_PU and #GPIO_OD_PU.
* The other logically identical modes #GPIO_IN, #GPIO_OUT and #GPIO_OD are
* emulated. Please keep this in mind when connecting these pins to other
* open-drain output pins that do not generate active signals. The #GPIO_IN_PD
* mode is not supported.
*
* After the initialization with the #pcf857x_init function, all
* expander I/O pins are in input mode and pulled-up to HIGH.
*
* The expander I/O pins can be addressed as GPIO pins using the following
* scheme:
*
* <center>
* PCF857X pin label | Port | Pin | RIOT symbol | Remark
* ----------------- |:----:|:---:|:--------------------------|:-----------------
* P00 | 0 | 0 | `PCF857X_GPIO_PIN(0, 0)` | PCF8574, PCF8574A and PCF8575
* P01 | 0 | 1 | `PCF857X_GPIO_PIN(0, 1)` | PCF8574, PCF8574A and PCF8575
* ... | ... | ... | ... | ...
* P07 | 0 | 7 | `PCF857X_GPIO_PIN(0, 7)` | PCF8574, PCF8574A and PCF8575
* P10 | 0 | 8 | `PCF857X_GPIO_PIN(0, 8)` | PCF8575 only
* P11 | 0 | 9 | `PCF857X_GPIO_PIN(0, 9)` | PCF8575 only
* ... | ... | ... | ... | ...
* P17 | 0 | 15 | `PCF857X_GPIO_PIN(0, 15)` | PCF8575 only
* </center>
*
* ## Interrupts
*
* PCF857X expanders have an open-drain, low-active interrupt (INT) signal,
* which generates an interrupt by any rising or falling edge of the expander
* pins in the input mode. Using this expander interrupt signal, the following
* features become available:
*
* - An interrupt service function can be attached to an expander input pin with
* the #pcf857x_gpio_init_int function. This interrupt service function is
* then called on any rising and/or falling edge of the expander input pin.
*
* - In addition, the driver uses the interrupt on changes of an expander
* input pin to internally maintain the current state of all expander
* input pins. Using this internal current state of the expander input
* pins avoids reading all expander input pins via I2C every time the input
* value of a single expander GPIO pin is read with #pcf857x_gpio_read.
*
* Since interrupts are handled in the context of a separate event thread (see
* section [The Interrupt Context Problem](#pcf857x_interrupt_context_problem))
* enabling interrupts requires more RAM. Therefore interrupts have to be
* explicitly enabled with the module `pcf857x_irq_<priority>`.
* `priority` can be one of `low`, `medium` or `highest`, which correspond to
* the priority of the event thread that processes the interrupts. If only the
* module `pcf857x_irq` is used without specifying the priority, the interrupt
* handling is enabled with a medium priority of the event thread. For more
* information on the priorities check the @ref sys_event module.
*
* Furthermore, the MCU GPIO pin to which the PCF857X `INT` signal is
* connected has to be defined by the default configuration parameter
* #PCF857X_PARAM_INT_PIN (pcf857x_params_t::int_pin) either in the
* configuration parameter file or at the command line, for example:
*
* CFLAGS="-DPCF857X_PARAM_INT_PIN=\(GPIO_PIN\(0,6\)\)" \
* USEMODULE="pcf8575 pcf857x_irq_medium" make -C tests/drivers/pcf857x BOARD=...
*
* <br>
* @note If an output of the expander is connected to an input of the same
* expander, there is no interrupt triggered by the input when the
* output changes. Therefore, a write operation to an output with
* #pcf857x_gpio_write, #pcf857x_gpio_clear, #pcf857x_gpio_set or
* #pcf857x_gpio_toggle leads to an additional read-after-write operation,
* if interrupts are used.<br><br>
* The use of interrupts therefore increases the read performance considerably,
* since I2C read operations are required only when the inputs change. But the
* write performance is reduced to the half.
*
* ## The Interrupt Context Problem {#pcf857x_interrupt_context_problem}
*
* Handling an interrupt of a PCF857x expander requires the driver to access
* the device directly via I2C. However, the mutex-based synchronization of
* I2C accesses does not work in the interrupt context. Therefore the ISR must
* not access the PCF857x expander device directly. Rather, the ISR must only
* indicate the occurrence of the interrupt which has to be handled
* asynchronously in thread context.
*
* For this purpose an event thread module is used when interrupts are
* enabled by the module `pcf857x_irq_<priority>`. The driver then
* handles the interrupts in the context of the event thread with given
* `priority`. For more information on the priorities check
* the @ref sys_event module.
*
* ## SAUL Capabilities
*
* The driver provides SAUL capabilities that are compatible to the SAUL
* capabilities of peripheral GPIOs. Each PCF857X expander I/O pin can be
* mapped directly to SAUL by defining an according entry in
* \c PCF857X_SAUL_GPIO_PARAMS. Please refer file
* `$RIOTBASE/drivers/pcf857x/include/pcf857x_params.h` for an example.
*
* @note Module `saul_gpio` has to be added to the
* project to enable SAUL capabilities of the PCF857X driver, e.g.:
*
* USEMODULE="pcf8575 saul_gpio" make -C tests/drivers/saul BOARD=...
*
* ## Using Multiple Devices
*
* It is possible to use multiple devices and different variants of PCF857X
* I/O expanders at the same time. Either the board definition or the
* application must specify used PCF857X I/O expander variants by a list of
* used pseudomodules. For example, to use a PCF8574A and a PCF8575 I/O
* expander in one application, the make command would be:
*
* USEMODULE="pcf8574a pcf8575" make -C tests/drivers/pcf857x BOARD=...
*
* Furthermore, used devices have to be configured by defining the
* configuration parameter set `pcf857x_params` of type #pcf857x_params_t.
* The default configuration for one device is defined in
* `drivers/pcf857x/pcf857x_params.h`. Either the board definition or the
* application can override it by placing a file `pcf857x_params.h` in the
* board definition directory or the application directory `$APPDIR`.
* For example, the definition of the configuration parameter array for the
* two devices above could be:
*
* static const pcf857x_params_t pcf857x_params[] = {
* {
* .dev = I2C_DEV(0),
* .addr = 0,
* .exp = PCF857X_EXP_PCF8574A,
* .int_pin = GPIO_PIN(0, 1),
* },
* {
* .dev = I2C_DEV(0),
* .addr = 0,
* .exp = PCF857X_EXP_PCF8575,
* .int_pin = GPIO_PIN(0, 2),
* },
* };
*
* @{
*
* @author Gunar Schorcht <gunar@schorcht.net>
* @file
*/
#ifndef PCF857X_H
#define PCF857X_H
#ifdef __cplusplus
extern "C"
{
#endif
#include <errno.h>
#include <stdbool.h>
#include <stdint.h>
#include "periph/gpio.h"
#include "periph/i2c.h"
#if IS_USED(MODULE_SAUL_GPIO) || DOXYGEN
#include "saul/periph.h"
#endif /* MODULE_SAUL_GPIO */
#if IS_USED(MODULE_PCF857X_IRQ) || DOXYGEN
#include "event.h"
#endif /* MODULE_PCF857X_IRQ */
/**
* @name PCF857X I2C slave addresses
*
* PCF857X I2C slave addresses are defined as an offset to a base address,
* which depends on the expander used. The address offset is in the range
* of 0 to 7.
* @{
*/
#ifndef PCF8575_BASE_ADDR
#define PCF8575_BASE_ADDR (0x20) /**< PCF8575 I2C slave base address.
Addresses are then in range from
0x20 to 0x27 */
#endif
#ifndef PCF8574_BASE_ADDR
#define PCF8574_BASE_ADDR (0x20) /**< PCF8574 I2C slave base address.
Addresses are then in range from
0x20 to 0x27 */
#endif
#ifndef PCF8574A_BASE_ADDR
#define PCF8574A_BASE_ADDR (0x38) /**< PCF8574A I2C slave base address.
Addresses are then in range from
0x38 to 0x3f */
#endif
/** @} */
/**
* @name PCF857X I/O expander pin number
* @{
*/
#define PCF8575_GPIO_PIN_NUM (16) /**< PCF8575 has 16 I/O pins */
#define PCF8574_GPIO_PIN_NUM (8) /**< PCF8574 has 8 I/O pins */
#define PCF8574A_GPIO_PIN_NUM (8) /**< PCF8574A has 8 I/O pins */
/** @} */
/** conversion of (port x : pin y) to a pin number */
#define PCF857X_GPIO_PIN(x,y) ((gpio_t)((x << 3) | y))
/**
* @name Module dependent definitions and declarations
* @{
*/
#if IS_USED(MODULE_PCF8575) || DOXYGEN
/**
* @brief Maximum number of GPIO pins
*
* Defines the maximum number of GPIO pins of all PCF857X I/O expanders
* used. If a PCF8575 is used, the maximum number is 16 I/O pins.
*/
#define PCF857X_GPIO_PIN_NUM (16)
/**
* @brief Data type that can mask all expander pins
*
* If a PCF8575 is used, the 16 I/O pins have to be masked.
*/
typedef uint16_t pcf857x_data_t;
#else /* MODULE_PCF8575 || DOXYGEN */
#define PCF857X_GPIO_PIN_NUM (8) /**< PCF8574, PCF8574 provide 8 I/O pins */
typedef uint8_t pcf857x_data_t; /**< type that can mask all expander pins */
#endif /* MODULE_PCF8575 || DOXYGEN */
/** @} */
/**
* @brief Definition of PCF857X driver error codes
*
* @deprecated These are aliases for errno error codes now, use them directly
*/
typedef enum {
PCF857X_OK = 0, /**< success */
PCF857X_ERROR_I2C = ENXIO, /**< I2C communication error */
PCF857X_ERROR_INV_EXP = ENOTSUP, /**< invalid expander variant */
PCF857X_ERROR_INV_MODE = EINVAL, /**< invalid pin mode */
PCF857X_ERROR_INV_FLANK = EINVAL, /**< invalid interrupt flank */
PCF857X_ERROR_INT_PIN = ENOSYS, /**< interrupt pin initialization failed */
} pcf857x_error_codes_t;
/**
* @brief Definition of PCF857X expander variants
*
* It is used in configuration parameters to specify the PCF857X expander
* used by device.
*
* @note Expander variants known by the driver depend on enabled pseudomodules
* `pcf8574`, `pcf8574a` and `pcf8575`.
*/
typedef enum {
#if IS_USED(MODULE_PCF8574) || DOXYGEN
PCF857X_EXP_PCF8574, /**< PCF8574 8 bit I/O expander used */
#endif
#if IS_USED(MODULE_PCF8574A) || DOXYGEN
PCF857X_EXP_PCF8574A, /**< PCF8574A 8 bit I/O expander */
#endif
#if IS_USED(MODULE_PCF8575) || DOXYGEN
PCF857X_EXP_PCF8575, /**< PCF8575 16 bit I/O expander */
#endif
PCF857X_EXP_MAX,
} pcf857x_exp_t;
/**
* @brief PCF857X device initialization parameters
*/
typedef struct {
i2c_t dev; /**< I2C device (default I2C_DEV(0)) */
uint16_t addr; /**< I2C slave address offset to the PCF7857X base
address (default 0) */
pcf857x_exp_t exp; /**< PCF857X expander variant used by the device
(default depends on used pseudomodules */
#if IS_USED(MODULE_PCF857X_IRQ) || DOXYGEN
gpio_t int_pin; /**< MCU GPIO pin or #GPIO_UNDEF if not used (default).
Using interrupt pin has the advantage that inputs
have to be read from expander only if any input
value changes.
@note To use interrupts for expander inputs, this
pin has to be defined. */
#endif /* MODULE_PCF857X_IRQ */
} pcf857x_params_t;
#if IS_USED(MODULE_PCF857X_IRQ) || DOXYGEN
/**
* @brief IRQ event type
*
* Handling an interrupt of a PCF857x expander requires the driver to access
* the device directly via I2C. However, the mutex-based synchronization of
* I2C accesses does not work in the interrupt context. Therefore the ISR must
* not access the PCF857x expander device directly. Rather, the ISR must only
* indicate the occurrence of the interrupt which has to be handled
* asynchronously in the thread context.
*
* The type defines the data structure that is part of each device data
* structure to indicate that an interrupt of the device occurred. Since there
* is only one interrupt source, only one interrupt can be pending per device.
* Thus, only one object of this type per device is required.
*/
typedef struct {
event_t event; /**< inherited event data structure */
void *dev; /**< PCF857X device reference */
} pcf857x_irq_event_t;
#endif /* MODULE_PCF857X_IRQ */
/**
* @brief PCF857X device data structure type
*/
typedef struct {
pcf857x_params_t params; /**< device initialization parameters */
uint8_t pin_num; /**< number of I/O pins, depends on used expander
variant */
pcf857x_data_t modes; /**< expander pin modes */
pcf857x_data_t in; /**< expander input pin values */
pcf857x_data_t out; /**< expander output pin values */
#if IS_USED(MODULE_PCF857X_IRQ) || DOXYGEN
gpio_isr_ctx_t isr[PCF857X_GPIO_PIN_NUM]; /**< ISR with arg for each expander pin */
gpio_flank_t flank[PCF857X_GPIO_PIN_NUM]; /**< interrupt flank for each expander pin */
bool enabled[PCF857X_GPIO_PIN_NUM]; /**< enabled flag for each expander pin */
pcf857x_irq_event_t irq_event; /**< IRQ event object used for the device */
#endif /* MODULE_PCF857X_IRQ */
} pcf857x_t;
#if IS_USED(MODULE_SAUL_GPIO) || DOXYGEN
/**
* @brief PCF857X configuration structure for mapping expander pins to SAUL
*
* This data structure is an extension of the GPIO configuration structure for
* mapping GPIOs to SAUL. The only additional information required is a
* reference to the according PCF857X device.
*
* @note To use PCF857X with SAUL, module `saul_gpio` has to be added to the
* project.
*/
typedef struct {
unsigned int dev; /**< PCF857X device index */
saul_gpio_params_t gpio; /**< GPIO configuration for mapping to SAUL */
} pcf857x_saul_gpio_params_t;
#endif
/**
* @brief Initialize the PCF857X I/O expander
*
* All expander pins are set to be input and are pulled up.
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] params configuration parameters, see #pcf857x_params_t
*
* @pre If the interrupt handling is enabled by one of the modules
* `pcf857x_irq*`, the MCU GPIO pin for the interrupt signal
* has to be defined by the default configuration parameter
* #PCF857X_PARAM_INT_PIN (pcf857x_params_t::int_pin).
*
* @retval 0 on success
* @retval <0 a negative errno error code on error
*/
int pcf857x_init(pcf857x_t *dev, const pcf857x_params_t *params);
/**
* @brief Initialize a PCF857X pin
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] pin pin to initialize, use PCF857X_GPIO_PIN(x,y) to specify
* @param[in] mode mode of the pin, see #gpio_t
*
* @note
* - Since the expander I/O pins are quasi-bidirectional without direction
* control, the only actively driven level is the output LOW. Therefore
* the driver physically supports only the modes #GPIO_IN_PU and
* #GPIO_OD_PU. The other logically identical modes #GPIO_IN, #GPIO_OUT
* and #GPIO_OD are emulated. For the #GPIO_IN_PU mode the function returns
* with `-EINVAL`.
* - After initialization in #GPIO_OUT mode the pin is actively driven LOW,
* after initialization in all other modes the pin is pulled-up to HIGH.
*
* @retval 0 on success
* @retval <0 a negative errno error code on error
*/
int pcf857x_gpio_init(pcf857x_t *dev, gpio_t pin, gpio_mode_t mode);
#if IS_USED(MODULE_PCF857X_IRQ) || DOXYGEN
/**
* @brief Initialize a PCF857X pin for external interrupt usage
*
* The registered callback function will be called in interrupt context every
* time the defined flank(s) are detected. Therefore, it MUST NOT be blocking
* or time-consuming.
*
* The interrupt is activated automatically after the initialization.
*
* @pre The MCU GPIO pin for the interrupt signal has to be defined by the
* default configuration parameter #PCF857X_PARAM_INT_PIN
* (pcf857x_params_t::int_pin).
*
* @note
* - This function is only available if interrupt handling is enabled by one
* of the modules `pcf857x_irq*`
* - Since the expander I/O pins are quasi-bidirectional without direction
* control, the only actively driven level is the output LOW. Therefore
* the driver physically supports only the modes #GPIO_IN_PU and
* #GPIO_OD_PU. The other logically identical modes #GPIO_IN, #GPIO_OUT
* and #GPIO_OD are emulated. For the #GPIO_IN_PU mode the function returns
* with `-EINVAL`.
* - After initialization in #GPIO_OUT mode the pin is actively driven LOW,
* after initialization in all other modes the pin is pulled-up to HIGH.
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] pin pin to initialize, use PCF857X_GPIO_PIN(x,y) to specify
* @param[in] mode mode of the pin, see #gpio_t
* @param[in] flank define the active flanks, see #gpio_flank_t
* @param[in] isr ISR that is called back from interrupt context
* @param[in] arg optional argument passed to the callback
*
* @retval 0 on success
* @retval <0 a negative errno error code on error
*/
int pcf857x_gpio_init_int(pcf857x_t *dev, gpio_t pin,
gpio_mode_t mode,
gpio_flank_t flank,
gpio_cb_t isr,
void *arg);
#endif /* MODULE_PCF857X_IRQ || DOXYGEN */
/**
* @brief Get the value from PCF857X input pin
*
* @note If the PCF857X interrupt is used, the read operation does not perform
* an I2C read operation since the last input pin value is already read.
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] pin pin to read, use PCF857X_GPIO_PIN(x,y) to specify
*/
int pcf857x_gpio_read(pcf857x_t *dev, gpio_t pin);
/**
* @brief Write the value to PCF857X input pin
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] pin pin to write, use PCF857X_GPIO_PIN(x,y) to specify
* @param[in] value value to write
*/
void pcf857x_gpio_write(pcf857x_t *dev, gpio_t pin, int value);
/**
* @brief Clear the PCF857X output pin
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] pin pin to clear, use PCF857X_GPIO_PIN(x,y) to specify
*/
void pcf857x_gpio_clear(pcf857x_t *dev, gpio_t pin);
/**
* @brief Set the PCF857X output pin
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] pin pin to set, use PCF857X_GPIO_PIN(x,y) to specify
*/
void pcf857x_gpio_set(pcf857x_t *dev, gpio_t pin);
/**
* @brief Toggle the value of the PCF857X output pin
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] pin pin to toggle, use PCF857X_GPIO_PIN(x,y) to specify
*/
void pcf857x_gpio_toggle(pcf857x_t *dev, gpio_t pin);
#if IS_USED(MODULE_PCF857X_IRQ) || DOXYGEN
/**
* @brief Enable pin interrupt
*
* @note This function is only available if interrupt handling is enabled
* by one of the modules `pcf857x_irq*`
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] pin pin to enable the interrupt for
*/
void pcf857x_gpio_irq_enable(pcf857x_t *dev, gpio_t pin);
/**
* @brief Disable pin interrupt
*
* @note This function is only available if interrupt handling is enabled
* by one of the modules `pcf857x_irq*`
*
* @param[in] dev descriptor of PCF857X I/O expander device
* @param[in] pin pin to enable the interrupt for
*/
void pcf857x_gpio_irq_disable(pcf857x_t *dev, gpio_t pin);
#endif /* MODULE_PCF857X_IRQ || DOXYGEN */
#ifdef __cplusplus
}
#endif
#endif /* PCF857X_H */
/** @} */