/*! ***************************************************************************** @file: AD5940_Temperature.c @author: Neo Xu @brief: AD5940 internal temperature sensor example with sequencer support. ----------------------------------------------------------------------------- Copyright (c) 2017-2019 Analog Devices, Inc. All Rights Reserved. This software is proprietary to Analog Devices, Inc. and its licensors. By using this software you agree to the terms of the associated Analog Devices Software License Agreement. *****************************************************************************/ #include "ad5940.h" #include #include "string.h" /** * This example shows how to configure temperature sensor and using sequencer to take * measurements. There is 'chop' function to remove offset errors from circuit, this * feature is in register REG_AFE_TEMPSENS and is not included in this example. Enable * this function will have better accuracy. */ #define SINC3OSR_SEL ADCSINC3OSR_4 #define SINC2OSR_SEL ADCSINC2OSR_22 #define MEASURE_FREQ 4.0f //4Hz(4SPS) #define FIFO_THRESHOLD 4 //generate FIFO threshold interrupt every 4 data. #define BUFF_SIZE 128 //this buffer will be used by sequence generator and used to store result from AD5940 uint32_t buff[BUFF_SIZE]; uint32_t data_count = 0; //the temperature data count in buffer. /* Initialize AD5940 basic blocks like clock */ static int32_t AD5940PlatformCfg(void){ CLKCfg_Type clk_cfg; FIFOCfg_Type fifo_cfg; SEQCfg_Type seq_cfg; AGPIOCfg_Type gpio_cfg; /* Use hardware reset */ AD5940_HWReset(); /* Platform configuration */ AD5940_Initialize(); /* Step1. Configure clock */ clk_cfg.ADCClkDiv = ADCCLKDIV_1; clk_cfg.ADCCLkSrc = ADCCLKSRC_HFOSC; clk_cfg.SysClkDiv = SYSCLKDIV_1; clk_cfg.SysClkSrc = SYSCLKSRC_HFOSC; clk_cfg.HfOSC32MHzMode = bFALSE; clk_cfg.HFOSCEn = bTRUE; clk_cfg.HFXTALEn = bFALSE; clk_cfg.LFOSCEn = bTRUE; AD5940_CLKCfg(&clk_cfg); /* Step2. Configure FIFO and Sequencer*/ fifo_cfg.FIFOEn = bFALSE; fifo_cfg.FIFOMode = FIFOMODE_FIFO; fifo_cfg.FIFOSize = FIFOSIZE_4KB; /* 4kB for FIFO, The reset 2kB for sequencer */ fifo_cfg.FIFOSrc = FIFOSRC_SINC2NOTCH; fifo_cfg.FIFOThresh = FIFO_THRESHOLD; AD5940_FIFOCfg(&fifo_cfg); /* Disable to reset FIFO. */ fifo_cfg.FIFOEn = bTRUE; AD5940_FIFOCfg(&fifo_cfg); /* Enable FIFO here */ /* Configure sequencer and stop it */ seq_cfg.SeqMemSize = SEQMEMSIZE_2KB; seq_cfg.SeqBreakEn = bFALSE; seq_cfg.SeqIgnoreEn = bFALSE; seq_cfg.SeqCntCRCClr = bTRUE; seq_cfg.SeqEnable = bFALSE; seq_cfg.SeqWrTimer = 0; AD5940_SEQCfg(&seq_cfg); /* Step3. Interrupt controller */ AD5940_INTCCfg(AFEINTC_1, AFEINTSRC_ALLINT, bTRUE); /* Enable all interrupt in Interrupt Controller 1, so we can check INTC flags */ AD5940_INTCCfg(AFEINTC_0, AFEINTSRC_DATAFIFOTHRESH, bTRUE); /* Interrupt Controller 0 will control GP0 to generate interrupt to MCU */ AD5940_INTCClrFlag(AFEINTSRC_ALLINT); /* Step4: Reconfigure GPIO */ gpio_cfg.FuncSet = GP6_SYNC|GP5_SYNC|GP2_TRIG|GP1_SYNC|GP0_INT; gpio_cfg.InputEnSet = AGPIO_Pin2; gpio_cfg.OutputEnSet = AGPIO_Pin0|AGPIO_Pin1|AGPIO_Pin5|AGPIO_Pin6; gpio_cfg.OutVal = 0; gpio_cfg.PullEnSet = AGPIO_Pin2; AD5940_AGPIOCfg(&gpio_cfg); AD5940_SleepKeyCtrlS(SLPKEY_UNLOCK); /* Enable AFE to enter sleep mode. */ return 0; } void _ad5940_analog_init(void){ AFERefCfg_Type aferef_cfg; ADCBaseCfg_Type adc_base; ADCFilterCfg_Type adc_filter; //init ad5940 for temperature measurement. AD5940_AFECtrlS(AFECTRL_ALL, bFALSE); /* Init all to disable state */ aferef_cfg.HpBandgapEn = bTRUE; aferef_cfg.Hp1V1BuffEn = bTRUE; aferef_cfg.Hp1V8BuffEn = bTRUE; /* The High speed buffers are automatically turned off during hibernate */ aferef_cfg.Disc1V1Cap = bFALSE; aferef_cfg.Disc1V8Cap = bFALSE; aferef_cfg.Hp1V8ThemBuff = bFALSE; aferef_cfg.Hp1V8Ilimit = bFALSE; aferef_cfg.Lp1V1BuffEn = bFALSE; aferef_cfg.Lp1V8BuffEn = bFALSE; /* LP reference control - turn off them to save power*/ aferef_cfg.LpBandgapEn = bFALSE; aferef_cfg.LpRefBufEn = bFALSE; aferef_cfg.LpRefBoostEn = bFALSE; AD5940_REFCfgS(&aferef_cfg); /* Initialize ADC basic function */ adc_base.ADCMuxP = ADCMUXP_TEMPP; adc_base.ADCMuxN = ADCMUXN_TEMPN; adc_base.ADCPga = ADCPGA_1P5; AD5940_ADCBaseCfgS(&adc_base); /* Initialize ADC filters ADCRawData-->SINC3-->SINC2+NOTCH */ adc_filter.ADCSinc3Osr = SINC3OSR_SEL; adc_filter.ADCSinc2Osr = SINC2OSR_SEL; adc_filter.ADCAvgNum = ADCAVGNUM_2; /* Don't care about it. Average function is only used for DFT */ adc_filter.ADCRate = ADCRATE_800KHZ; /* If ADC clock is 32MHz, then set it to ADCRATE_1P6MHZ. Default is 16MHz, use ADCRATE_800KHZ. */ adc_filter.BpNotch = bTRUE; /* SINC2+Notch is one block, when bypass notch filter, we can get fresh data from SINC2 filter. */ adc_filter.BpSinc3 = bFALSE; /* We use SINC3 filter. */ adc_filter.Sinc2NotchEnable = bTRUE; /* Enable the SINC2+Notch block. You can also use function AD5940_AFECtrlS */ AD5940_ADCFilterCfgS(&adc_filter); AD5940_AFECtrlS(AFECTRL_TEMPSPWR, bTRUE); /* Turn on temperature sensor power */ } /** * @brief Init everything we need to measure temperature. */ void AD5940_TemperatureInit(void){ uint32_t const *pSeqCmd; uint32_t seq_len; SEQInfo_Type seq_info; WUPTCfg_Type wupt_cfg; ClksCalInfo_Type clks_cal; uint32_t WaitClks; clks_cal.DataType = DATATYPE_SINC2; clks_cal.DataCount = 1; /* Sample one data when wakeup */ clks_cal.ADCSinc2Osr = SINC2OSR_SEL; clks_cal.ADCSinc3Osr = SINC3OSR_SEL; clks_cal.ADCAvgNum = 0; clks_cal.RatioSys2AdcClk = 1; /* Assume ADC clock is same as system clock */ AD5940_ClksCalculate(&clks_cal, &WaitClks); _ad5940_analog_init(); //generate sequence to measure temperature sensor output AD5940_SEQGenInit(buff, BUFF_SIZE); //init sequence generator AD5940_SEQGenCtrl(bTRUE); //from now on, record all register operations rather than write them to AD5940 through SPI. AD5940_SEQGpioCtrlS(AGPIO_Pin1); //pull high AGPIO1 so we know the sequencer is running by observing pin status with oscilloscope etc. AD5940_SEQGenInsert(SEQ_WAIT(16*200)); /* Time for reference settling(if ad5940 is just wake up from hibernate mode) */ AD5940_AFECtrlS(AFECTRL_ADCPWR, bTRUE); /* Turn ON ADC power */ AD5940_SEQGenInsert(SEQ_WAIT(16*50)); /* wait another 50us for ADC to settle. */ AD5940_AFECtrlS(AFECTRL_TEMPCNV|AFECTRL_ADCCNV, bTRUE); /* Start ADC convert */ AD5940_SEQGenInsert(SEQ_WAIT(WaitClks)); AD5940_AFECtrlS(AFECTRL_TEMPCNV|AFECTRL_ADCPWR, bFALSE); /* Stop ADC */ AD5940_SEQGenInsert(SEQ_WAIT(20)); /* Add some delay before put AD5940 to hibernate, needs some clock to move data to FIFO. */ AD5940_SEQGpioCtrlS(0); /* pull low AGPIO so we know end of sequence.*/ AD5940_EnterSleepS();/* Goto hibernate */ AD5940_SEQGenCtrl(bFALSE); /* stop sequence generator */ if(AD5940_SEQGenFetchSeq(&pSeqCmd, &seq_len) != AD5940ERR_OK){ puts("Sequence generator error!"); } seq_info.pSeqCmd = pSeqCmd; seq_info.SeqId = SEQID_0; //use SEQ0 to run this sequence seq_info.SeqLen = seq_len; seq_info.SeqRamAddr = 0; //place this sequence from start of SRAM. seq_info.WriteSRAM = bTRUE;// we need to write this sequence to AD5940 SRAM. AD5940_SEQInfoCfg(&seq_info); //now configure wakeup timer to trigger above sequence periodically to measure temperature data. wupt_cfg.WuptEn = bFALSE; // do not start it right now. wupt_cfg.WuptEndSeq = WUPTENDSEQ_A; wupt_cfg.WuptOrder[0] = SEQID_0; wupt_cfg.SeqxSleepTime[SEQID_0] = 4-1; wupt_cfg.SeqxWakeupTime[SEQID_0] = (uint32_t)(32e3f/MEASURE_FREQ)-4-1; AD5940_WUPTCfg(&wupt_cfg); //enable sequencer AD5940_SEQCtrlS(bTRUE); //now sequencer is ready to be triggered. } void AD5940_TemperatureISR(void){ //process data from AD5940 FIFO. uint32_t FifoCnt, IntcFlag; if(AD5940_WakeUp(10) > 10){ /* Wakeup AFE by read register, read 10 times at most */ printf("Failed to wakeup AD5940!\n"); return; } AD5940_SleepKeyCtrlS(SLPKEY_LOCK); /* We need time to read data from FIFO, so, do not let AD5940 goes to hibernate automatically */ IntcFlag = AD5940_INTCGetFlag(AFEINTC_0); if(IntcFlag&AFEINTSRC_DATAFIFOTHRESH){ FifoCnt = AD5940_FIFOGetCnt(); FifoCnt = FifoCnt>BUFF_SIZE?BUFF_SIZE:FifoCnt; data_count = FifoCnt; AD5940_FIFORd(buff, FifoCnt); AD5940_INTCClrFlag(AFEINTSRC_DATAFIFOTHRESH); AD5940_SleepKeyCtrlS(SLPKEY_UNLOCK); /* Allow AFE to enter sleep mode. AFE will stay at active mode until sequencer trigger sleep */ AD5940_EnterSleepS(); //If MCU is too slow, comment this line, otherwise there is chance the sequencer is running at this point. } } void AD5940_PrintResult(void){ for(int i=0; i