[技术问答] M051单片机内部有几路adc,能实现同步采集吗

[复制链接]
6015|36
 楼主| energyplants 发表于 2016-7-5 10:44 | 显示全部楼层 |阅读模式
想用两路adc,实现同步采集,不知道M051能不能够实现这个功能
gejigeji521 发表于 2016-7-29 21:36 | 显示全部楼层
可以,有很多路。好像8路。
给你个例程
  1. /****************************************************************************
  2. * [url=home.php?mod=space&uid=288409]@file[/url]     main.c
  3. * [url=home.php?mod=space&uid=895143]@version[/url]  V3.0
  4. * $Revision: 5 $
  5. * $Date: 14/01/28 11:44a $
  6. * [url=home.php?mod=space&uid=247401]@brief[/url]    M051 Series ADC Interface Controller Driver Sample Code
  7. *
  8. * @note
  9. * Copyright (C) 2014 Nuvoton Technology Corp. All rights reserved.
  10. *
  11. ******************************************************************************/
  12. #include <stdio.h>
  13. #include "M051Series.h"

  14. #define PLL_CLOCK       50000000


  15. /*---------------------------------------------------------------------------------------------------------*/
  16. /* Define Function Prototypes                                                                              */
  17. /*---------------------------------------------------------------------------------------------------------*/
  18. void SYS_Init(void);
  19. void UART0_Init(void);
  20. void AdcContScanModeTest(void);


  21. void SYS_Init(void)
  22. {
  23.     /*---------------------------------------------------------------------------------------------------------*/
  24.     /* Init System Clock                                                                                       */
  25.     /*---------------------------------------------------------------------------------------------------------*/

  26.     /* Enable Internal RC 22.1184MHz clock */
  27.     CLK_EnableXtalRC(CLK_PWRCON_OSC22M_EN_Msk);

  28.     /* Waiting for Internal RC clock ready */
  29.     CLK_WaitClockReady(CLK_CLKSTATUS_OSC22M_STB_Msk);

  30.     /* Switch HCLK clock source to Internal RC and HCLK source divide 1 */
  31.     CLK_SetHCLK(CLK_CLKSEL0_HCLK_S_HIRC, CLK_CLKDIV_HCLK(1));

  32.     /* Enable external XTAL 12MHz clock */
  33.     CLK_EnableXtalRC(CLK_PWRCON_XTL12M_EN_Msk);

  34.     /* Waiting for external XTAL clock ready */
  35.     CLK_WaitClockReady(CLK_CLKSTATUS_XTL12M_STB_Msk);

  36.     /* Set core clock as PLL_CLOCK from PLL */
  37.     CLK_SetCoreClock(PLL_CLOCK);

  38.     /* Enable UART module clock */
  39.     CLK_EnableModuleClock(UART0_MODULE);

  40.     /* Enable ADC module clock */
  41.     CLK_EnableModuleClock(ADC_MODULE);

  42.     /* Select UART module clock source */
  43.     CLK_SetModuleClock(UART0_MODULE, CLK_CLKSEL1_UART_S_HXT, CLK_CLKDIV_UART(1));

  44.     /* ADC clock source is 22.1184MHz, set divider to 7, ADC clock is 22.1184/7 MHz */
  45.     CLK_SetModuleClock(ADC_MODULE, CLK_CLKSEL1_ADC_S_HIRC, CLK_CLKDIV_ADC(7));

  46.     /*---------------------------------------------------------------------------------------------------------*/
  47.     /* Init I/O Multi-function                                                                                 */
  48.     /*---------------------------------------------------------------------------------------------------------*/

  49.     /* Set P3 multi-function pins for UART0 RXD and TXD */
  50.     SYS->P3_MFP &= ~(SYS_MFP_P30_Msk | SYS_MFP_P31_Msk);
  51.     SYS->P3_MFP |= SYS_MFP_P30_RXD0 | SYS_MFP_P31_TXD0;

  52.     /* Disable the P1.0 - P1.3 digital input path to avoid the leakage current */
  53.     GPIO_DISABLE_DIGITAL_PATH(P1, 0xF);

  54.     /* Configure the P1.0 - P1.3 ADC analog input pins */
  55.     SYS->P1_MFP &= ~(SYS_MFP_P10_Msk | SYS_MFP_P11_Msk | SYS_MFP_P12_Msk | SYS_MFP_P13_Msk);
  56.     SYS->P1_MFP |= SYS_MFP_P10_AIN0 | SYS_MFP_P11_AIN1 | SYS_MFP_P12_AIN2 | SYS_MFP_P13_AIN3 ;

  57. }

  58. /*---------------------------------------------------------------------------------------------------------*/
  59. /* Init UART                                                                                               */
  60. /*---------------------------------------------------------------------------------------------------------*/
  61. void UART0_Init()
  62. {
  63.     /* Reset IP */
  64.     SYS_ResetModule(UART0_RST);

  65.     /* Configure UART0 and set UART0 Baudrate */
  66.     UART_Open(UART0, 115200);
  67. }

  68. /*---------------------------------------------------------------------------------------------------------*/
  69. /* Function: ADC_GetConversionRate                                                                         */
  70. /*                                                                                                         */
  71. /* Parameters:                                                                                             */
  72. /*   None.                                                                                                 */
  73. /*                                                                                                         */
  74. /* Returns:                                                                                                */
  75. /*      Return the A/D conversion rate (sample/second)                                                     */
  76. /*                                                                                                         */
  77. /* Description:                                                                                            */
  78. /*   The conversion rate depends on the clock source of ADC clock.                                         */
  79. /*   It only needs 21 ADC clocks to complete an A/D conversion.                                            */
  80. /*---------------------------------------------------------------------------------------------------------*/
  81. static __INLINE uint32_t ADC_GetConversionRate()
  82. {
  83.     uint32_t u32AdcClkSrcSel;
  84.     uint32_t u32ClkTbl[4] = {__HXT, 0, 0, __HIRC};

  85.     /* Set the PLL clock frequency */
  86.     u32ClkTbl[1] = PllClock;
  87.     /* Set the system core clock frequency */
  88.     u32ClkTbl[2] = SystemCoreClock;
  89.     /* Get the clock source setting */
  90.     u32AdcClkSrcSel = ((CLK->CLKSEL1 & CLK_CLKSEL1_ADC_S_Msk) >> CLK_CLKSEL1_ADC_S_Pos);
  91.     /* Return the ADC conversion rate */
  92.     return ((u32ClkTbl[u32AdcClkSrcSel]) / (((CLK->CLKDIV & CLK_CLKDIV_ADC_N_Msk) >> CLK_CLKDIV_ADC_N_Pos) + 1) / 21);
  93. }

  94. /*---------------------------------------------------------------------------------------------------------*/
  95. /* Function: AdcContScanModeTest                                                                           */
  96. /*                                                                                                         */
  97. /* Parameters:                                                                                             */
  98. /*   None.                                                                                                 */
  99. /*                                                                                                         */
  100. /* Returns:                                                                                                */
  101. /*   None.                                                                                                 */
  102. /*                                                                                                         */
  103. /* Description:                                                                                            */
  104. /*   ADC continuous scan mode test.                                                                        */
  105. /*---------------------------------------------------------------------------------------------------------*/
  106. void AdcContScanModeTest()
  107. {
  108.     uint8_t  u8Option;
  109.     uint32_t u32ChannelCount;
  110.     int32_t  i32ConversionData;

  111.     printf("\n\nConversion rate: %d samples/second\n", ADC_GetConversionRate());
  112.     printf("\n");
  113.     printf("+----------------------------------------------------------------------+\n");
  114.     printf("|                 ADC continuous scan mode sample code                 |\n");
  115.     printf("+----------------------------------------------------------------------+\n");

  116.     printf("\nIn this test, software will get 2 cycles of conversion result from the specified channels.\n");

  117.     while(1)
  118.     {
  119.         printf("\n\nSelect input mode:\n");
  120.         printf("  [1] Single end input (channel 0, 1, 2 and 3)\n");
  121.         printf("  [2] Differential input (input channel pair 0 and 1)\n");
  122.         printf("  Other keys: exit continuous scan mode test\n");
  123.         u8Option = getchar();
  124.         if(u8Option == '1')
  125.         {
  126.             /* Set the ADC operation mode as continuous scan, input mode as single-end and
  127.                  enable the analog input channel 0, 1, 2 and 3 */
  128.             ADC_Open(ADC, ADC_ADCR_DIFFEN_SINGLE_END, ADC_ADCR_ADMD_CONTINUOUS, 0xF);

  129.             /* Power on ADC module */
  130.             ADC_POWER_ON(ADC);

  131.             /* clear the A/D interrupt flag for safe */
  132.             ADC_CLR_INT_FLAG(ADC, ADC_ADF_INT);

  133.             /* start A/D conversion */
  134.             ADC_START_CONV(ADC);

  135.             /* Wait conversion done */
  136.             while(!ADC_GET_INT_FLAG(ADC, ADC_ADF_INT));

  137.             /* clear the A/D interrupt flag for safe */
  138.             ADC_CLR_INT_FLAG(ADC, ADC_ADF_INT);

  139.             for(u32ChannelCount = 0; u32ChannelCount < 4; u32ChannelCount++)
  140.             {
  141.                 i32ConversionData = ADC_GET_CONVERSION_DATA(ADC, u32ChannelCount);
  142.                 printf("Conversion result of channel %d: 0x%X (%d)\n", u32ChannelCount, i32ConversionData, i32ConversionData);
  143.             }

  144.             /* Wait conversion done */
  145.             while(!ADC_GET_INT_FLAG(ADC, ADC_ADF_INT));

  146.             /* Stop A/D conversion */
  147.             ADC_STOP_CONV(ADC);

  148.             for(u32ChannelCount = 0; u32ChannelCount < 4; u32ChannelCount++)
  149.             {
  150.                 i32ConversionData = ADC_GET_CONVERSION_DATA(ADC, u32ChannelCount);
  151.                 printf("Conversion result of channel %d: 0x%X (%d)\n", u32ChannelCount, i32ConversionData, i32ConversionData);
  152.             }

  153.             /* clear the A/D interrupt flag for safe */
  154.             ADC_CLR_INT_FLAG(ADC, ADC_ADF_INT);

  155.         }
  156.         else if(u8Option == '2')
  157.         {
  158.             /* Set the ADC operation mode as continuous scan, input mode as differential and
  159.                enable analog input channel 0 and 2 */
  160.             ADC_Open(ADC, ADC_ADCR_DIFFEN_DIFFERENTIAL, ADC_ADCR_ADMD_CONTINUOUS, 0x5);

  161.             /* Power on ADC module */
  162.             ADC_POWER_ON(ADC);

  163.             /* clear the A/D interrupt flag for safe */
  164.             ADC_CLR_INT_FLAG(ADC, ADC_ADF_INT);

  165.             /* start A/D conversion */
  166.             ADC_START_CONV(ADC);

  167.             /* Wait conversion done */
  168.             while(!ADC_GET_INT_FLAG(ADC, ADC_ADF_INT));

  169.             /* clear the A/D interrupt flag for safe */
  170.             ADC_CLR_INT_FLAG(ADC, ADC_ADF_INT);

  171.             for(u32ChannelCount = 0; u32ChannelCount < 2; u32ChannelCount++)
  172.             {
  173.                 i32ConversionData = ADC_GET_CONVERSION_DATA(ADC, u32ChannelCount * 2);
  174.                 printf("Conversion result of differential input pair %d: 0x%X (%d)\n", u32ChannelCount, i32ConversionData, i32ConversionData);
  175.             }

  176.             /* Wait conversion done */
  177.             while(!ADC_GET_INT_FLAG(ADC, ADC_ADF_INT));

  178.             /* Stop A/D conversion */
  179.             ADC_STOP_CONV(ADC);

  180.             for(u32ChannelCount = 0; u32ChannelCount < 2; u32ChannelCount++)
  181.             {
  182.                 i32ConversionData = ADC_GET_CONVERSION_DATA(ADC, u32ChannelCount * 2);
  183.                 printf("Conversion result of differential input pair %d: 0x%X (%d)\n", u32ChannelCount, i32ConversionData, i32ConversionData);
  184.             }

  185.             /* clear the A/D interrupt flag for safe */
  186.             ADC_CLR_INT_FLAG(ADC, ADC_ADF_INT);

  187.         }
  188.         else
  189.             return ;

  190.     }
  191. }

  192. /*---------------------------------------------------------------------------------------------------------*/
  193. /* MAIN function                                                                                           */
  194. /*---------------------------------------------------------------------------------------------------------*/

  195. main(void)
  196. {

  197.     /* Unlock protected registers */
  198.     SYS_UnlockReg();

  199.     /* Init System, IP clock and multi-function I/O */
  200.     SYS_Init();

  201.     /* Lock protected registers */
  202.     SYS_LockReg();

  203.     /* Init UART0 for printf */
  204.     UART0_Init();

  205.     /*---------------------------------------------------------------------------------------------------------*/
  206.     /* SAMPLE CODE                                                                                             */
  207.     /*---------------------------------------------------------------------------------------------------------*/

  208.     printf("\nSystem clock rate: %d Hz", SystemCoreClock);

  209.     /* Continuous scan mode test */
  210.     AdcContScanModeTest();

  211.     /* Disable ADC module */
  212.     ADC_Close(ADC);

  213.     /* Disable ADC IP clock */
  214.     CLK_DisableModuleClock(ADC_MODULE);

  215.     /* Disable External Interrupt */
  216.     NVIC_DisableIRQ(ADC_IRQn);

  217.     printf("\nExit ADC sample code\n");

  218.     while(1);

  219. }


