| // Copyright lowRISC contributors. |
| // Licensed under the Apache License, Version 2.0, see LICENSE for details. |
| // SPDX-License-Identifier: Apache-2.0 |
| |
| #ifndef OPENTITAN_SW_DEVICE_LIB_DIF_DIF_ADC_CTRL_H_ |
| #define OPENTITAN_SW_DEVICE_LIB_DIF_DIF_ADC_CTRL_H_ |
| |
| /** |
| * @file |
| * @brief <a href="/hw/ip/adc_ctrl/doc/">ADC Controller</a> Device Interface |
| * Functions |
| */ |
| |
| #include "sw/device/lib/dif/autogen/dif_adc_ctrl_autogen.h" |
| |
| #ifdef __cplusplus |
| extern "C" { |
| #endif // __cplusplus |
| |
| /** |
| * Helper X macro for defining enums and case statements related to ADC |
| * Controller channels. If an additional channel is ever added to the hardware, |
| * this list can be updated. |
| */ |
| #define DIF_ADC_CTRL_CHANNEL_LIST(X) \ |
| X(0) \ |
| X(1) |
| |
| /** |
| * Helper X macro for defining enums and case statements related to ADC |
| * Controller filters. If an additional filter is ever added to the hardware, |
| * this list can be updated. |
| */ |
| #define DIF_ADC_CTRL_FILTER_LIST(X) \ |
| X(0) \ |
| X(1) \ |
| X(2) \ |
| X(3) \ |
| X(4) \ |
| X(5) \ |
| X(6) \ |
| X(7) |
| |
| /** |
| * Helper macro for defining a `dif_adc_ctrl_channel_t` enumeration constant. |
| * @channel_ ADC Controller channel of the enumeration constant. |
| */ |
| #define DIF_ADC_CTRL_CHANNEL_ENUM_INIT_(channel_) \ |
| kDifAdcCtrlChannel##channel_ = channel_, |
| |
| /** |
| * An ADC Controller Channel. |
| */ |
| typedef enum dif_adc_ctrl_channel { |
| DIF_ADC_CTRL_CHANNEL_LIST(DIF_ADC_CTRL_CHANNEL_ENUM_INIT_) |
| } dif_adc_ctrl_channel_t; |
| |
| #undef DIF_ADC_CTRL_CHANNEL_ENUM_INIT_ |
| |
| /** |
| * Helper macro for defining a `dif_adc_ctrl_filter_t` enumeration constant. |
| * @filter_ ADC Controller filter of the enumeration constant. |
| */ |
| #define DIF_ADC_CTRL_FILTER_ENUM_INIT_(filter_) \ |
| kDifAdcCtrlFilter##filter_ = filter_, |
| |
| /** |
| * An ADC Controller filter. |
| * |
| * Each channel has a separate instance of each filter. For example, if there |
| * are two channels and eight filters, there would be a total of 16 filter |
| * instances that may be configured. |
| */ |
| typedef enum dif_adc_ctrl_filter { |
| DIF_ADC_CTRL_FILTER_LIST(DIF_ADC_CTRL_FILTER_ENUM_INIT_) |
| } dif_adc_ctrl_filter_t; |
| |
| #undef DIF_ADC_CTRL_FILTER_ENUM_INIT_ |
| |
| /** |
| * Helper macro for defining a `dif_adc_ctrl_irq_cause_t` enumeration constant. |
| * @filter_cause_ ADC Controller IRQ filter cause of the enumeration constant. |
| */ |
| #define DIF_ADC_CTRL_IRQ_CAUSE_ENUM_INIT_(filter_cause_) \ |
| kDifAdcCtrlIrqCauseFilter##filter_cause_ = 1U << filter_cause_, |
| |
| /** |
| * An ADC Controller IRQ cause. |
| * |
| * The ADC Controller can only generate a single interrupt (the `debug_cable` |
| * interrupt). However, depending on how the ADC Controller is configured, there |
| * are several causes that could trigger this interrupt. These include filter |
| * matches (when in Normal Power Scan mode), or sample completion (when in |
| * Oneshot mode). |
| */ |
| typedef enum dif_adc_ctrl_irq_cause { |
| DIF_ADC_CTRL_FILTER_LIST(DIF_ADC_CTRL_IRQ_CAUSE_ENUM_INIT_) |
| /** |
| * Sample ready cause in Oneshot mode. |
| */ |
| kDifAdcCtrlIrqCauseOneshot = 1U << 8, |
| /** |
| * All IRQ causes ORed together. |
| * |
| * This is useful when clearing all IRQ causes at once, to initialize the ADC |
| * Controller. |
| */ |
| kDifAdcCtrlIrqCauseAll = (1U << 9) - 1, |
| } dif_adc_ctrl_irq_cause_t; |
| |
| #undef DIF_ADC_CTRL_IRQ_CAUSE_ENUM_INIT_ |
| |
| /** |
| * Operation mode of the ADC Controller. |
| */ |
| typedef enum dif_adc_ctrl_mode { |
| /** |
| * Low Power (Continuous) Scan mode. |
| * |
| * In Low Power Scan mode, the ADC periodically samples enabled channels, and |
| * upon matching a set of enabled filters, a set number of times, will |
| * transition to Normal Power Scan mode. If no filters are enabled, then the |
| * ADC controller will never transition to Normal Power Scan mode. |
| */ |
| kDifAdcCtrlLowPowerScanMode = 0, |
| /** |
| * Normal Power (Continuous) Scan mode. |
| * |
| * In Normal Power Scan mode, the ADC samples enabled channels as fast as |
| * possible, and upon matching a set of enabled filters, a set number of |
| * consecutive times, may trigger a system wakeup and/or IRQ. Similar to Low |
| * Power Scan mode, if no filters are enabled, then a system wakeup and/or IRQ |
| * will never be triggered. |
| */ |
| kDifAdcCtrlNormalPowerScanMode = 1, |
| /** |
| * Oneshot mode. |
| * |
| * In Oneshot mode, an ADC channel is triggered to take a single sample, upon |
| * being enabled, and optionally, raises an interrupt upon completion. Unlike |
| * the Scan modes, in Oneshot mode, the ADC Controller does not attempt to |
| * filter samples. Rather, an IRQ may be raised immediately upon the sample |
| * being ready, regardless of what the sample is. After the sample is |
| * completed the ADC is powered down, until another sample is triggered, |
| * either by toggling the channel's enable bit on and off, or by resetting the |
| * sampling FSM. |
| */ |
| kDifAdcCtrlOneshotMode = 2, |
| } dif_adc_ctrl_mode_t; |
| |
| /** |
| * Runtime configuration for an ADC Controller. |
| */ |
| typedef struct dif_adc_ctrl_config { |
| /** |
| * The sampling mode to configure the ADC Controller in. |
| */ |
| dif_adc_ctrl_mode_t mode; |
| /** |
| * The time to allow the ADC to power up. |
| * |
| * Units: always-on clock cycles |
| */ |
| uint8_t power_up_time_aon_cycles; |
| /** |
| * The sampling period when in Low Power Scan mode, i.e., how often the ADC |
| * Controller wakes up the ADC to take a sample. |
| * |
| * Units: always-on clock cycles |
| * |
| * Only relevant in Low Power Scan mode. |
| */ |
| uint32_t wake_up_time_aon_cycles; |
| /** |
| * The number of filter-matching samples to count in Low Power Scan mode |
| * before switching to Normal Power Scan mode. |
| * |
| * Only relevant in Low Power Scan mode. |
| */ |
| uint8_t num_low_power_samples; |
| /** |
| * The number of filter-matching samples to count in Normal Power Scan mode |
| * before triggering a system wakeup and/or interrupt. |
| */ |
| uint16_t num_normal_power_samples; |
| } dif_adc_ctrl_config_t; |
| |
| /** |
| * Runtime configuration for an ADC Controller filter. |
| */ |
| typedef struct dif_adc_ctrl_filter_config { |
| /** |
| * The ADC Controller filter this configuration applies to. |
| */ |
| dif_adc_ctrl_filter_t filter; |
| /** |
| * The minimum voltage (inclusive) of the range defined by this filter. |
| * |
| * Valid range: [0, 1024) |
| * Units: 2.148 mV (i.e., range / 2 ^ 10) |
| */ |
| uint16_t min_voltage; |
| /** |
| * The maximum voltage (inclusive) of the range defined by this filter. |
| * |
| * Valid range: [0, 1024) |
| * Units: 2.148 mV (i.e., range / 2 ^ 10) |
| */ |
| uint16_t max_voltage; |
| /** |
| * Where a filter hit is classfied as an (inclusive) in-range hit, or |
| * (exclusive) out-of-range hit. |
| */ |
| bool in_range; |
| /** |
| * Whether to generate a system wakeup on a filter match after saturating the |
| * `num_normal_power_samples` threshold in Normal Power Scan mode. |
| */ |
| bool generate_wakeup_on_match; |
| /** |
| * Whether to generate a `debug_cable` interrupt on a filter match after |
| * saturating the `num_normal_power_samples` threshold in Normal Power Scan |
| * mode. |
| */ |
| bool generate_irq_on_match; |
| } dif_adc_ctrl_filter_config_t; |
| |
| /** |
| * Configures an ADC Controller. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param config Runtime configuration parameters. |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_configure(const dif_adc_ctrl_t *adc_ctrl, |
| dif_adc_ctrl_config_t config); |
| |
| /** |
| * Configures a channel filter. |
| * |
| * This should be invoked for each desired filter _before_ the sampling sequence |
| * is enabled via `dif_adc_ctrl_set_enabled()`. |
| * |
| * This only applies in Low / Normal Power Scan sampling modes. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param channel The channel of the filter to configure. |
| * @param config Runtime configuration parameters for the filter. |
| * @param enabled The enablement state to configure the filter in. |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_configure_filter(const dif_adc_ctrl_t *adc_ctrl, |
| dif_adc_ctrl_channel_t channel, |
| dif_adc_ctrl_filter_config_t config, |
| dif_toggle_t enabled); |
| |
| /** |
| * Sets the enablement state of the ADC Controller. |
| * |
| * Enabling the ADC Controller powers it up, while disabling the ADC Controller |
| * powers it down and resets the sampling FSM. After powering up, sampling |
| * begins, regardless of the operation mode. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param enabled The enablement state to configure the ADC Controller in. |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_set_enabled(const dif_adc_ctrl_t *adc_ctrl, |
| dif_toggle_t enabled); |
| |
| /** |
| * Gets the enablement state of the ADC Controller. |
| * |
| * If the ADC Controller is enabled, it is powered up, or being powered up. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param[out] is_enabled The enablement state of the ADC Controller. |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_get_enabled(const dif_adc_ctrl_t *adc_ctrl, |
| dif_toggle_t *is_enabled); |
| |
| /** |
| * Sets the enablement state of the specified filter for the specified channel. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param channel The channel the filter resides in. |
| * @param filter The filter to set the enablement state of. |
| * @param enabled The enablement state to configure the filter in. |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_filter_set_enabled(const dif_adc_ctrl_t *adc_ctrl, |
| dif_adc_ctrl_channel_t channel, |
| dif_adc_ctrl_filter_t filter, |
| dif_toggle_t enabled); |
| |
| /** |
| * Gets the enablement state of the specified filter for the specified channel. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param channel The channel the filter resides in. |
| * @param filter The filter to get the enablement state of. |
| * @param[out] is_enabled The enablement state of the filter. |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_filter_get_enabled(const dif_adc_ctrl_t *adc_ctrl, |
| dif_adc_ctrl_channel_t channel, |
| dif_adc_ctrl_filter_t filter, |
| dif_toggle_t *enabled); |
| |
| /** |
| * Get the sampled value from the specified channel that triggered the IRQ. |
| * |
| * Values are 10-bits in the range from 0V to 2.2V. Based on this, the |
| * resolution (and units) of the sample are in increments of 2.148mV. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param channel The channel to read the sample from. |
| * @param[out] value The value of the sample. |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_get_triggered_value(const dif_adc_ctrl_t *adc_ctrl, |
| dif_adc_ctrl_channel_t channel, |
| uint16_t *value); |
| |
| /** |
| * Get the latest sampled value from the specified channel. |
| * |
| * Since in Normal Power Scan mode, sampling continues even after an IRQ has |
| * been raised, the value returned by this function may be different than the |
| * value returned by `dif_adc_ctrl_get_irq_value()`. |
| * |
| * Values are 10-bits in the range from 0V to 2.2V. Based on this, the |
| * resolution (and units) of the sample are in increments of 2.148mV. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param channel The channel to read the sample from. |
| * @param[out] value The value of the sample. |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_get_latest_value(const dif_adc_ctrl_t *adc_ctrl, |
| dif_adc_ctrl_channel_t channel, |
| uint16_t *value); |
| |
| /** |
| * Reset all ADC Controller FSMs and counters, and if enabled, begin sampling |
| * sequence. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_reset(const dif_adc_ctrl_t *adc_ctrl); |
| |
| /** |
| * Gets the cause(s) of a `debug_cable` IRQ. |
| * |
| * IRQs can be triggered by filter matches in Normal Power Scan mode (after |
| * saturating the `num_normal_power_samples` threshold), or after a single |
| * sample capture in Oneshot mode. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param[out] causes The causes of the IRQ (one or more |
| * `dif_adc_ctrl_irq_cause_t`s ORed together). |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_irq_get_causes(const dif_adc_ctrl_t *adc_ctrl, |
| uint32_t *causes); |
| |
| /** |
| * Clears the cause(s) of a `debug_cable` IRQ. |
| * |
| * TODO(lowRISC/opentitan:#11354): future releases of the HW should hide the |
| * filter and interrupt status registers behind the standardized IRQ registers. |
| * For now, the autogenerated `dif_adc_ctrl_irq_acknowledge[_all]()` DIF may be |
| * used to clear the main IRQ status register, while this DIF may be used to |
| * clear the local cause / filter status registers. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param causes The causes of the IRQ (one or more `dif_adc_ctrl_irq_cause_t`s |
| * ORed together). |
| * @return The result of the operation. |
| */ |
| OT_WARN_UNUSED_RESULT |
| dif_result_t dif_adc_ctrl_irq_clear_causes(const dif_adc_ctrl_t *adc_ctrl, |
| uint32_t causes); |
| |
| /** |
| * Sets the enablement of generating system wakeups on a filter match. |
| * |
| * Only relevant in Normal Power Scan mode (and Low Power Scan mode, which can |
| * transition to Normal Power Scan mode). |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param filter A filter to enable wakeup triggering for. |
| * @param enabled The enablement state to set. |
| * @return The result of the operation. |
| */ |
| dif_result_t dif_adc_ctrl_filter_match_wakeup_set_enabled( |
| const dif_adc_ctrl_t *adc_ctrl, dif_adc_ctrl_filter_t filter, |
| dif_toggle_t enabled); |
| |
| /** |
| * Gets the enablement of generating system wakeups on a filter match. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param filter A filter to enable wakeup triggering for. |
| * @param[out] is_enabled The enablement state retrieved. |
| * @return The result of the operation. |
| */ |
| dif_result_t dif_adc_ctrl_filter_match_wakeup_get_enabled( |
| const dif_adc_ctrl_t *adc_ctrl, dif_adc_ctrl_filter_t filter, |
| dif_toggle_t *is_enabled); |
| |
| /** |
| * Sets the enablement of generating a `debug_cable` IRQ for given cause(s). |
| * |
| * Causes can be filter matches (in Normal Power Scan mode), or when a sample is |
| * complete (in Oneshot mode). |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param causes Causes (one or more `dif_adc_ctrl_irq_cause_t`s ORed together) |
| * to generate the `debug_cable` IRQ for. |
| * @param enabled The enablement state to set. |
| * @return The result of the operation. |
| */ |
| dif_result_t dif_adc_ctrl_irq_cause_set_enabled(const dif_adc_ctrl_t *adc_ctrl, |
| uint32_t causes, |
| dif_toggle_t enabled); |
| |
| /** |
| * Gets the causes that will generate a `debug_cable` IRQ. |
| * |
| * @param adc_ctrl An adc_ctrl handle. |
| * @param[out] enabled_causes Causes (one or more `dif_adc_ctrl_irq_cause_t`s |
| * ORed together) that will generate the |
| * `debug_cable` IRQ. |
| * @return The result of the operation. |
| */ |
| dif_result_t dif_adc_ctrl_irq_cause_get_enabled(const dif_adc_ctrl_t *adc_ctrl, |
| uint32_t *enabled_causes); |
| |
| #ifdef __cplusplus |
| } // extern "C" |
| #endif // __cplusplus |
| |
| #endif // OPENTITAN_SW_DEVICE_LIB_DIF_DIF_ADC_CTRL_H_ |