4
0
mirror of https://github.com/RT-Thread/rt-thread.git synced 2025-01-18 10:03:30 +08:00
2021-05-12 19:15:17 +08:00

1118 lines
30 KiB
C

/**************************************************************************//**
* @file pwm.c
* @brief N9H30 series PWM driver source file
*
* @note
* SPDX-License-Identifier: Apache-2.0
* Copyright (C) 2018 Nuvoton Technology Corp. All rights reserved.
*****************************************************************************/
#include "N9H30.h"
#include "nu_sys.h"
#include "nu_pwm.h"
/** @addtogroup N9H30_Device_Driver N9H30 Device Driver
@{
*/
/** @addtogroup N9H30_PWM_Driver PWM Driver
@{
*/
/** @addtogroup N9H30_PWM_EXPORTED_FUNCTIONS PWM Exported Functions
@{
*/
//Internal function definition
/// @cond HIDDEN_SYMBOLS
void pwmISR(PVOID pvParam);
static INT pwmInitGPIO(const INT nTimerIdentity, const INT nValue);
static INT pwmInitTimer(const INT nTimerIdentity);
static INT pwmStartTimer(const INT nTimerIdentity);
static INT pwmStopTimer(const INT nTimerIdentity, const INT nMethod);
// Register operation
static INT pwmSetCP(const INT nTimerIdentity, const INT nValue);
static INT pwmSetDZI(const INT nTimerIdentity, const INT nValue);
static INT pwmSetCSR(const INT nTimerIdentity, const INT nValue);
static INT pwmSetDZGenerator(const INT nTimerIdentity, const INT nStatus);
static INT pwmSetTimerState(const INT nTimerIdentity, const INT nStatus);
static INT pwmSetInverter(const INT nTimerIdentity, const INT nStatus);
static INT pwmSetMode(const INT nTimerIdentity, const INT nStatus);
static INT pwmSetCNR(const INT nTimerIdentity, const INT nValue);
static INT pwmSetCMR(const INT nTimerIdentity, const INT nValue);
static UINT pwmGetPDR(const INT nTimerIdentity);
static INT pwmSetPIER(const INT nTimerIdentity, const INT value);
static INT pwmCleanPIIR(const INT nTimerIdentity);
//Global variable
static BOOL bPWMIRQFlag = FALSE; //IRQ enable flag, set after PWM IRQ enable
static BOOL bPWMTimerOpenStatus[PWM_TIMER_NUM]; //timer flag which set after open(for disable IRQ decision)
static BOOL bPWMTimerStartStatus[PWM_TIMER_NUM]; //timer flag which set after Start count(to avoid incorrectly stop procedure)
static BOOL bPWMTimerMode[PWM_TIMER_NUM]; //PWM timer toggle/one shot mode
static BOOL volatile bPWMIntFlag[PWM_TIMER_NUM]; //interrupt flag which set by ISR
/// @endcond /* HIDDEN_SYMBOLS */
/**
* @brief The init function of PWM device driver
*/
INT pwmInit(void)
{
UINT temp;
// Enable PWM clock
temp = inpw(REG_CLK_PCLKEN1);
temp = temp | 0x8000000;
outpw(REG_CLK_PCLKEN1, temp);
sysInstallISR(IRQ_LEVEL_1, PWM_IRQn, (PVOID)pwmISR);
sysSetLocalInterrupt(ENABLE_IRQ); // Enable CPSR I bit
return 0;
}
/**
* @brief The exit function of PWM device driver
*/
INT pwmExit(void)
{
return 0;
}
/**
* @brief The open function of PWM device driver
* @param[in] nTimerIdentity PWM Timer channel identity
* @retval Successful PWM successfully opened
* @retval pwmTimerBusy PWM timer already open
* @retval pwmInvalidTimerChannel PWM Timer channel number error
*/
INT pwmOpen(const INT nTimerIdentity)
{
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// nTimerIdentity value error
}
if (bPWMTimerOpenStatus[nTimerIdentity] == TRUE)
{
return pwmTimerBusy;
}
if (bPWMIRQFlag == FALSE)
{
sysEnableInterrupt(PWM_IRQn);
bPWMIRQFlag = TRUE;
}
bPWMTimerOpenStatus[nTimerIdentity] = TRUE;
// Set PWM timer default value(CSR->PPR->PCR->CMR->CNR)
pwmInitTimer(nTimerIdentity);
//Enable PIER
pwmSetPIER(nTimerIdentity, PWM_ENABLE);
//Reset PIIR
pwmCleanPIIR(nTimerIdentity);
//Reset PWM timer start count flag
bPWMTimerStartStatus[nTimerIdentity] = FALSE;
return Successful;
}
/**
* @brief The close function of PWM device driver
* @param[in] nTimerIdentity PWM Timer channel identity
* @retval Successful PWM successfully closed
* @retval pwmTimerNotOpen PWM timer not open
* @retval pwmInvalidTimerChannel PWM Timer channel number error
*/
INT pwmClose(const INT nTimerIdentity)
{
INT nLoop;
BOOL uAllTimerClose = TRUE;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// nTimerIdentity value error
}
if (bPWMTimerOpenStatus[nTimerIdentity] == FALSE)
{
return pwmTimerNotOpen;
}
bPWMTimerOpenStatus[nTimerIdentity] = FALSE;
//Check if all timer stop, IRQ can be disable
for (nLoop = PWM_TIMER_MIN; nLoop < PWM_TIMER_NUM; nLoop++)
{
if (bPWMTimerOpenStatus[nLoop] == TRUE)
{
uAllTimerClose = FALSE;
}
}
//All timer stop, disable IRQs
if (uAllTimerClose == TRUE)
{
sysDisableInterrupt(PWM_IRQn);
bPWMIRQFlag = FALSE;
}
pwmSetPIER(nTimerIdentity, PWM_DISABLE);
pwmCleanPIIR(nTimerIdentity);
return Successful;
}
/**
* @brief The read function of PWM device driver
* @param[in] nTimerIdentity PWM Timer channel identity
* @param[out] pucStatusValue The point of typePWMSTATUS
* @param[in] uLength The length of typePWMSTATUS
* @retval Successful Read PWM value successfully
* @retval pwmTimerNotOpen PWM timer not open
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidStructLength Struct length error(struct type error)
*/
INT pwmRead(const INT nTimerIdentity, PUCHAR pucStatusValue, const UINT uLength)
{
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// nTimerIdentity value error
}
if (bPWMTimerOpenStatus[nTimerIdentity] == FALSE)
{
return pwmTimerNotOpen;
}
if (uLength != sizeof(typePWMSTATUS))
{
return pwmInvalidStructLength;// Struct length error(struct type error)
}
if (sizeof(*((typePWMSTATUS *)pucStatusValue)) != sizeof(typePWMSTATUS))
{
return pwmInvalidStructLength;// Struct length error(struct type error)
}
((typePWMSTATUS *)pucStatusValue)->PDR = pwmGetPDR(nTimerIdentity);
if (bPWMIntFlag[nTimerIdentity] == TRUE)
{
bPWMIntFlag[nTimerIdentity] = FALSE;
((typePWMSTATUS *)pucStatusValue)->InterruptFlag = TRUE;
}
else
{
((typePWMSTATUS *)pucStatusValue)->InterruptFlag = FALSE;
}
return Successful;
}
/**
* @brief The write function of PWM device driver
* @param[in] nTimerIdentity PWM Timer channel identity
* @param[in] pucCNRCMRValue The value of CNR and CMR
* @param[in] uLength For future usage
* @retval Successful Write PWM setting successfully
* @retval pwmTimerNotOpen PWM timer not open
* @retval pwmInvalidTimerChannel PWM Timer channel number error
*/
INT pwmWrite(const INT nTimerIdentity, PUCHAR pucCNRCMRValue, const UINT uLength)
{
typePWMVALUE pwmvalue;
INT nStatus;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// nTimerIdentity value error
}
if (bPWMTimerOpenStatus[nTimerIdentity] == FALSE)
{
return pwmTimerNotOpen;
}
if (uLength != sizeof(typePWMVALUE))
{
return pwmInvalidStructLength;// Struct length error(struct type error)
}
pwmvalue.value = ((typePWMVALUE *)pucCNRCMRValue)->value;
nStatus = pwmSetCNR(nTimerIdentity, pwmvalue.field.cnr);
if (nStatus != Successful)
{
return nStatus;
}
nStatus = pwmSetCMR(nTimerIdentity, pwmvalue.field.cmr);
if (nStatus != Successful)
{
return nStatus;
}
return Successful;
}
/**
* @brief The ioctl function of PWM device driver
* @param[in] nTimerIdentity PWM Timer channel identity
* @param[in] uCommand Ioctl command which indicates different operation
* @param[in] uIndication Not use in PWM
* @param[in] uValue The value which use with uCommand
* @retval Successful PWM ioctl execute successfully
* @retval pwmTimerNotOpen PWM timer not open
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidIoctlCommand Ioctl command error
* @retval Others Error according to different uCommand
*/
INT pwmIoctl(const INT nTimerIdentity, const UINT uCommand, const UINT uIndication, UINT uValue)
{
INT nStatus;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// nTimerIdentity value error
}
if (bPWMTimerOpenStatus[nTimerIdentity] == FALSE)
{
return pwmTimerNotOpen;
}
switch (uCommand)
{
case START_PWMTIMER:
{
nStatus = pwmStartTimer(nTimerIdentity);
break;
}
case STOP_PWMTIMER:
{
// default stop method is 2
nStatus = pwmStopTimer(nTimerIdentity, PWM_STOP_METHOD2);
break;
}
case SET_CSR:
{
nStatus = pwmSetCSR(nTimerIdentity, uValue);
break;
}
case SET_CP:
{
nStatus = pwmSetCP(nTimerIdentity, uValue);
break;
}
case SET_DZI:
{
nStatus = pwmSetDZI(nTimerIdentity, uValue);
break;
}
case SET_INVERTER:
{
nStatus = pwmSetInverter(nTimerIdentity, uValue);
break;
}
case SET_MODE:
{
nStatus = pwmSetMode(nTimerIdentity, uValue);
break;
}
case ENABLE_DZ_GENERATOR:
{
nStatus = pwmSetDZGenerator(nTimerIdentity, PWM_ENABLE);
break;
}
case DISABLE_DZ_GENERATOR:
{
nStatus = pwmSetDZGenerator(nTimerIdentity, PWM_DISABLE);
break;
}
case ENABLE_PWMGPIOOUTPUT:
{
nStatus = pwmInitGPIO(nTimerIdentity, uValue);
break;
}
default:
{
return pwmInvalidIoctlCommand;
}
}
return nStatus;
}
/// @cond HIDDEN_SYMBOLS
/**
* @brief The interrupt service routines of PWM
* @param[in] pvParam IRQ Parameter(not use in PWM)
*/
VOID pwmISR(PVOID pvParam)
{
INT i;
UINT32 uRegisterValue = 0;
uRegisterValue = inpw(REG_PWM_PIIR);// Get PIIR value
for (i = 0; i < PWM_TIMER_NUM ; i++)
{
if (uRegisterValue & (1 << i))
{
bPWMIntFlag[i] = 1;
outpw(REG_PWM_PIIR, (1 << i));
}
}
}
/**
* @brief This function set corresponding GPIO as PWM function according to the
* parameter nTimerIdentity
* @param[in] nTimerIdentity Timer channel number
* @retval Successful PWM init GPIO successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidPin PWM output pin setting error
*/
static INT pwmInitGPIO(const INT nTimerIdentity, const INT nValue)
{
UINT temp = 0;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nTimerIdentity == PWM_TIMER0)
{
if (nValue == PWM0_GPA12)
{
temp = inpw(REG_SYS_GPA_MFPH);
temp = (temp & ~0x000F0000) | 0xD0000;
outpw(REG_SYS_GPA_MFPH, temp);
}
else if (nValue == PWM0_GPB2)
{
temp = inpw(REG_SYS_GPB_MFPL);
temp = (temp & ~0xF00) | 0xD00;
outpw(REG_SYS_GPB_MFPL, temp);
}
else
return pwmInvalidPin;
}
else if (nTimerIdentity == PWM_TIMER1)
{
if (nValue == PWM1_GPA13)
{
temp = inpw(REG_SYS_GPA_MFPH);
temp = (temp & ~0x00F00000) | 0xD00000;
outpw(REG_SYS_GPA_MFPH, temp);
}
else if (nValue == PWM1_GPB3)
{
temp = inpw(REG_SYS_GPB_MFPL);
temp = (temp & ~0xF000) | 0xD000;
outpw(REG_SYS_GPB_MFPL, temp);
}
else
return pwmInvalidPin;
}
else if (nTimerIdentity == PWM_TIMER2)
{
if (nValue == PWM2_GPA14)
{
temp = inpw(REG_SYS_GPA_MFPH);
temp = (temp & ~0x0F000000) | 0xD000000;
outpw(REG_SYS_GPA_MFPH, temp);
}
else if (nValue == PWM2_GPH2)
{
temp = inpw(REG_SYS_GPH_MFPL);
temp = (temp & ~0xF00) | 0xD00;
outpw(REG_SYS_GPH_MFPL, temp);
}
else
return pwmInvalidPin;
}
else
{
if (nValue == PWM3_GPA15)
{
temp = inpw(REG_SYS_GPA_MFPH);
temp = (temp & ~0xF0000000) | 0xD0000000;
outpw(REG_SYS_GPA_MFPH, temp);
}
else if (nValue == PWM3_GPH3)
{
temp = inpw(REG_SYS_GPH_MFPL);
temp = (temp & ~0xF000) | 0xD000;
outpw(REG_SYS_GPH_MFPL, temp);
}
else
return pwmInvalidPin;
}
return Successful;
}
/**
* @brief This function initiates PWM timer n and set the default setting to CSR,
* PPR, PCR, CNR, CMR
* @param[in] nTimerIdentity Timer channel number
* @retval Successful PWM init timer successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
*/
static INT pwmInitTimer(const INT nTimerIdentity)
{
typePPR PWMPPR;
INT nStatus;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// nTimerIdentity value error
}
//Set CSR
nStatus = pwmSetCSR(nTimerIdentity, DEFAULT_CSR);
if (nStatus != Successful)
{
return nStatus;
}
//Set PPR
PWMPPR.value = (UINT)inpw(REG_PWM_PPR);
switch (nTimerIdentity)
{
case PWM_TIMER0:
{
if (PWMPPR.field.cp0 == 0)
{
pwmSetCP(nTimerIdentity, DEFAULT_CP);
}
break;
}
case PWM_TIMER1:
{
if (PWMPPR.field.cp0 == 0)
{
pwmSetCP(nTimerIdentity, DEFAULT_CP);
}
break;
}
case PWM_TIMER2:
{
if (PWMPPR.field.cp1 == 0)
{
pwmSetCP(nTimerIdentity, DEFAULT_CP);
}
break;
}
case PWM_TIMER3:
{
if (PWMPPR.field.cp1 == 0)
{
pwmSetCP(nTimerIdentity, DEFAULT_CP);
}
break;
}
}
//Set PCR
nStatus = pwmSetMode(nTimerIdentity, DEFAULT_MODE);
if (nStatus != Successful)
{
return nStatus;
}
bPWMTimerMode[nTimerIdentity] = DEFAULT_MODE;
//Set CMR
nStatus = pwmSetCMR(nTimerIdentity, DEFAULT_CMR);
if (nStatus != Successful)
{
return nStatus;
}
//Set CNR
nStatus = pwmSetCNR(nTimerIdentity, DEFAULT_CNR);
if (nStatus != Successful)
{
return nStatus;
}
return Successful;
}
/**
* @brief This function starts PWM timer according to the parameter
* @param[in] nTimerIdentity Timer channel number
* @retval Successful PWM start timer successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
*/
static INT pwmStartTimer(const INT nTimerIdentity)
{
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
pwmSetTimerState(nTimerIdentity, PWM_ENABLE);
if (bPWMTimerMode[nTimerIdentity] == PWM_TOGGLE)
{
bPWMTimerStartStatus[nTimerIdentity] = TRUE;
}
return Successful;
}
/**
* @brief This function stops PWM timer n using method 1, 2, or 3 according to the
* parameter nTimerIdentity and nStatus
* @param[in] nTimerIdentity Timer channel number
* @param[in] nMethod Stop PWM timer method
* @retval Successful PWM stop timer successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidStopMethod Stop method error
*/
static INT pwmStopTimer(const INT nTimerIdentity, INT nMethod)
{
typeCNR PWMCNR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
// Timer_num value error
return pwmInvalidTimerChannel;
}
//Can't stop before open PWM timer
if (bPWMTimerOpenStatus[nTimerIdentity] == FALSE)
{
return Successful;
}
// one shot mode didn't need stop procedure
if (bPWMTimerMode[nTimerIdentity] == PWM_ONESHOT)
{
return Successful;
}
// Timer stop already, no need to stop again
if (bPWMTimerStartStatus[nTimerIdentity] == FALSE)
{
return Successful;
}
// Set CNR as 0
PWMCNR.field.cnr = 0;
outpw(REG_PWM_CNR0 + (PWM_OFFSET * nTimerIdentity), PWMCNR.value);
switch (nMethod)
{
case PWM_STOP_METHOD1:
{
while (1)
{
if (pwmGetPDR(nTimerIdentity) == 0) // Wait PDR reach to 0
{
pwmSetTimerState(nTimerIdentity, PWM_DISABLE);// Disable pwm timer
bPWMIntFlag[nTimerIdentity] = FALSE;
bPWMTimerStartStatus[nTimerIdentity] = FALSE;
break;
}
}
break;
}
case PWM_STOP_METHOD2:
{
while (1)
{
if (bPWMIntFlag[nTimerIdentity] == TRUE) // Wait interrupt happen
{
pwmSetTimerState(nTimerIdentity, PWM_DISABLE);// Disable pwm timer
bPWMIntFlag[nTimerIdentity] = FALSE;
bPWMTimerStartStatus[nTimerIdentity] = FALSE;
break;
}
}
break;
}
/*case PWM_STOP_METHOD3:
{
pwmSetPCRState(nTimerIdentity, PWM_DISABLE);// Disable pwm timer
bPWMIntFlag[nTimerIdentity] = FALSE;
bPWMTimerStartStatus[nTimerIdentity] = FALSE;
break;
}*/
default:
{
return pwmInvalidStopMethod;// Stop method value error
}
}
return Successful;
}
/**
* @brief This function set CPn value according to the parameter nTimerIdentity and nValue
* @param[in] nTimerIdentity Timer channel number
* @param[in] nValue The value which want to set in CSRn
* @retval Successful Set CPn successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidCPValue PWM_PPR CPn value out of range
*/
static INT pwmSetCP(const INT nTimerIdentity, const INT nValue)
{
typePPR PWMPPR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nValue < CP_MIN || nValue > CP_MAX)
{
return pwmInvalidCPValue;// CP value error
}
PWMPPR.value = (UINT)inpw(REG_PWM_PPR);
switch (nTimerIdentity)
{
case PWM_TIMER0:
{
PWMPPR.field.cp0 = nValue;
break;
}
case PWM_TIMER1:
{
PWMPPR.field.cp0 = nValue;
break;
}
case PWM_TIMER2:
{
PWMPPR.field.cp1 = nValue;
break;
}
case PWM_TIMER3:
{
PWMPPR.field.cp1 = nValue;
break;
}
}
outpw(REG_PWM_PPR, PWMPPR.value);
return Successful;
}
/**
* @brief This function set DZIn value according to the parameter nTimerIdentity and nValue
* @param[in] nTimerIdentity Timer channel number
* @param[in] nValue The value which want to set in DZIn
* @retval Successful Set DZIn successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidDZIValue PWM_PPR DZIn value out of range
*/
static INT pwmSetDZI(const INT nTimerIdentity, const INT nValue)
{
typePPR PWMPPR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nValue < DZI_MIN || nValue > DZI_MAX)
{
return pwmInvalidDZIValue;// CSR value error
}
PWMPPR.value = (UINT)inpw(REG_PWM_PPR);
switch (nTimerIdentity)
{
case PWM_TIMER0:
{
PWMPPR.field.dzi0 = nValue;
break;
}
case PWM_TIMER1:
{
PWMPPR.field.dzi0 = nValue;
break;
}
case PWM_TIMER2:
{
PWMPPR.field.dzi1 = nValue;
break;
}
case PWM_TIMER3:
{
PWMPPR.field.dzi1 = nValue;
break;
}
}
outpw(REG_PWM_PPR, PWMPPR.value);
return Successful;
}
/**
* @brief This function set CSRn value according to the parameter nTimerIdentity and nValue
* @param[in] nTimerIdentity Timer channel number
* @param[in] nValue The value which want to set in CSRn
* @retval Successful Set CSRn successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
*/
static INT pwmSetCSR(const INT nTimerIdentity, const INT nValue)
{
typeCSR PWMCSR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nValue < CSR_MIN || nValue > CSR_MAX)
{
return pwmInvalidCSRValue;// CSR value error
}
PWMCSR.value = (UINT)inpw(REG_PWM_CSR);
switch (nTimerIdentity)
{
case PWM_TIMER0:
{
PWMCSR.field.csr0 = nValue;
break;
}
case PWM_TIMER1:
{
PWMCSR.field.csr1 = nValue;
break;
}
case PWM_TIMER2:
{
PWMCSR.field.csr2 = nValue;
break;
}
case PWM_TIMER3:
{
PWMCSR.field.csr3 = nValue;
break;
}
}
outpw(REG_PWM_CSR, PWMCSR.value);
return Successful;
}
/**
* @brief This function enable/disable PWM channel n dead zone function according to the
* parameter nTimerIdentity and nStatus
* @param[in] nTimerIdentity Timer channel number
* @param[in] nStatus PWMDZG_ENABLE/PWMDZG_DISABLE
* @retval Successful Set dead zone successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidDZGStatus PWM Dead-Zone Generator enable/disable status error
*/
static INT pwmSetDZGenerator(const INT nTimerIdentity, INT nStatus)
{
typePCR PWMPCR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nStatus != PWMDZG_ENABLE && nStatus != PWMDZG_DISABLE)
{
return pwmInvalidDZGStatus;// PCR inverter value error
}
PWMPCR.value = (UINT)inpw(REG_PWM_PCR);
switch (nTimerIdentity)
{
case PWM_TIMER0:
{
PWMPCR.field.grpup0_dzen = nStatus;
break;
}
case PWM_TIMER1:
{
PWMPCR.field.grpup0_dzen = nStatus;
break;
}
case PWM_TIMER2:
{
PWMPCR.field.grpup1_dzen = nStatus;
break;
}
case PWM_TIMER3:
{
PWMPCR.field.grpup1_dzen = nStatus;
break;
}
}
outpw(REG_PWM_PCR, PWMPCR.value);
return Successful;
}
/**
* @brief This function set PWM channel n enable/disable according to the
* parameter nTimerIdentity and nStatus
* @param[in] nTimerIdentity Timer channel number
* @param[in] nStatus PWM_ENABLE/PWMDISABLE
* @retval Successful Set channel enable/disable successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
*/
static INT pwmSetTimerState(const INT nTimerIdentity, INT nStatus)
{
typePCR PWMPCR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nStatus != PWM_ENABLE && nStatus != PWM_DISABLE)
{
return pwmInvalidTimerStatus;
}
PWMPCR.value = (UINT)inpw(REG_PWM_PCR);
switch (nTimerIdentity)
{
case PWM_TIMER0:
{
PWMPCR.field.ch0_en = nStatus;
break;
}
case PWM_TIMER1:
{
PWMPCR.field.ch1_en = nStatus;
break;
}
case PWM_TIMER2:
{
PWMPCR.field.ch2_en = nStatus;
break;
}
case PWM_TIMER3:
{
PWMPCR.field.ch3_en = nStatus;
break;
}
}
outpw(REG_PWM_PCR, PWMPCR.value);
return Successful;
}
/**
* @brief This function set PWM channel n inverter on/off according to the
* parameter nTimerIdentity and nStatus
* @param[in] nTimerIdentity Timer channel number
* @param[in] nStatus PWM_ENABLE/PWM_DISABLE
* @retval Successful Set inverter successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidInverterValue Inverter value error
*/
static INT pwmSetInverter(const INT nTimerIdentity, INT nStatus)
{
typePCR PWMPCR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nStatus != PWM_INVON && nStatus != PWM_INVOFF)
{
return pwmInvalidInverterValue;// PCR inverter value error
}
PWMPCR.value = (UINT)inpw(REG_PWM_PCR);
switch (nTimerIdentity)
{
case PWM_TIMER0:
{
PWMPCR.field.ch0_inverter = nStatus;
break;
}
case PWM_TIMER1:
{
PWMPCR.field.ch1_inverter = nStatus;
break;
}
case PWM_TIMER2:
{
PWMPCR.field.ch2_inverter = nStatus;
break;
}
case PWM_TIMER3:
{
PWMPCR.field.ch3_inverter = nStatus;
break;
}
}
outpw(REG_PWM_PCR, PWMPCR.value);
return Successful;
}
/**
* @brief This function set PWM channel n toggle/one shot mode according to the
* parameter nTimerIdentity and nStatus
* @param[in] nTimerIdentity Timer channel number
* @param[in] nStatus PWM_TOGGLE/PWM_ONESHOT
* @retval Successful Set operation mode successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidModeStatus Operating mode error
*/
static INT pwmSetMode(const INT nTimerIdentity, INT nStatus)
{
typePCR PWMPCR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nStatus != PWM_TOGGLE && nStatus != PWM_ONESHOT)
{
return pwmInvalidModeStatus;// PCR inverter value error
}
PWMPCR.value = (UINT)inpw(REG_PWM_PCR);
switch (nTimerIdentity)
{
case PWM_TIMER0:
{
PWMPCR.field.ch0_mode = nStatus;
break;
}
case PWM_TIMER1:
{
PWMPCR.field.ch1_mode = nStatus;
break;
}
case PWM_TIMER2:
{
PWMPCR.field.ch2_mode = nStatus;
break;
}
case PWM_TIMER3:
{
PWMPCR.field.ch3_mode = nStatus;
break;
}
}
outpw(REG_PWM_PCR, PWMPCR.value);
bPWMTimerMode[nTimerIdentity] = nStatus;
return Successful;
}
/**
* @brief This function set PWM_CNRn value according to the parameter nTimerIdentity and nValue
* @param[in] nTimerIdentity Timer channel number
* @param[in] nValue CNR value
* @retval Successful Set CNR successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidCNRValue Invalid CNR value
*/
static INT pwmSetCNR(const INT nTimerIdentity, INT nValue)
{
typeCNR PWMCNR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nValue < CNR_MIN || nValue > CNR_MAX)
{
return pwmInvalidCNRValue;// PCR inverter value error
}
PWMCNR.field.cnr = nValue;
outpw(REG_PWM_CNR0 + (PWM_OFFSET * nTimerIdentity), PWMCNR.value);
return Successful;
}
/**
* @brief This function set PWM_CMRn value according to the parameter nTimerIdentity and nValue
* @param[in] nTimerIdentity Timer channel number
* @param[in] nValue CMR value
* @retval Successful Set CMR successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval pwmInvalidCMRValue Invalid CMR value
*/
static INT pwmSetCMR(const INT nTimerIdentity, INT nValue)
{
typeCMR PWMCMR;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
if (nValue < CMR_MIN || nValue > CMR_MAX)
{
return pwmInvalidCMRValue;// CMR value error
}
PWMCMR.field.cmr = nValue;
outpw(REG_PWM_CMR0 + (PWM_OFFSET * nTimerIdentity), PWMCMR.value);
return Successful;
}
/**
* @brief This function return the PDR value of PWM timer n
* @param[in] nTimerIdentity Timer channel number
* @retval pwmInvalidTimerChannel PWM Timer channel number error
* @retval Others Current PDR value
*/
static UINT pwmGetPDR(const INT nTimerIdentity)
{
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
else
{
return (UINT)inpw(REG_PWM_PDR0 + (PWM_OFFSET * nTimerIdentity)); // Return PDR value
}
}
/**
* @brief This function set the PIERn bit of PWM timer n as 1 or 0 according to the
* parameter nTimerIdentity and nValue
* @param[in] nTimerIdentity Timer channel number
* @param[in] nValue PWM_ENABLE/PWM_DISABLE
* @retval Successful Set PIER successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
*/
static INT pwmSetPIER(const INT nTimerIdentity, INT nValue)
{
UINT uRegisterValue = 0;;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// Timer_num value error
}
else
{
uRegisterValue = (UINT)inpw(REG_PWM_PIER);
if (nValue == PWM_ENABLE)
{
uRegisterValue = uRegisterValue | (1 << nTimerIdentity); // Set PIER
}
else
{
uRegisterValue = uRegisterValue & (0 << nTimerIdentity); // Clear PIER
}
outpw(REG_PWM_PIER, uRegisterValue);// Write value to PIER
return Successful;
}
}
/**
* @brief This function clear PIIRn bit according to the parameter nTimerIdentity
* @param[in] nTimerIdentity Timer channel number
* @retval Successful Clear PIIR successfully
* @retval pwmInvalidTimerChannel PWM Timer channel number error
*/
static INT pwmCleanPIIR(const INT nTimerIdentity)
{
UINT uRegisterValue = 0;
if (nTimerIdentity < PWM_TIMER_MIN || nTimerIdentity > PWM_TIMER_MAX)
{
return pwmInvalidTimerChannel;// nTimerIdentity value error
}
uRegisterValue = (UINT)inpw(REG_PWM_PIIR);
uRegisterValue = uRegisterValue & ~(1 << nTimerIdentity);
outpw(REG_PWM_PIIR, uRegisterValue);
return Successful;
}
/// @endcond /* HIDDEN_SYMBOLS */
/*@}*/ /* end of group N9H30_PWM_EXPORTED_FUNCTIONS */
/*@}*/ /* end of group N9H30_PWM_Driver */
/*@}*/ /* end of group N9H30_Device_Driver */
/*** (C) COPYRIGHT 2018 Nuvoton Technology Corp. ***/