You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

535 lines
14 KiB

// SPDX-FileCopyrightText: 2020 Foundation Devices, Inc. <hello@foundationdevices.com>
// SPDX-License-Identifier: GPL-3.0-or-later
//
// Copyright 2020 - Foundation Devices Inc.
//
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "stm32h7xx_hal.h"
#include "adc.h"
#define MAX_ADC_16BIT 65535
#define REF_VOLTAGE_MV 3000
#define MAX_SAMPLES_CNT 0xFFFF
#define MILLIVOLTS_PER_REVISION 500
#define PWRMON_I_SENSE_RESISTOR 5
/*
* When doing single samples you cannot rely on
* the value being exact, so adding a small offset
* to the computed milli-volts resolves that issue.
*/
#define BOARD_REV_MV_OFFSET 20
/*
* The number of samples to collect for the average
* Current bounces around a lot so take more samples
* and use an average.
* Voltage may not be as bad so take fewer samples
*/
#define MAX_I_SAMPLES 20
#define MAX_V_SAMPLES 4
static ADC_HandleTypeDef hadc3;
static ADC_HandleTypeDef hadc2;
static HAL_StatusTypeDef adc2_init(void)
{
GPIO_InitTypeDef GPIO_InitStruct = {0};
HAL_StatusTypeDef rc;
hadc2.Instance = ADC2;
rc = HAL_ADC_DeInit(&hadc2);
if (rc != HAL_OK) {
printf("Failed to DeInit ADC2\n");
return rc;
}
__HAL_RCC_ADC12_CLK_ENABLE();
/* ADC Periph interface clock configuration */
__HAL_RCC_ADC_CONFIG(RCC_ADCCLKSOURCE_CLKP);
/**ADC2 GPIO Configuration
PC0 ------> ADC2_INP10 - NOISE_OUT2
PC1 ------> ADC2_INP11 - NOISE_OUT1
PC4 ------> ADC2_INP4 - PWRMON_V
PC5 ------> ADC2_INP8 - PWRMON_I
*/
GPIO_InitStruct.Pin = GPIO_PIN_0 | GPIO_PIN_1 | GPIO_PIN_4| GPIO_PIN_5; //
GPIO_InitStruct.Mode = GPIO_MODE_ANALOG;
GPIO_InitStruct.Pull = GPIO_NOPULL;
HAL_GPIO_Init(GPIOC, &GPIO_InitStruct);
/** Common config
*/
// The ClockPrescaler can only be modified if ALL ADC instances are disabled!!!
hadc2.Init.ClockPrescaler = ADC_CLOCK_ASYNC_DIV4; // ADC_CLOCK_ASYNC_DIV1
hadc2.Init.Resolution = ADC_RESOLUTION_16B;
hadc2.Init.ScanConvMode = ADC_SCAN_DISABLE; // Needs to be ENABLE it processing more than 1 channel
hadc2.Init.EOCSelection = ADC_EOC_SINGLE_CONV; // ADC_EOC_SEQ_CONV If doing more than one channel
hadc2.Init.LowPowerAutoWait = ENABLE; // Says to use this with Polling was DISABLE;
hadc2.Init.ContinuousConvMode = ENABLE;
hadc2.Init.NbrOfConversion = 1; // Number of ranks to be converted within regular group sequencer
hadc2.Init.DiscontinuousConvMode = DISABLE; // Cannot be used if continuous mode is enabled.
hadc2.Init.NbrOfDiscConversion = 1;
hadc2.Init.ExternalTrigConv = ADC_SOFTWARE_START; // NOTE: This is common to ALL ADC instances..
hadc2.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE;
hadc2.Init.ConversionDataManagement = ADC_CONVERSIONDATA_DR; // Can specify DMA for management of converted data
hadc2.Init.Overrun = ADC_OVR_DATA_OVERWRITTEN; // ADC_OVR_DATA_PRESERVED
hadc2.Init.LeftBitShift = ADC_LEFTBITSHIFT_NONE;
/*
* Perform oversampling to read multiple samples
* and compute the average in HW.
*/
hadc3.Init.OversamplingMode = ENABLE;
hadc3.Init.Oversampling.Ratio = 0x20; /* Bit for 32x oversampling */
hadc3.Init.Oversampling.RightBitShift = ADC_RIGHTBITSHIFT_5;
hadc3.Init.Oversampling.TriggeredMode = ADC_TRIGGEREDMODE_SINGLE_TRIGGER;
hadc3.Init.Oversampling.OversamplingStopReset = ADC_REGOVERSAMPLING_CONTINUED_MODE;
rc = HAL_ADC_Init(&hadc2);
if (rc != HAL_OK)
{
printf("Failed to init ADC2\n");;
return rc;
}
/* Run the ADC calibration in single-ended mode */
rc = HAL_ADCEx_Calibration_Start(&hadc2, ADC_CALIB_OFFSET, ADC_SINGLE_ENDED);
if (rc != HAL_OK)
{
printf("ADC3 calibration failed\n");
return rc;
}
return HAL_OK;
}
HAL_StatusTypeDef adc3_init(void)
{
HAL_StatusTypeDef rc;
hadc3.Instance = ADC3;
rc = HAL_ADC_DeInit(&hadc3);
if (rc != HAL_OK)
{
printf("Failed to deinit ADC3\n");
return rc;
}
__HAL_RCC_ADC3_CLK_ENABLE();
/**ADC3 GPIO Configuration
PC2_C ------> ADC3_INP0 - ALS_OUT
PC3_C ------> ADC3_INP1 - BDREV
*/
HAL_SYSCFG_AnalogSwitchConfig(SYSCFG_SWITCH_PC2 | SYSCFG_SWITCH_PC3,
SYSCFG_SWITCH_PC2_OPEN | SYSCFG_SWITCH_PC3_OPEN);
hadc3.Instance = ADC3;
hadc3.Init.ClockPrescaler = ADC_CLOCK_ASYNC_DIV2; // 4
hadc3.Init.Resolution = ADC_RESOLUTION_16B;
hadc3.Init.ScanConvMode = ADC_SCAN_DISABLE;
hadc3.Init.EOCSelection = ADC_EOC_SINGLE_CONV;
hadc3.Init.LowPowerAutoWait = DISABLE;
hadc3.Init.ContinuousConvMode = ENABLE; // DIS
hadc3.Init.NbrOfConversion = 1;
hadc3.Init.DiscontinuousConvMode = DISABLE;
hadc3.Init.ExternalTrigConv = ADC_SOFTWARE_START;
hadc3.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE;
hadc3.Init.ConversionDataManagement = ADC_CONVERSIONDATA_DR;
hadc3.Init.Overrun = ADC_OVR_DATA_OVERWRITTEN;
hadc3.Init.LeftBitShift = ADC_LEFTBITSHIFT_NONE;
/*
* Perform oversampling to read multiple samples
* and compute the average in HW.
*/
hadc3.Init.OversamplingMode = ENABLE;
hadc3.Init.Oversampling.Ratio = 0x20; /* Bit for 32x oversampling */
hadc3.Init.Oversampling.RightBitShift = ADC_RIGHTBITSHIFT_5;
hadc3.Init.Oversampling.TriggeredMode = ADC_TRIGGEREDMODE_SINGLE_TRIGGER;
hadc3.Init.Oversampling.OversamplingStopReset = ADC_REGOVERSAMPLING_CONTINUED_MODE;
rc = HAL_ADC_Init(&hadc3);
if (rc != HAL_OK)
{
printf("ADC3 init failed\n");
return rc;
}
return HAL_OK;
}
void adc_enable_noise(void)
{
GPIO_InitTypeDef GPIO_InitStruct = {0};
/*
* PD8 Amp2_enable
* PD9 Amp1_enable
* PD10 Noise Bias enable
*/
GPIO_InitStruct.Pin = GPIO_PIN_8 | GPIO_PIN_9 | GPIO_PIN_10;
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
HAL_GPIO_Init(GPIOD, &GPIO_InitStruct);
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_8, 1);
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_9, 1);
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_10, 1);
}
void adc_disable_noise(void)
{
/*
* PD8 Amp2_enable
* PD9 Amp1_enable
* PD10 Noise Bias enable
*/
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_8, 0);
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_9, 0);
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_10, 0);
}
/*
* read_noise_inputs() - Reads the two noise output channels and returns
* the count values read.
*/
int adc_read_noise_inputs(
uint32_t *noise1,
uint32_t *noise2
)
{
HAL_StatusTypeDef rc;
ADC_ChannelConfTypeDef sConfig = {0};
/* Configure Noiseout 1 input Channel */
sConfig.Channel = ADC_CHANNEL_11; // Noise 1 channel 11 PC1 INP11
sConfig.Rank = ADC_REGULAR_RANK_1;
sConfig.SamplingTime = ADC_SAMPLETIME_8CYCLES_5;
sConfig.SingleDiff = ADC_SINGLE_ENDED;
sConfig.OffsetNumber = ADC_OFFSET_NONE;
sConfig.Offset = 0;
sConfig.OffsetRightShift = DISABLE; /* No Right Offset Shift */
sConfig.OffsetSignedSaturation = DISABLE; /* No Signed Saturation */
rc = HAL_ADC_ConfigChannel(&hadc2, &sConfig);
if (rc != HAL_OK)
{
printf("Failed to config ADC2 channel 8\n");;
return -1;
}
/* Start processing for current (I) values */
rc = HAL_ADC_Start(&hadc2);
if (rc != HAL_OK)
{
printf("ADC2 start failed\n");
return -1;
}
rc = HAL_ADC_PollForConversion(&hadc2, HAL_MAX_DELAY);
if (rc != HAL_OK)
{
printf("ADC2 poll for conversion failed\n");
return -1;
}
*noise1 = HAL_ADC_GetValue(&hadc2);
rc = HAL_ADC_Stop(&hadc2);
if (rc != HAL_OK)
{
printf("ADC2 start failed\n");
return -1;
}
/* Configure Noiseout 2 input Channel */
/* Noise 2 channel 10 PC0 INP10 */
sConfig.Channel = ADC_CHANNEL_10;
sConfig.Rank = ADC_REGULAR_RANK_1;
sConfig.SamplingTime = ADC_SAMPLETIME_8CYCLES_5;
rc = HAL_ADC_ConfigChannel(&hadc2, &sConfig);
if (rc != HAL_OK)
{
printf("Failed to config ADC2 channel 4\n");;
return -1;
}
/* Now sample for Noise output 2 */
rc = HAL_ADC_Start(&hadc2);
if (rc != HAL_OK)
{
printf("ADC2 start failed\n");
return -1;
}
rc = HAL_ADC_PollForConversion(&hadc2, HAL_MAX_DELAY);
if (rc != HAL_OK)
{
printf("ADC2 poll for conversion failed\n");
return -1;
}
*noise2 = HAL_ADC_GetValue(&hadc2);
rc = HAL_ADC_Stop(&hadc2);
if (rc != HAL_OK)
{
printf("ADC2 start failed\n");
return -1;
}
return 0;
}
/*
* adc_read_powermon() Reads the power monitor current and voltage channels
*/
int adc_read_powermon(
uint16_t *current,
uint16_t *voltage
)
{
HAL_StatusTypeDef rc;
ADC_ChannelConfTypeDef sConfig = {0};
uint32_t adc_value_i;
uint32_t adc_value_v;
/* Configure Regular Power Monitor Current Channel */
sConfig.Channel = ADC_CHANNEL_8; // PWRMON_I channel PC5 INP8
sConfig.Rank = ADC_REGULAR_RANK_1;
sConfig.SamplingTime = ADC_SAMPLETIME_8CYCLES_5;
sConfig.SingleDiff = ADC_SINGLE_ENDED;
sConfig.OffsetNumber = ADC_OFFSET_NONE;
sConfig.Offset = 0;
sConfig.OffsetRightShift = DISABLE; /* No Right Offset Shift */
sConfig.OffsetSignedSaturation = DISABLE; /* No Signed Saturation */
rc = HAL_ADC_ConfigChannel(&hadc2, &sConfig);
if (rc != HAL_OK)
{
printf("Failed to config ADC2 channel 8\n");;
return -1;
}
// Start processing for current (I) values
rc = HAL_ADC_Start(&hadc2);
if (rc != HAL_OK)
{
printf("ADC2 start failed\n");
return -1;
}
rc = HAL_ADC_PollForConversion(&hadc2, HAL_MAX_DELAY);
if (rc != HAL_OK)
{
printf("ADC2 poll for conversion failed\n");
return -1;
}
adc_value_i = HAL_ADC_GetValue(&hadc2);
/*
* Current is I sense voltage divided by
* the sense resistor value which is 5 ohms
*/
*current = ((adc_value_i * REF_VOLTAGE_MV) / MAX_SAMPLES_CNT) / PWRMON_I_SENSE_RESISTOR;
rc = HAL_ADC_Stop(&hadc2);
if (rc != HAL_OK)
{
printf("ADC2 start failed\n");
return -1;
}
/* Switch to the voltage channel */
/* PWRMON_V channel PC4 INP4 */
sConfig.Channel = ADC_CHANNEL_4;
sConfig.Rank = ADC_REGULAR_RANK_1;
sConfig.SamplingTime = ADC_SAMPLETIME_8CYCLES_5;
rc = HAL_ADC_ConfigChannel(&hadc2, &sConfig);
if (rc != HAL_OK)
{
printf("Failed to config ADC2 channel 4\n");;
return -1;
}
/* Now sample for voltage (V) */
rc = HAL_ADC_Start(&hadc2);
if (rc != HAL_OK)
{
printf("ADC2 start failed\n");
return -1;
}
rc = HAL_ADC_PollForConversion(&hadc2, HAL_MAX_DELAY);
if (rc != HAL_OK)
{
printf("ADC2 poll for conversion failed\n");
return -1;
}
adc_value_v = HAL_ADC_GetValue(&hadc2);
*voltage = (adc_value_v * REF_VOLTAGE_MV) / MAX_SAMPLES_CNT;
rc = HAL_ADC_Stop(&hadc2);
if (rc != HAL_OK)
{
printf("ADC2 start failed\n");
return -1;
}
return 0;
}
/*
* adc_read_als() - Reads the ambient light sensor channel
* and returns a numeric value based on the milli-volts
* read divided by the number of milli-volts per revision.
*/
int adc_read_als(
uint16_t *als
)
{
HAL_StatusTypeDef rc;
ADC_ChannelConfTypeDef sConfig = {0};
uint32_t adc_value;
uint16_t millivolts;
*als = 0;
/** Configure Regular Channel
*/
sConfig.Channel = ADC_CHANNEL_0;
sConfig.Rank = ADC_REGULAR_RANK_1;
sConfig.SamplingTime = ADC_SAMPLETIME_8CYCLES_5;
sConfig.SingleDiff = ADC_SINGLE_ENDED;
sConfig.OffsetNumber = ADC_OFFSET_NONE;
sConfig.Offset = 0;
rc = HAL_ADC_ConfigChannel(&hadc3, &sConfig);
if (rc != HAL_OK)
{
printf("Failed to config ADC3 channel\n");
return -1;
}
/* Run the ADC calibration in single-ended mode */
rc = HAL_ADCEx_Calibration_Start(&hadc3, ADC_CALIB_OFFSET, ADC_SINGLE_ENDED);
if (rc != HAL_OK)
{
printf("ADC3 calibration failed\n");
return -1;
}
rc = HAL_ADC_Start(&hadc3);
if (rc != HAL_OK)
{
printf("ADC3 start failed\n");
return -1;
}
rc = HAL_ADC_PollForConversion(&hadc3, HAL_MAX_DELAY);
if (rc != HAL_OK)
{
printf("ADC3 poll for conversion failed\n");
return -1;
}
adc_value = HAL_ADC_GetValue(&hadc3);
HAL_ADC_Stop(&hadc3);
millivolts = (((adc_value) * REF_VOLTAGE_MV) / MAX_SAMPLES_CNT);
*als = millivolts; /* Upper-level code will scale this as needed */
return 0;
}
/*
* adc_read_boardrev() - Reads the board revision channel
* and returns a numeric value based on the milli-volts
* read divided by the number of milli-volts per revision.
*/
int adc_read_boardrev(
uint16_t *board_rev
)
{
HAL_StatusTypeDef rc;
ADC_ChannelConfTypeDef sConfig = {0};
uint32_t adc_value;
uint16_t millivolts;
*board_rev = 0;
/** Configure Regular Channel
*/
sConfig.Channel = ADC_CHANNEL_1;
sConfig.Rank = ADC_REGULAR_RANK_1;
sConfig.SamplingTime = ADC_SAMPLETIME_8CYCLES_5;
sConfig.SingleDiff = ADC_SINGLE_ENDED;
sConfig.OffsetNumber = ADC_OFFSET_NONE;
sConfig.Offset = 0;
rc = HAL_ADC_ConfigChannel(&hadc3, &sConfig);
if (rc != HAL_OK)
{
printf("Failed to config ADC3 channel\n");
return -1;
}
/* Run the ADC calibration in single-ended mode */
rc = HAL_ADCEx_Calibration_Start(&hadc3, ADC_CALIB_OFFSET, ADC_SINGLE_ENDED);
if (rc != HAL_OK)
{
printf("ADC3 calibration failed\n");
return -1;
}
rc = HAL_ADC_Start(&hadc3);
if (rc != HAL_OK)
{
printf("ADC3 start failed\n");
return -1;
}
rc = HAL_ADC_PollForConversion(&hadc3, HAL_MAX_DELAY);
if (rc != HAL_OK)
{
printf("ADC3 poll for conversion failed\n");
return -1;
}
adc_value = HAL_ADC_GetValue(&hadc3);
HAL_ADC_Stop(&hadc3);
millivolts = (((adc_value) * REF_VOLTAGE_MV) / MAX_SAMPLES_CNT);
printf("[%s] millivolts: %u\n", __func__, millivolts);
*board_rev = millivolts / MILLIVOLTS_PER_REVISION;
return 0;
}
int adc_init(void)
{
HAL_StatusTypeDef rc;
rc = adc2_init();
if (rc != HAL_OK)
return -1;
rc = adc3_init();
if (rc != HAL_OK)
return -1;
return 0;
}