N32G451CCL7芯片ADC数据采集及片内温度主动获取(PA1、PA3,内部温度)


系统供电3.3V,PA1连接电池电压(最高4.2v锂电池)分压电阻为910K、390K,PA3用于测试OBD接口车载电压(9-26V),分压电阻1M、100K,另外代码中还有芯片内部温度检测,没有需要的,不用细究。同样,前面是头文件代码,后面是应用代码,仅做记录,方便以后移植应用,代码已测试。

#ifndef __ADC_VOLTAGE_H__

#define __ADC_VOLTAGE_H__

// #include "n32wb452.h"

#include "main.h"

// #define HARD_202303 1

#ifdef HARD_202303 //LDO芯片3.3V贴错成了2.8V的.

#define ADC1_TO_V_FACTOR   (2800*(390.0f+910)/4096/390)  //The coefficient:adc value to voltage value,mv.

#else

#define ADC1_TO_V_FACTOR  ((3300*(1300.0f))/(4096*390)) //The coefficient:adc value to voltage value,mv.

// #define ADC_TO_V_FACTOR  ((3300*(390.0f+910))/(4096*390.0f)) //转换系数:ADC采样值转换到电池电压单位mv

#endif

// #define ADC2_TO_V_FACTOR  (3300*(1100.0f)/4096/100)

#define ADC2_TO_V_FACTOR  (3300*(11.0f)/4096)


 

//__packed  

struct __ADC__

{

    u16 voltage;     //转换出来的电压值,单位mv

    u16 value;       //ADC原始数据

    u32 sampleTime;  //采样时间

    float tem;      //温度

    u8  ok;          //采样OK标志

    u8  debug;       //调试模式控制。

};

// float TempCal(uint16_t TempAdVal);

extern struct __ADC__ adc0;   //cpu Temperature

extern struct __ADC__ adc1;   //battery

extern struct __ADC__ adc2;   //obd vol

// extern u16 cpuTemperature; //0.1摄氏度

// extern float cpuTemperatureFloat;

uint16_t ADC_GetData(ADC_Module* ADCx, uint8_t ADC_Channel);

void adcInit(void); //adc 初始化

void adcDeal(void); //ADC数据处理

#endif

#include "adc.h"

// u16 voltage = 0;  //power voltage

// u16 cpuTemperature = 0; //0.1摄氏度

// float cpuTemperatureFloat = 0;

struct __ADC__ adc1;  //PA1 ADC1 battery voltage

struct __ADC__ adc2;  //PA3 ADC2 obd 12V

struct __ADC__ adc0;  //CPU 温度

void adcInit(void)  //adc 初始化 PA1:ADC1 电池电压  PA3:OBD电压

{

    RCC_EnableAPB2PeriphClk(RCC_APB2_PERIPH_GPIOA, ENABLE);

   

    RCC_EnableAHBPeriphClk(RCC_AHB_PERIPH_ADC1,ENABLE);

    ADC_ConfigClk(ADC_CTRL3_CKMOD_AHB, RCC_ADCHCLK_DIV16);

    // RCC_ConfigAdc1mClk(RCC_ADC1MCLK_SRC_HSE, RCC_ADC1MCLK_DIV8);  //selsect HSE as RCC ADC1M CLK Source      

    RCC_ConfigAdc1mClk(RCC_ADC1MCLK_SRC_HSI, RCC_ADC1MCLK_DIV8);  //selsect HSI as RCC ADC1M CLK Source        

    /* Configure PA1 PA2 as analog input -------------------------*/

    GPIO_InitType GPIO_InitStructure;

    GPIO_InitStruct(&GPIO_InitStructure); //初始化结构体

    GPIO_InitStructure.Pin       = GPIO_PIN_1|GPIO_PIN_3;

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;

    GPIO_InitPeripheral(GPIOA, &GPIO_InitStructure);    

    ADC_InitType ADC_InitStructure;

    ADC_InitStructure.MultiChEn      = DISABLE;                   //单通道

    ADC_InitStructure.ContinueConvEn = DISABLE;                   //单次转换

    ADC_InitStructure.ExtTrigSelect  = ADC_EXT_TRIGCONV_NONE;     //软件触发方式

    ADC_InitStructure.DatAlign       = ADC_DAT_ALIGN_R;           //数据右对齐

    ADC_InitStructure.ChsNumber      = 1;//

    ADC_Init(ADC1, &ADC_InitStructure);

    ADC_Enable(ADC1, ENABLE);

    while(ADC_GetFlagStatusNew(ADC1,ADC_FLAG_RDY) == RESET);

    ADC_StartCalibration(ADC1);

    while (ADC_GetCalibrationStatus(ADC1));

    adc1.sampleTime = 5000;      //This code just for testing.

    adc2.sampleTime = 5000;      //This code just for testing.  

    adc1.debug = TRUE;  //debug mode :default disable

    // adc1.debug = FALSE; //debug mode :default disable    

    adc2.debug = TRUE; //debug mode :default disable

    // adc2.debug = FALSE; //debug mode :default disable

   

}

