/*
    ChibiOS - Copyright (C) 2006..2018 Giovanni Di Sirio
    ChibiOS - Copyright (C) 2023..2025 HorrorTroll
    ChibiOS - Copyright (C) 2023..2025 Zhaqian
    ChibiOS - Copyright (C) 2024..2025 Maxjta

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
*/

/**
 * @file    ADCv2/hal_adc_lld.h
 * @brief   AT32 ADC subsystem low level driver header.
 *
 * @addtogroup ADC
 * @{
 */

#ifndef HAL_ADC_LLD_H
#define HAL_ADC_LLD_H

#if HAL_USE_ADC || defined(__DOXYGEN__)

/*===========================================================================*/
/* Driver constants.                                                         */
/*===========================================================================*/

/**
 * @name    Possible ADC errors mask bits.
 * @{
 */
#define ADC_ERR_DMAFAILURE      1U  /**< DMA operations failure.            */
#define ADC_ERR_VM              4U  /**< Voltage monitoring triggered.      */
/** @} */

/**
 * @name    Absolute Maximum Ratings
 * @{
 */
/**
 * @brief   Minimum ADC clock frequency.
 */
#define AT32_ADCCLK_MIN        600000

/**
 * @brief   Maximum ADC clock frequency.
 */
#define AT32_ADCCLK_MAX        28000000
/** @} */

/**
 * @name    Triggers selection
 * @{
 */
#define ADC_CTRL2_OCTESEL_MASK    (7U << 17U)
#define ADC_CTRL2_OCTESEL_SRC(n)  ((n) << 17U)
/** @} */

/**
 * @name    Available analog channels
 * @{
 */
#define ADC_CHANNEL_IN0         0   /**< @brief External analog input 0.    */
#define ADC_CHANNEL_IN1         1   /**< @brief External analog input 1.    */
#define ADC_CHANNEL_IN2         2   /**< @brief External analog input 2.    */
#define ADC_CHANNEL_IN3         3   /**< @brief External analog input 3.    */
#define ADC_CHANNEL_IN4         4   /**< @brief External analog input 4.    */
#define ADC_CHANNEL_IN5         5   /**< @brief External analog input 5.    */
#define ADC_CHANNEL_IN6         6   /**< @brief External analog input 6.    */
#define ADC_CHANNEL_IN7         7   /**< @brief External analog input 7.    */
#define ADC_CHANNEL_IN8         8   /**< @brief External analog input 8.    */
#define ADC_CHANNEL_IN9         9   /**< @brief External analog input 9.    */
#define ADC_CHANNEL_IN10        10  /**< @brief External analog input 10.   */
#define ADC_CHANNEL_IN11        11  /**< @brief External analog input 11.   */
#define ADC_CHANNEL_IN12        12  /**< @brief External analog input 12.   */
#define ADC_CHANNEL_IN13        13  /**< @brief External analog input 13.   */
#define ADC_CHANNEL_IN14        14  /**< @brief External analog input 14.   */
#define ADC_CHANNEL_IN15        15  /**< @brief External analog input 15.   */
#define ADC_CHANNEL_SENSOR      16  /**< @brief Internal temperature sensor.
                                         @note Available on ADC1 only.      */
#define ADC_CHANNEL_VREF        17  /**< @brief Internal reference.
                                         @note Available on ADC1 only.      */
/** @} */

/**
 * @name    Sampling rates
 * @{
 */
#define ADC_SAMPLE_1P5          0   /**< @brief 1.5 cycles sampling time.   */
#define ADC_SAMPLE_7P5          1   /**< @brief 7.5 cycles sampling time.   */
#define ADC_SAMPLE_13P5         2   /**< @brief 13.5 cycles sampling time.  */
#define ADC_SAMPLE_28P5         3   /**< @brief 28.5 cycles sampling time.  */
#define ADC_SAMPLE_41P5         4   /**< @brief 41.5 cycles sampling time.  */
#define ADC_SAMPLE_55P5         5   /**< @brief 55.5 cycles sampling time.  */
#define ADC_SAMPLE_71P5         6   /**< @brief 71.5 cycles sampling time.  */
#define ADC_SAMPLE_239P5        7   /**< @brief 239.5 cycles sampling time. */
/** @} */

/**
 * @name    OVSP register configuration helpers
 * @{
 */