gejigeji521 发表于 2016-7-29 21:40 | 显示全部楼层
NuMicro M051™系列包含 一个8通道12位的逐次逼近式 模拟 – 数字转换器 (SAR A/D转换器). A/D 转换
器支持 四种工作模式: 单次转换模式、突发转换模式、单周期扫描模式和连续扫描模式.开始A/D 转换可
软件设定和外部STADC/P3.2引脚启动。
gejigeji521 发表于 2016-7-29 21:41 | 显示全部楼层
特征
y 模拟输入电压范围: 0~0~AVDD(最大5.0V).
y 12位分辨率和10位精确度保证.
y 多达 8 路单端模拟输入通道或4路差分模拟输入通道.
y 最大 ADC 时钟频率 16MHz.
y 高达600k SPS 转换速率.
y 四种操作模式
- 单次转换模式:A/D转换在指定通道完成一次转换.
- 单周期扫描模式:A/D 转换在所有指定通道完成一个周期(从低序号通道到高序号通道)转换.
- 连续扫描模式: A/D 转换器连续执行单周期扫描模式直到软件停止A/D转换.
- 突发模式: A/D 转换 采样和转换在指定单个通道进行,并将结果顺序地存入FIFO.
y A/D转换开始条件
- 软件向ADST 位写1
- 外部引脚STADC触发
y 每通道转换结果存储在相应数据寄存器内,并带有有效或超出限度的标志.
y 转换结果可和指定的值相比较, 当转换值和设定值相匹配时,用户可设定是否产生中断请求.
y 通道 7 支持 2 输入源:外部模拟电压, 内部带隙电压.
y 支持自身校正功能以减少转换的误差.

