/******************************************************************************
* [url=home.php?mod=space&uid=288409]@file[/url] main.c
* [url=home.php?mod=space&uid=895143]@version[/url] V1.00
* $Revision: 3 $
* $Date: 19/6/25 3:57p $
* [url=home.php?mod=space&uid=247401]@brief[/url] PWM trigger ADC and PWM brake sample for Mini51DE series MCU
*
* @note
* Copyright (C) 2019 Nuvoton Technology Corp. All rights reserved.
*****************************************************************************/
#include <stdio.h>
#include "Mini51Series.h"
volatile uint32_t u32Flag;
void ADC_IRQHandler(void)
{
/* Get ADC comparator interrupt flag */
u32Flag = ADC_GET_INT_FLAG(ADC, ADC_CMP0_INT | ADC_CMP1_INT);
if (u32Flag & ADC_CMP0_INT)
printf("Channel 0 input < 0x200\n");
if (u32Flag & ADC_CMP1_INT)
printf("Channel 0 input >= 0x200\n");
ADC_CLR_INT_FLAG(ADC, u32Flag);
}
void EINT1_IRQHandler(void)
{
P55 = 0; /* debug IO */
printf("INT1");
P5->ISRC |= 0x04; /* CLR INT1 interrupt Flag */
PWM->PFBCON |= PWM_PFBCON_BKF_Msk; /* CLR Brake Function Flag */
}
void PWM_IRQHandler(void)
{
static uint32_t cnt;
static uint32_t out;
/* Channel 2 frequency is 1000Hz, every 1 second enter this IRQ handler 1000 times. */
if (++cnt == 100)
{
if (out)
PWM_EnableOutput(PWM, BIT2);
else
PWM_DisableOutput(PWM, BIT2);
out ^= 1;
cnt = 0;
}
/* Clear channel 2 period and trigger ADC interrupt flag */
PWM_ClearPeriodIntFlag(PWM, 2);
PWM_ClearADCTriggerFlag(PWM, 2, PWM_TRIGGER_ADC_CNTR_IS_CNR);
}
void SYS_Init(void)
{
/*---------------------------------------------------------------------------------------------------------*/
/* Init System Clock */
/*---------------------------------------------------------------------------------------------------------*/
/* Unlock protected registers */
SYS_UnlockReg();
/* Enable external 12MHz XTAL, internal 22.1184MHz */
CLK->PWRCON = CLK_PWRCON_XTL12M | CLK_PWRCON_IRC22M_EN_Msk;
/* Waiting for clock ready */
CLK_WaitClockReady(CLK_CLKSTATUS_XTL_STB_Msk | CLK_CLKSTATUS_IRC22M_STB_Msk);
/* Enable IP clock */
CLK_EnableModuleClock(UART_MODULE);
CLK_EnableModuleClock(ADC_MODULE);
CLK_EnableModuleClock(PWM23_MODULE);
/* Select IP clock source */
CLK_SetModuleClock(UART_MODULE, CLK_CLKSEL1_UART_S_XTAL, CLK_CLKDIV_UART(1));
CLK_SetModuleClock(ADC_MODULE, CLK_CLKSEL1_ADC_S_HIRC, CLK_CLKDIV_ADC(4));
CLK_SetModuleClock(PWM23_MODULE, CLK_CLKSEL1_PWM23_S_HCLK, 0);
/* Update System Core Clock */
/* User can use SystemCoreClockUpdate() to calculate SystemCoreClock and CycylesPerUs automatically. */
SystemCoreClockUpdate();
/*---------------------------------------------------------------------------------------------------------*/
/* Init I/O Multi-function */
/*---------------------------------------------------------------------------------------------------------*/
/* Set P1 multi-function pins for UART RXD, TXD */
SYS->P0_MFP = SYS_MFP_P00_TXD | SYS_MFP_P01_RXD;
/* Set P2.4 multi-function pins for PWM Channel 2 */
SYS->P2_MFP |= SYS_MFP_P24_PWM2;
/* Set P5.3 to ADC channel 0 input pin */
SYS->P5_MFP |= SYS_MFP_P53_AIN0;
/* Analog pin OFFD to prevent leakage */
P5->OFFD |= (1 << 3) << GPIO_OFFD_OFFD_Pos;
/* Set P5.5 to GPIO for INT1 debug */
SYS->P5_MFP |= SYS_MFP_P55_GPIO;
GPIO_SetMode(P5, BIT5, GPIO_PMD_OUTPUT);
/* set P5.2 to INT1 mode to use the PWM Brake function */
SYS->P5_MFP |= SYS_MFP_P52_INT1;
/* Lock protected registers */
SYS_LockReg();
GPIO_SetMode(P5, BIT2, GPIO_PMD_QUASI);
}
int32_t main(void)
{
/* Init System, IP clock and multi-function I/O
In the end of SYS_Init() will issue SYS_LockReg()
to lock protected register. If user want to write
protected register, please issue SYS_UnlockReg()
to unlock protected register if necessary */
SYS_Init();
/* Init UART to 115200-8n1 for print message */
UART_Open(UART, 115200);
printf("\nThis sample code will output PWM channel 2 (pin 24) to trigger ADC channel 0 (pin 41),\n");
printf("and setting ADC conversion event source to PWM trigger,\n");
printf("and setting Brake Function event source to INT1 (pin 20),\n");
printf("the INT1 interrupt event is seting falling trigger,\n");
printf("so user need use INT1 pin connect to VCC then connect to VSS,\n");
printf("that well generate brake to stop PWM waveform. ADC also stops conversion.\n");
printf("**------------------------------------------------------------------------------------**\n");
/* Enable channel 0 */
ADC_Open(ADC, 0, 0, 0x01);
/* Power on ADC */
ADC_POWER_ON(ADC);
/* Set ADC trigger source to PWM and trigger delay to 1 HCLK */
ADC_EnableHWTrigger(ADC, ADC_TRIGGER_BY_PWM, 1);
/* Enable ADC interrupt */
ADC_EnableInt(ADC, ADC_ADF_INT);
/* Clear ADC interrupt */
ADC_CLR_INT_FLAG(ADC, u32Flag);
/* Configure and enable Comparator 0 to monitor channel 0 input less than 0x200, match count 16 */
ADC_ENABLE_CMP0(ADC, 0, ADC_CMP_LESS_THAN, 0x200, 16);
/* Configure and enable Comparator 1 to monitor channel 0 input greater or equal to 0x200, match count 16 */
ADC_ENABLE_CMP1(ADC, 0, ADC_CMP_GREATER_OR_EQUAL_TO, 0x200, 16);
/* Enable ADC comparator 0 and 1 interrupt */
ADC_EnableInt(ADC, ADC_CMP0_INT);
ADC_EnableInt(ADC, ADC_CMP1_INT);
NVIC_EnableIRQ(ADC_IRQn);
/* PWM2 frequency is 1000Hz, duty 50% */
PWM_ConfigOutputChannel(PWM, 2, 1000, 50);
PWM_EnableDeadZone(PWM, 2, 200);
/* Set PWM channel 2 counter matching CNR trigger ADC */
PWM->TRGCON0 |= PWM_TRGCON0_CNT2TRGEN_Msk;
PWM_SET_ALIGNED_TYPE(PWM, 2, PWM_CENTER_ALIGNED);
/* Enable output of PWM channel 2 */
PWM_EnableOutput(PWM, 1 << 2);
/* Enable PWM channel 2 period interrupt */
PWM_EnablePeriodInt(PWM, 2, PWM_PERIOD_INT_MATCH_CNR);
NVIC_EnableIRQ(PWM_IRQn);
/* PWM PFBCON: PWMBKO2 Mask */
PWM->PFBCON |= PWM_PFBCON_PWMBKO2_Msk;
/* PWM brake Flag clear */
PWM->PFBCON |= PWM_PFBCON_BKF_Msk;
/* PWM BRKIE */
PWM->PIER |= PWM_PIER_BRKIE_Msk;
PWM->PHCHGNXT |= PWM_PHCHGNXT_CE1_Msk;
/* Comparator 0 as fault break 1 source */
PWM->PFBCON |= PWM_FB1_EINT1;
PWM->PFBCON &= ~PWM_PFBCON_CPO1BKEN_Msk;
/* EN P5.2 INT1 Falling */
P5->IEN |= 0x04;
NVIC_EnableIRQ(EINT1_IRQn);
/* Enable PWM channel 2 counter */
PWM_Start(PWM, 1 << 2);
while (1);
}
/*** (C) COPYRIGHT 2019 Nuvoton Technology Corp. ***/