#if (AT32_ADC_SUPPORTS_OVERSAMPLING == TRUE) || defined(__DOXYGEN__)
#define ADC_OVSP_OSRSEL_MASK      (7U << 2U)
#define ADC_OVSP_OSRSEL_2X        (0U << 2U)
#define ADC_OVSP_OSRSEL_4X        (1U << 2U)
#define ADC_OVSP_OSRSEL_8X        (2U << 2U)
#define ADC_OVSP_OSRSEL_16X       (3U << 2U)
#define ADC_OVSP_OSRSEL_32X       (4U << 2U)
#define ADC_OVSP_OSRSEL_64X       (5U << 2U)
#define ADC_OVSP_OSRSEL_128X      (6U << 2U)
#define ADC_OVSP_OSRSEL_256X      (7U << 2U)

#define ADC_OVSP_OSSSEL_MASK      (15 << 5U)
#define ADC_OVSP_OSSSEL_SHIFT(n)  ((n) << 5U)
#endif
/** @} */

/*===========================================================================*/
/* Driver pre-compile time settings.                                         */
/*===========================================================================*/

/**
 * @name    Configuration options
 * @{
 */
/**
 * @brief   ADC common clock divider.
 * @note    This setting is influenced by the VDDA voltage and other
 *          external conditions, please refer to the datasheet for more
 *          info.
 */
#if !defined(AT32_ADC_ADCDIV) || defined(__DOXYGEN__)
#define AT32_ADC_ADCDIV                     8
#endif

/**
 * @brief   ADC1 driver enable switch.
 * @details If set to @p TRUE the support for ADC1 is included.
 * @note    The default is @p TRUE.
 */
#if !defined(AT32_ADC_USE_ADC1) || defined(__DOXYGEN__)
#define AT32_ADC_USE_ADC1                   FALSE
#endif

/**
 * @brief   DMA stream used for ADC1 operations.
 */
#if !defined(AT32_ADC_ADC1_DMA_STREAM) || defined(__DOXYGEN__)
#define AT32_ADC_ADC1_DMA_STREAM            AT32_DMA_STREAM_ID_ANY
#endif

/**
 * @brief   ADC1 DMA priority (0..3|lowest..highest).
 */
#if !defined(AT32_ADC_ADC1_DMA_PRIORITY) || defined(__DOXYGEN__)
#define AT32_ADC_ADC1_DMA_PRIORITY          2
#endif

/**
 * @brief   ADC interrupt priority level setting.
 */
#if !defined(AT32_ADC_IRQ_PRIORITY) || defined(__DOXYGEN__)
#define AT32_ADC_IRQ_PRIORITY               5
#endif

/**
 * @brief   ADC1 DMA interrupt priority level setting.
 */
#if !defined(AT32_ADC_ADC1_DMA_IRQ_PRIORITY) || defined(__DOXYGEN__)
#define AT32_ADC_ADC1_DMA_IRQ_PRIORITY      5
#endif

/** @} */

/*===========================================================================*/
/* Derived constants and error checks.                                       */
/*===========================================================================*/

#if AT32_ADC_USE_ADC1 && !AT32_HAS_ADC1
#error "ADC1 not present in the selected device"
#endif

#if !AT32_ADC_USE_ADC1
#error "ADC driver activated but no ADC peripheral assigned"
#endif

#if !defined(AT32_ADCCLKIN)
#define AT32_ADCCLKIN                      AT32_HCLK
#endif

#if (AT32_ADC_ADCDIV >= 2) || (AT32_ADC_ADCDIV <= 17) 
#define AT32_ADCCLK                        (AT32_ADCCLKIN / AT32_ADC_ADCDIV)
#else
#error "invalid AT32_ADC_ADCDIV value specified"
#endif

#if (AT32_ADCCLK < AT32_ADCCLK_MIN) || (AT32_ADCCLK > AT32_ADCCLK_MAX) || (AT32_ADCCLK > AT32_PCLK2)
#error "AT32_ADCCLK outside acceptable range (AT32_ADCCLK_MIN...AT32_ADCCLK_MAX)"
#endif

#if !defined(AT32_DMA_REQUIRED)
#define AT32_DMA_REQUIRED
#endif

/*===========================================================================*/
/* Driver data structures and types.                                         */
/*===========================================================================*/

/**
 * @brief   ADC sample data type.
 */
typedef uint16_t adcsample_t;