uint16_t ADC_GetData(ADC_Module* ADCx, uint8_t ADC_Channel)

{

    uint16_t dat;

   

    ADC_ConfigRegularChannel(ADCx, ADC_Channel, 1, ADC_SAMP_TIME_239CYCLES5);

    /* Start ADC Software Conversion */

    ADC_EnableSoftwareStartConv(ADCx, ENABLE);

    while(ADC_GetFlagStatus(ADCx, ADC_FLAG_ENDC)==0){}

    ADC_ClearFlag(ADCx, ADC_FLAG_ENDC);

    ADC_ClearFlag(ADCx, ADC_FLAG_STR);

    dat=ADC_GetDat(ADCx);

    return dat;

}

#define Vc0        0    //X

#define Tc1        1.5f

#define AVG_SLOPE  0.0041f

float TempCal(uint16_t TempAdVal)

{

    float Temperate;

    uint16_t  V30 = 0;

    V30 = *(__IO uint32_t*)((uint32_t)0x1FFFF7D0);

    /* Get the temperature inside the chip */

    Temperate=((V30+Vc0-TempAdVal)*3.3/4095)/AVG_SLOPE+30.0f-Tc1;

    return Temperate;

}

float TempGet(void) //CPU内部温度获取。

{

    float Temperate;

    u16 adcValue;    

    ADC_ConfigRegularChannel(ADC1, ADC_CH_16, 1, ADC_SAMP_TIME_239CYCLES5);//ADC_CH_17:测温度,ADC_CH_18:测内部1.2V基准

    ADC_EnableTempSensorVrefint(ENABLE);

    ADC_EnableSoftwareStartConv(ADC1, ENABLE);

    while(ADC_GetFlagStatus(ADC1, ADC_FLAG_ENDC)==0){}

    ADC_ClearFlag(ADC1, ADC_FLAG_ENDC);

    ADC_ClearFlag(ADC1, ADC_FLAG_STR);

    adcValue=ADC_GetDat(ADC1);

    adc0.value = adcValue;

    // Voltage value of temperature sensor

    Temperate = TempCal(adcValue);      

    return Temperate;

}

void adcDeal(void) //ADC数据处理

{  

    static u32 adc1Time = 0;

    static u32 adc2Time = 0;

    static u32 cpuTemTime = 0;

    if (Timer.ms - cpuTemTime>5000)

    {

        cpuTemTime = Timer.ms;

        float Temperate = TempGet();                

        adc0.tem = Temperate;

        // cpuTemperature = Temperate*10;

        // cpuTemperatureFloat = Temperate;

        // printf("CPU TEM:%f.\n",adc0.tem);

    }  

    if(Timer.ms-adc1Time>adc1.sampleTime) //sampling time control.

    {

        adc1Time = Timer.ms;

        adc1.value = ADC_GetData(ADC1,ADC1_Channel_02_PA1);

        adc1.voltage = adc1.value*ADC1_TO_V_FACTOR; //equipment.voltage分辨率 mV

        adc1.ok = TRUE;

        // printf("BAT VOL:%dmv.\r\n",adc1.voltage);

    }

    if(Timer.ms-adc2Time>adc2.sampleTime) //sampling time control.

    {

        adc2Time = Timer.ms;

        adc2.value = adc1.value = ADC_GetData(ADC1,ADC1_Channel_04_PA3);

        adc2.voltage = adc2.value*ADC2_TO_V_FACTOR; //      

        adc2.ok = TRUE;

        // printf("OBD VOL:%dmv.\r\n",adc2.voltage);

    }

   

}

  • 33
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

kevinpan2011

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值