[DemoCode下载] 新唐M051 ADC模数转换程序

[复制链接]
 楼主| heisexingqisi 发表于 2016-7-14 07:13 | 显示全部楼层
Roderman_z 发表于 2016-7-9 20:45
如何设置两个adc同步采集呢

如果是两个不停的扫描那就是连续循环扫描
  1. /****************************************************************************
  2. * @file     main.c
  3. * @version  V3.0
  4. * $Revision: 5 $
  5. * $Date: 14/01/28 11:44a $
  6. * @brief    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. }


天灵灵地灵灵 发表于 2016-7-14 23:49 | 显示全部楼层
* ADC时钟使能 */      
    set_ADEN_CLK;
    if (ADC_CLK_Source==0x00000000 )
    {
        ADCClkSource_ex12MHZ;
    }
    else if(ADC_CLK_Source==0x00000004 )
    {
        ADCClkSource_PLL;
    }
    else if(ADC_CLK_Source==0x00000008 )
    {
        ADCClkSource_int22MHZ;
时钟选择很重要。
huangcunxiake 发表于 2016-7-17 23:03 | 显示全部楼层
set_ADST;            //启动ADC
这个不是官方的宏吧。
 楼主| heisexingqisi 发表于 2016-7-21 22:19 | 显示全部楼层
代码来自新唐的官方BSP包。
yujinjin616 发表于 2020-8-15 23:27 | 显示全部楼层
你好,这颗芯片外围用那颗ADC芯片了?
您需要登录后才可以回帖 登录 | 注册

本版积分规则

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