/**
 * @brief   Channels number in a conversion group.
 */
typedef uint16_t adc_channels_num_t;

/**
 * @brief   Type of an ADC error mask.
 */
typedef uint32_t adcerror_t;

/*===========================================================================*/
/* Driver macros.                                                            */
/*===========================================================================*/

/**
 * @brief   Low level fields of the ADC driver structure.
 */
#define adc_lld_driver_fields                                               \
  /* Pointer to the ADCx registers block.*/                                 \
  ADC_TypeDef               *adc;                                           \
  /* Pointer to associated DMA channel.*/                                   \
  const at32_dma_stream_t   *dmastp;                                        \
  /* DMA mode bit mask.*/                                                   \
  uint32_t                  dmamode

/**
 * @brief   Low level fields of the ADC configuration structure.
 */
#define adc_lld_config_fields                                               \
  /* Dummy configuration, it is not needed.*/                               \
  uint32_t                  dummy

/**
 * @brief   Low level fields of the ADC configuration structure.
 */
#define adc_lld_configuration_group_fields                                  \
  /* ADC CTRL1 register initialization data.                                \
     NOTE: All the required bits must be defined into this field except     \
           @p ADC_CTRL1_SQEN that is enforced inside the driver.*/          \
  uint32_t                  ctrl1;                                          \
  /* ADC CTRL2 register initialization data.                                \
     NOTE: All the required bits must be defined into this field except     \
           @p ADC_CTRL2_OCDMAEN, @p ADC_CTRL2_RPEN and @p ADC_CTRL2_ADCEN   \
		   that are enforced inside the driver.*/                           \
  uint32_t                  ctrl2;                                          \
  /* ADC SPT1 register initialization data.                                 \
     NOTE: In this field must be specified the sample times for channels    \
           10...17.*/                                                       \
  uint32_t                  spt1;                                           \
  /* ADC SPT2 register initialization data.                                 \
     NOTE: In this field must be specified the sample times for channels    \
           0...9.*/                                                         \
  uint32_t                  spt2;                                           \
  /* ADC voltage monitoring high boundary register.                         \
     NOTE: This field defines the high boundary of the analog               \
           voltage monitoring.*/                                            \
  uint16_t                  vmhb;                                           \
  /* ADC voltage monitoring low boundary register.                          \
     NOTE: This field defines the low boundary of the analog                \
           voltage monitoring.*/                                            \
  uint16_t                  vmlb;                                           \
  /* ADC OSQ1 register initialization data.                                 \
     NOTE: Conversion group sequence 13...16 + sequence length.*/           \
  uint32_t                  osq1;                                           \
  /* ADC OSQ2 register initialization data.                                 \
     NOTE: Conversion group sequence 7...12.*/                              \
  uint32_t                  osq2;                                           \
  /* ADC OSQ3 register initialization data.                                 \
     NOTE: Conversion group sequence 1...6.*/                               \
  uint32_t                  osq3

/**
 * @name    Sequences building helper macros
 * @{
 */
/**
 * @brief   Number of channels in a conversion sequence.
 */
#define ADC_OSQ1_NUM_CH(n)      (((n) - 1) << 20)

#define ADC_OSQ3_OSN1_N(n)      ((n) << 0)  /**< @brief 1st channel in seq. */
#define ADC_OSQ3_OSN2_N(n)      ((n) << 5)  /**< @brief 2nd channel in seq. */
#define ADC_OSQ3_OSN3_N(n)      ((n) << 10) /**< @brief 3rd channel in seq. */
#define ADC_OSQ3_OSN4_N(n)      ((n) << 15) /**< @brief 4th channel in seq. */
#define ADC_OSQ3_OSN5_N(n)      ((n) << 20) /**< @brief 5th channel in seq. */
#define ADC_OSQ3_OSN6_N(n)      ((n) << 25) /**< @brief 6th channel in seq. */

#define ADC_OSQ2_OSN7_N(n)      ((n) << 0)  /**< @brief 7th channel in seq. */
#define ADC_OSQ2_OSN8_N(n)      ((n) << 5)  /**< @brief 8th channel in seq. */
#define ADC_OSQ2_OSN9_N(n)      ((n) << 10) /**< @brief 9th channel in seq. */
#define ADC_OSQ2_OSN10_N(n)     ((n) << 15) /**< @brief 10th channel in seq.*/
#define ADC_OSQ2_OSN11_N(n)     ((n) << 20) /**< @brief 11th channel in seq.*/
#define ADC_OSQ2_OSN12_N(n)     ((n) << 25) /**< @brief 12th channel in seq.*/

