/******************************************************************************
* [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: 13/10/07 3:57p $
* [url=home.php?mod=space&uid=247401]@brief[/url] PWM sample for Mini51 series MCU
*
* @note
* Copyright (C) 2013 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 0 frequency is 100Hz, every 1 second enter this IRQ handler 100 times.
if(++cnt == 100) {
if(out)
PWM_EnableOutput(PWM, 0x3F);
else
PWM_DisableOutput(PWM, 0x3F);
out ^= 1;
cnt = 0;
}
// Clear channel 0 period interrupt flag
PWM_ClearPeriodIntFlag(PWM, 0);
PWM_ClearADCTriggerFlag(PWM, 0, PWM_TRIGGER_ADC_CNTR_IS_CNR);
//PWM->PFBCON |= PWM_PFBCON_BKF_Msk; //CLR Brake Function Flag
}
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->APBCLK = CLK_APBCLK_UART_EN_Msk | CLK_APBCLK_PWM01_EN_Msk | CLK_APBCLK_PWM23_EN_Msk | CLK_APBCLK_PWM45_EN_Msk | CLK_APBCLK_ADC_EN_Msk;
/* ADC clock source is 22.1184MHz, set divider to (3 + 1), ADC clock is 22.1184/4 MHz */
CLK->CLKDIV |= (3 << CLK_CLKDIV_ADC_N_Pos);
/* Select UART clock source from external crystal*/
CLK->CLKSEL1 = (CLK->CLKSEL1 & ~CLK_CLKSEL1_UART_S_Msk) | CLK_CLKSEL1_UART_S_XTAL;
/* 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 P0 multi-function pins for PWM Channel 5 */
SYS->P0_MFP |= SYS_MFP_P04_PWM5;
/* Set P2 multi-function pins for PWM Channel 0~4 */
SYS->P2_MFP |= SYS_MFP_P22_PWM0 | SYS_MFP_P23_PWM1 | SYS_MFP_P24_PWM2 | SYS_MFP_P25_PWM3 | SYS_MFP_P26_PWM4;
/* 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;
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 and removal connect,\n");
printf("well be generate the brake Flag to STOP PWM waveform\n");
printf("**------------------------------------------------------------------------------------**\n");
// Enable channel 0
ADC_Open(ADC, 0, 0, 0x01);
// Power on ADC
ADC_POWER_ON(ADC);
ADC->ADCR |=ADC_ADCR_TRGEN_Msk;
ADC->ADCR |=ADC_TRIGGER_BY_PWM;
ADC->ADCR |=ADC_ADCR_ADIE_Msk;
ADC_CLR_INT_FLAG(ADC, u32Flag);
// Configure and enable Comperator 0 to monitor channel 0 input less than 0x200
ADC_ENABLE_CMP0(ADC, 0, ADC_CMP_LESS_THAN, 0x200, 16);
// Configure and enable Comperator 1 to monitor channel 0 input greater or euqal to 0x200
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);
//setting counter matching CNR2 trigger ADC
PWM->TRGCON0 |= 0x00020000;
PWM_SET_ALIGNED_TYPE(pwm, 2, PWM_CENTER_ALIGNED);
// PWM2 frequency is 300Hz, duty 50%
PWM_ConfigOutputChannel(PWM, 2, 300, 50);
PWM_EnableDeadZone(PWM, 2, 200);
// Enable output of all PWM channels
PWM_EnableOutput(PWM, 0x3f);
// Enable PWM channel 0 period interrupt, use channel 0 to measure time.
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;
P5->IEN |=0x04; //EN P5.2 INT1 Falling
NVIC_EnableIRQ(EINT1_IRQn);
// Start
PWM_Start(PWM, 0x3f);
while(1);
}
/*** (C) COPYRIGHT 2013 Nuvoton Technology Corp. ***/