戈卫东 发表于 2016-7-29 22:04 | 显示全部楼层
不能同步采集。
chengming334 发表于 2024-5-13 13:55 | 显示全部楼层

怎么采集的呢?
burgessmaggie 发表于 2024-6-4 16:12 | 显示全部楼层
大多数M051系列单片机会内置多路ADC通道,常见的是8路、12路
maudlu 发表于 2024-6-5 16:25 | 显示全部楼层
提供灵活的触发机制,如定时器触发、外部事件触发等,用于同步ADC的启动。
maudlu 发表于 2024-6-7 19:28 | 显示全部楼层
基于8051内核的单片机如果具有多通道ADC,一般都可以通过编程来实现对多个通道的同步采集。
bartonalfred 发表于 2024-6-8 16:27 | 显示全部楼层
12位和10位的ADC分别提供了不同级别的分辨率,满足从低端到高端不同精度需求的应用场景。
qiufengsd 发表于 2024-6-8 20:00 | 显示全部楼层
M051单片机的ADC可用于环境监测、健康跟踪等多种功能,提升设备的智能化水平。
tifmill 发表于 2024-6-9 12:37 | 显示全部楼层
通常具有一定数量的模拟-数字转换器(ADC)通道,用于将模拟信号转换为数字信号。
wangdezhi 发表于 2024-6-9 16:15 | 显示全部楼层
一种常见的实现同步采集的方式,可以在一组已配置的通道上自动序列化执行转换。
adolphcocker 发表于 2024-6-9 20:12 | 显示全部楼层
ADC可以配置为同步模式,以便在同一时钟周期内对多个输入进行采样,从而实现同步采集功能。
pmp 发表于 2024-6-10 20:02 | 显示全部楼层
NUC029xAN系列的A/D转换器支持四种操作模式:单次模式、Burst模式、扫描模式和连续模式。这些模式提供了灵活的数据采集方案,以适应不同的应用需求。
kmzuaz 发表于 2024-6-11 14:46 | 显示全部楼层
建议查阅M051系列的官方数据手册和应用笔记
lzmm 发表于 2024-6-11 17:56 | 显示全部楼层
请查阅新唐科技官方发布的M051型号的 data sheet,或者联系新唐科技的客服以获取详细的技术支持和数据手册。
updownq 发表于 2024-6-11 22:32 | 显示全部楼层
常见的ADC分辨率包括12位或更高,提供较为精确的模拟信号数字化能力。
jkl21 发表于 2024-6-12 11:56 | 显示全部楼层
一般会有8路或更多。              
febgxu 发表于 2024-6-12 15:18 | 显示全部楼层
内部内建2组12-bit ADC
您需要登录后才可以回帖 登录 | 注册

本版积分规则

21

主题

123

帖子

2

粉丝
快速回复 在线客服 返回列表 返回顶部