#define ADC_OSQ1_OSN13_N(n)     ((n) << 0)  /**< @brief 13th channel in seq.*/
#define ADC_OSQ1_OSN14_N(n)     ((n) << 5)  /**< @brief 14th channel in seq.*/
#define ADC_OSQ1_OSN15_N(n)     ((n) << 10) /**< @brief 15th channel in seq.*/
#define ADC_OSQ1_OSN16_N(n)     ((n) << 15) /**< @brief 16th channel in seq.*/
/** @} */

/**
 * @name    Sampling rate settings helper macros
 * @{
 */
#define ADC_SPT2_CSPT_AN0(n)    ((n) << 0)  /**< @brief AN0 sampling time.  */
#define ADC_SPT2_CSPT_AN1(n)    ((n) << 3)  /**< @brief AN1 sampling time.  */
#define ADC_SPT2_CSPT_AN2(n)    ((n) << 6)  /**< @brief AN2 sampling time.  */
#define ADC_SPT2_CSPT_AN3(n)    ((n) << 9)  /**< @brief AN3 sampling time.  */
#define ADC_SPT2_CSPT_AN4(n)    ((n) << 12) /**< @brief AN4 sampling time.  */
#define ADC_SPT2_CSPT_AN5(n)    ((n) << 15) /**< @brief AN5 sampling time.  */
#define ADC_SPT2_CSPT_AN6(n)    ((n) << 18) /**< @brief AN6 sampling time.  */
#define ADC_SPT2_CSPT_AN7(n)    ((n) << 21) /**< @brief AN7 sampling time.  */
#define ADC_SPT2_CSPT_AN8(n)    ((n) << 24) /**< @brief AN8 sampling time.  */
#define ADC_SPT2_CSPT_AN9(n)    ((n) << 27) /**< @brief AN9 sampling time.  */

#define ADC_SPT1_CSPT_AN10(n)   ((n) << 0)  /**< @brief AN10 sampling time. */
#define ADC_SPT1_CSPT_AN11(n)   ((n) << 3)  /**< @brief AN11 sampling time. */
#define ADC_SPT1_CSPT_AN12(n)   ((n) << 6)  /**< @brief AN12 sampling time. */
#define ADC_SPT1_CSPT_AN13(n)   ((n) << 9)  /**< @brief AN13 sampling time. */
#define ADC_SPT1_CSPT_AN14(n)   ((n) << 12) /**< @brief AN14 sampling time. */
#define ADC_SPT1_CSPT_AN15(n)   ((n) << 15) /**< @brief AN15 sampling time. */
#define ADC_SPT1_CSPT_SENSOR(n) ((n) << 18) /**< @brief Temperature Sensor
                                                 sampling time.             */
#define ADC_SPT1_CSPT_VREF(n)   ((n) << 21) /**< @brief Voltage Reference
                                                 sampling time.             */
/** @} */

/**
 * @name    Threshold settings helper macros
 * @{
 */
/**
 * @brief   High voltage monitoring boundary.
 */
#define ADC_VMHB(n)             ((n > ADC_VMHB_HB) ? ADC_VMHB_HB : n)
/**
 * @brief   Low voltage monitoring boundary.
 */
#define ADC_VMLB(n)             ((n > ADC_VMLB_LB) ? ADC_VMLB_LB : n)
/** @} */

/*===========================================================================*/
/* External declarations.                                                    */
/*===========================================================================*/

#if AT32_ADC_USE_ADC1 && !defined(__DOXYGEN__)
extern ADCDriver ADCD1;
#endif

#ifdef __cplusplus
extern "C" {
#endif
  void adc_lld_init(void);
  void adc_lld_start(ADCDriver *adcp);
  void adc_lld_stop(ADCDriver *adcp);
  void adc_lld_start_conversion(ADCDriver *adcp);
  void adc_lld_stop_conversion(ADCDriver *adcp);
  void adcAT32EnableITSRVEN(void);
  void adcAT32DisableITSRVEN(void);
#ifdef __cplusplus
}
#endif

#endif /* HAL_USE_ADC */

#endif /* HAL_ADC_LLD_H */

/** @} */
