Freescale KV3064PM100SFARM芯片 ADC模块配置 驱动开发

ADC模块驱动开发,这里只教ADC驱动开发技术小白或者第一次做ADC驱动开发的,入不了大佬就法眼。

因为我也是第一次做ADC驱动开发配置,自己摸索,找文档,看各种大佬经验之谈,后面才发现有些内容都是雷同,不过看多了,也能学到很多,比如知道A模块可以软件启动,B模块不支持软件启动,只支持硬件启动,所以下面我在开发ADC模块的时候,我选择A模块软件启动。由于用了半个月的时间才使ADC模块可以工作,这其中的困惑和不明之处我都知道,也明白ADC驱动开发针对不同MCU,有不同的方法去实现,但是万变不离其宗,就是给ADC模块正确的状态,当然这要你的硬件设计是正确的情况下,当然你也不要一味的相信你的硬件设计工程师,我就是相信我的硬件工程师,说他使用了差分模式,结果验证下来证明他没有使用差分模式,好了,废话不多说。

首先你要先看原理图,弄清楚MCU的引脚使用情况,明白ADC引脚的配置,明白输入如何输入的,然后再去查看MCU芯片手册,找到ADC这部分,首先先简单查看一下资源分配情况,以及各个工作状态,接下来你要确定你使用哪个编辑器,我这边使用的是IAR编辑器,至于这些库的调用自己可以上网查看一下如何配置,接下来最重要的是查看ADC软件部分每个function具体是做什么内容的,它的目的,最终是给哪个寄存器或者控制器赋值的,在这部分,只有你真正的弄懂,接下来你才可以知道如何去开发ADC驱动,配置你想要的状态,这也是最耗时间的,当然这对你的帮助也是最大的,因为当你真正懂的了ADC工作的原理后,以后不管哪种MCU,ADC驱动开发都难不倒你。下面是我配置ADC的代码和经验,仅供参考。

ADC模块配置方法:

在IAR编辑器上编写ADC模块,配置KV3064PM100SFARM芯片 ADC模块。第一次配置ADC模块,编写ADC驱动,刚开始的时候一点都不懂,一点思绪都没有,而且对这个芯片也不熟悉,只能硬着头皮去看芯片文档,找到ADC部分查看各种寄存器,控制器的相关信息,然后在网上看各种资料,以及别人配置ADC时是怎么配置的,忽然发现网上几乎所有的内容都是说调用ADC_start函数,但是在IAR上找不到这个函数,最后找到一篇关于K60系列的ADC配置,有一点懂的ADC配置是怎么配置的,但是和KV30芯片还是不一样的,最后在IAR上查看各种库函数,理清各个函数之间的关系,忽然对ADC配置和如何工作有了一个清晰理解,这是对ADC配置最关键的一步,所以如果不懂ADC是如何工作的,最好把它们之间的库函数关系弄懂,弄懂以后,接下来再仔细查看芯片文档,如果ADC要工作的话,各个寄存器,控制器的状态会是什么,比如要差分工作,硬件上是使用ADC0还是ADC1,然后SC1,CFG1,CFG2各个状态是什么样的,这就需要你把他们的状态给出来,根据你的需求来,然后直接赋值,给他们状态,直接上代码:

这是先获取默认配置,然后设置ADC初始状态,使用A模式,不能选用B模式,B模式用于硬件触发,选择软件触发。

static void ADC16_Configuration(void)
{
    adc16_config_t adcUserConfig;  
    /*
     * Initialization ADC for 16bit resolution, DMA mode, normal convert speed, VREFH/L as reference,
     * enable continuous convert mode.
     */
    ADC16_GetDefaultConfig(&adcUserConfig);
    adcUserConfig.resolution                 = kADC16_Resolution16Bit;
    adcUserConfig.enableContinuousConversion = true;
    adcUserConfig.clockSource                = kADC16_ClockSourceAsynchronousClock;
    adcUserConfig.clockDivider               = kADC16_ClockDivider8;//八分频
    adcUserConfig.enableLowPower             = true;
    adcUserConfig.longSampleMode             = kADC16_LongSampleCycle24;
    adcUserConfig.referenceVoltageSource     = kADC16_ReferenceVoltageSourceVref;

    ADC16_Init(DEMO_ADC16_BASEADDR0, &adcUserConfig);
#if defined(FSL_FEATURE_ADC16_HAS_CALIBRATION) && FSL_FEATURE_ADC16_HAS_CALIBRATION
    /* Auto calibration */
    if (kStatus_Success == ADC16_DoAutoCalibration(DEMO_ADC16_BASEADDR0))
    {
        PRINTF("ADC16_DoAutoCalibration() Done.\r\n");
    }
    else
    {
        PRINTF("ADC16_DoAutoCalibration() Failed.\r\n");
    }
    
#endif
    
#if defined(FSL_FEATURE_ADC16_HAS_MUX_SELECT) && FSL_FEATURE_ADC16_HAS_MUX_SELECT
    ADC16_SetChannelMuxMode(DEMO_ADC16_BASEADDR0,kADC16_ChannelMuxA);
#endif 
    /* Enable software trigger.  */
    ADC16_EnableHardwareTrigger(DEMO_ADC16_BASEADDR0, false);
}

根据硬件连接,自己想要的工作状态,配置寄存器和控制器的状态。

    DEMO_ADC16_BASEADDR0->CFG1 = 0x1C;
    DEMO_ADC16_BASEADDR0->CFG2 = 0U;
    DEMO_ADC16_BASEADDR0->SC2 = 0U;
    DEMO_ADC16_BASEADDR0->SC3 = 0x8U;
    DEMO_ADC16_BASEADDR0->SC1[0] = 0x40; 

这个函数是配置每个引脚

ADC16_SetChannelConfig(DEMO_ADC16_BASEADDR0, DEMO_ADC16_CHANNEL_GROUP0, &ADC0_channelConfig[0]);

 使用单端模式,00100引脚

adc16_channel_config_t ADC0_channelConfig[4] = {
  {
    .channelNumber = 4U,
    .enableDifferentialConversion = false,
    .enableInterruptOnConversionCompleted = true,
  }

 配置好后,就可以通过判断Channel的状态来确认ADC是否开始工作了,最后获取Channel转换值,并打印

 if ( 0U != ADC16_GetChannelStatusFlags(DEMO_ADC16_BASEADDR0,DEMO_ADC16_CHANNEL_GROUP0))
      {
        ADC0_SE4=ADC16_GetChannelConversionValue(DEMO_ADC16_BASEADDR0, DEMO_ADC16_CHANNEL_GROUP0);  
        PRINTF("ADC0_SE4 = %u\r\n",ADC0_SE4);
        PRINTF("ADC0_SE4 十六进制 = %0x\r\n",ADC0_SE4);
      }

这就是ADC驱动开发的整体流程,针对其他的MCU也一样的思路,就是如何正确的给各个寄存器或者控制器状态,使其正常工作。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值