Date: March 20, 2011 This library is created from "LPC17xx CMSIS-Compliant Standard Peripheral Firmware Driver Library (GNU, Keil, IAR) (Jan 28, 2011)", available from NXP's website, under "All microcontrollers support documents" [[http://ics.nxp.com/support/documents/microcontrollers/?type=software]] You will need to follow [[/projects/libraries/svn/mbed/trunk/LPC1768/LPC17xx.h]] while using this library Examples provided here [[/users/frank26080115/programs/LPC1700CMSIS_Examples/]] The beautiful thing is that NXP does not place copyright protection on any of the files in here Only a few modifications are made to make it compile with the mbed online compiler, I fixed some warnings as well. This is untested as of March 20, 2011 Forum post about this library: [[/forum/mbed/topic/2030/]]

lpc17xx_pwm.c

Committer:
frank26080115
Date:
2011-03-20
Revision:
0:84d7747641aa

File content as of revision 0:84d7747641aa:

/***********************************************************************//**
 * @file        lpc17xx_pwm.c
 * @brief        Contains all functions support for PWM firmware library on LPC17xx
 * @version        2.0
 * @date        21. May. 2010
 * @author        NXP MCU SW Application Team
 **************************************************************************
 * Software that is described herein is for illustrative purposes only
 * which provides customers with programming information regarding the
 * products. This software is supplied "AS IS" without any warranties.
 * NXP Semiconductors assumes no responsibility or liability for the
 * use of the software, conveys no license or title under any patent,
 * copyright, or mask work right to the product. NXP Semiconductors
 * reserves the right to make changes in the software without
 * notification. NXP Semiconductors also make no representation or
 * warranty that such application will be suitable for the specified
 * use without further testing or modification.
 **********************************************************************/

/* Peripheral group ----------------------------------------------------------- */
/** @addtogroup PWM
 * @{
 */

/* Includes ------------------------------------------------------------------- */
#include "lpc17xx_pwm.h"
#include "lpc17xx_clkpwr.h"

/* If this source file built with example, the LPC17xx FW library configuration
 * file in each example directory ("lpc17xx_libcfg.h") must be included,
 * otherwise the default FW library configuration file must be included instead
 */
#ifdef __BUILD_WITH_EXAMPLE__
#include "lpc17xx_libcfg.h"
#else
#include "lpc17xx_libcfg_default.h"
#endif /* __BUILD_WITH_EXAMPLE__ */


#ifdef _PWM


/* Public Functions ----------------------------------------------------------- */
/** @addtogroup PWM_Public_Functions
 * @{
 */


/*********************************************************************//**
 * @brief         Check whether specified interrupt flag in PWM is set or not
 * @param[in]    PWMx: PWM peripheral, should be LPC_PWM1
 * @param[in]    IntFlag: PWM interrupt flag, should be:
 *                 - PWM_INTSTAT_MR0: Interrupt flag for PWM match channel 0
 *                 - PWM_INTSTAT_MR1: Interrupt flag for PWM match channel 1
 *                 - PWM_INTSTAT_MR2: Interrupt flag for PWM match channel 2
 *                 - PWM_INTSTAT_MR3: Interrupt flag for PWM match channel 3
 *                 - PWM_INTSTAT_MR4: Interrupt flag for PWM match channel 4
 *                 - PWM_INTSTAT_MR5: Interrupt flag for PWM match channel 5
 *                 - PWM_INTSTAT_MR6: Interrupt flag for PWM match channel 6
 *                 - PWM_INTSTAT_CAP0: Interrupt flag for capture input 0
 *                 - PWM_INTSTAT_CAP1: Interrupt flag for capture input 1
 * @return         New State of PWM interrupt flag (SET or RESET)
 **********************************************************************/
IntStatus PWM_GetIntStatus(LPC_PWM_TypeDef *PWMx, uint32_t IntFlag)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_PWM_INTSTAT(IntFlag));

    return ((PWMx->IR & IntFlag) ? SET : RESET);
}



/*********************************************************************//**
 * @brief         Clear specified PWM Interrupt pending
 * @param[in]    PWMx: PWM peripheral, should be LPC_PWM1
 * @param[in]    IntFlag: PWM interrupt flag, should be:
 *                 - PWM_INTSTAT_MR0: Interrupt flag for PWM match channel 0
 *                 - PWM_INTSTAT_MR1: Interrupt flag for PWM match channel 1
 *                 - PWM_INTSTAT_MR2: Interrupt flag for PWM match channel 2
 *                 - PWM_INTSTAT_MR3: Interrupt flag for PWM match channel 3
 *                 - PWM_INTSTAT_MR4: Interrupt flag for PWM match channel 4
 *                 - PWM_INTSTAT_MR5: Interrupt flag for PWM match channel 5
 *                 - PWM_INTSTAT_MR6: Interrupt flag for PWM match channel 6
 *                 - PWM_INTSTAT_CAP0: Interrupt flag for capture input 0
 *                 - PWM_INTSTAT_CAP1: Interrupt flag for capture input 1
 * @return         None
 **********************************************************************/
void PWM_ClearIntPending(LPC_PWM_TypeDef *PWMx, uint32_t IntFlag)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_PWM_INTSTAT(IntFlag));
    PWMx->IR = IntFlag;
}



/*****************************************************************************//**
* @brief        Fills each PWM_InitStruct member with its default value:
*                 - If PWMCounterMode = PWM_MODE_TIMER:
*                     + PrescaleOption = PWM_TIMER_PRESCALE_USVAL
*                     + PrescaleValue = 1
*                 - If PWMCounterMode = PWM_MODE_COUNTER:
*                     + CountInputSelect = PWM_COUNTER_PCAP1_0
*                     + CounterOption = PWM_COUNTER_RISING
* @param[in]    PWMTimerCounterMode Timer or Counter mode, should be:
*                 - PWM_MODE_TIMER: Counter of PWM peripheral is in Timer mode
*                 - PWM_MODE_COUNTER: Counter of PWM peripheral is in Counter mode
* @param[in]    PWM_InitStruct Pointer to structure (PWM_TIMERCFG_Type or
*                  PWM_COUNTERCFG_Type) which will be initialized.
* @return        None
* Note: PWM_InitStruct pointer will be assigned to corresponding structure
*         (PWM_TIMERCFG_Type or PWM_COUNTERCFG_Type) due to PWMTimerCounterMode.
*******************************************************************************/
void PWM_ConfigStructInit(uint8_t PWMTimerCounterMode, void *PWM_InitStruct)
{
    PWM_TIMERCFG_Type *pTimeCfg;
    PWM_COUNTERCFG_Type *pCounterCfg;
    CHECK_PARAM(PARAM_PWM_TC_MODE(PWMTimerCounterMode));

    pTimeCfg = (PWM_TIMERCFG_Type *) PWM_InitStruct;
    pCounterCfg = (PWM_COUNTERCFG_Type *) PWM_InitStruct;

    if (PWMTimerCounterMode == PWM_MODE_TIMER )
    {
        pTimeCfg->PrescaleOption = PWM_TIMER_PRESCALE_USVAL;
        pTimeCfg->PrescaleValue = 1;
    }
    else if (PWMTimerCounterMode == PWM_MODE_COUNTER)
    {
        pCounterCfg->CountInputSelect = PWM_COUNTER_PCAP1_0;
        pCounterCfg->CounterOption = PWM_COUNTER_RISING;
    }
}


/*********************************************************************//**
 * @brief         Initializes the PWMx peripheral corresponding to the specified
 *               parameters in the PWM_ConfigStruct.
 * @param[in]    PWMx PWM peripheral, should be LPC_PWM1
 * @param[in]    PWMTimerCounterMode Timer or Counter mode, should be:
 *                 - PWM_MODE_TIMER: Counter of PWM peripheral is in Timer mode
 *                 - PWM_MODE_COUNTER: Counter of PWM peripheral is in Counter mode
 * @param[in]    PWM_ConfigStruct Pointer to structure (PWM_TIMERCFG_Type or
 *                  PWM_COUNTERCFG_Type) which will be initialized.
 * @return         None
 * Note: PWM_ConfigStruct pointer will be assigned to corresponding structure
 *         (PWM_TIMERCFG_Type or PWM_COUNTERCFG_Type) due to PWMTimerCounterMode.
 **********************************************************************/
void PWM_Init(LPC_PWM_TypeDef *PWMx, uint32_t PWMTimerCounterMode, void *PWM_ConfigStruct)
{
    PWM_TIMERCFG_Type *pTimeCfg;
    PWM_COUNTERCFG_Type *pCounterCfg;
    uint64_t clkdlycnt;

    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_PWM_TC_MODE(PWMTimerCounterMode));

    pTimeCfg = (PWM_TIMERCFG_Type *)PWM_ConfigStruct;
    pCounterCfg = (PWM_COUNTERCFG_Type *)PWM_ConfigStruct;


    CLKPWR_ConfigPPWR (CLKPWR_PCONP_PCPWM1, ENABLE);
    CLKPWR_SetPCLKDiv (CLKPWR_PCLKSEL_PWM1, CLKPWR_PCLKSEL_CCLK_DIV_4);
    // Get peripheral clock of PWM1
    clkdlycnt = (uint64_t) CLKPWR_GetPCLK (CLKPWR_PCLKSEL_PWM1);


    // Clear all interrupts pending
    PWMx->IR = 0xFF & PWM_IR_BITMASK;
    PWMx->TCR = 0x00;
    PWMx->CTCR = 0x00;
    PWMx->MCR = 0x00;
    PWMx->CCR = 0x00;
    PWMx->PCR = 0x00;
    PWMx->LER = 0x00;

    if (PWMTimerCounterMode == PWM_MODE_TIMER)
    {
        CHECK_PARAM(PARAM_PWM_TIMER_PRESCALE(pTimeCfg->PrescaleOption));

        /* Absolute prescale value */
        if (pTimeCfg->PrescaleOption == PWM_TIMER_PRESCALE_TICKVAL)
        {
            PWMx->PR   = pTimeCfg->PrescaleValue - 1;
        }
        /* uSecond prescale value */
        else
        {
            clkdlycnt = (clkdlycnt * pTimeCfg->PrescaleValue) / 1000000;
            PWMx->PR = ((uint32_t) clkdlycnt) - 1;
        }

    }
    else if (PWMTimerCounterMode == PWM_MODE_COUNTER)
    {
        CHECK_PARAM(PARAM_PWM_COUNTER_INPUTSEL(pCounterCfg->CountInputSelect));
        CHECK_PARAM(PARAM_PWM_COUNTER_EDGE(pCounterCfg->CounterOption));

        PWMx->CTCR |= (PWM_CTCR_MODE((uint32_t)pCounterCfg->CounterOption)) \
                        | (PWM_CTCR_SELECT_INPUT((uint32_t)pCounterCfg->CountInputSelect));
    }
}

/*********************************************************************//**
 * @brief        De-initializes the PWM peripheral registers to their
*                  default reset values.
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @return         None
 **********************************************************************/
void PWM_DeInit (LPC_PWM_TypeDef *PWMx)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));

    // Disable PWM control (timer, counter and PWM)
    PWMx->TCR = 0x00;
    CLKPWR_ConfigPPWR (CLKPWR_PCONP_PCPWM1, DISABLE);

}


/*********************************************************************//**
 * @brief         Enable/Disable PWM peripheral
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @param[in]    NewState    New State of this function, should be:
 *                             - ENABLE: Enable PWM peripheral
 *                             - DISABLE: Disable PWM peripheral
 * @return         None
 **********************************************************************/
void PWM_Cmd(LPC_PWM_TypeDef *PWMx, FunctionalState NewState)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_FUNCTIONALSTATE(NewState));

    if (NewState == ENABLE)
    {
        PWMx->TCR    |=  PWM_TCR_PWM_ENABLE;
    }
    else
    {
        PWMx->TCR &= (~PWM_TCR_PWM_ENABLE) & PWM_TCR_BITMASK;
    }
}


/*********************************************************************//**
 * @brief         Enable/Disable Counter in PWM peripheral
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @param[in]    NewState New State of this function, should be:
 *                             - ENABLE: Enable Counter in PWM peripheral
 *                             - DISABLE: Disable Counter in PWM peripheral
 * @return         None
 **********************************************************************/
void PWM_CounterCmd(LPC_PWM_TypeDef *PWMx, FunctionalState NewState)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_FUNCTIONALSTATE(NewState));
    if (NewState == ENABLE)
    {
        PWMx->TCR    |=  PWM_TCR_COUNTER_ENABLE;
    }
    else
    {
        PWMx->TCR &= (~PWM_TCR_COUNTER_ENABLE) & PWM_TCR_BITMASK;
    }
}


/*********************************************************************//**
 * @brief         Reset Counter in PWM peripheral
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @return         None
 **********************************************************************/
void PWM_ResetCounter(LPC_PWM_TypeDef *PWMx)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    PWMx->TCR |= PWM_TCR_COUNTER_RESET;
    PWMx->TCR &= (~PWM_TCR_COUNTER_RESET) & PWM_TCR_BITMASK;
}


/*********************************************************************//**
 * @brief         Configures match for PWM peripheral
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @param[in]   PWM_MatchConfigStruct    Pointer to a PWM_MATCHCFG_Type structure
*                    that contains the configuration information for the
*                    specified PWM match function.
 * @return         None
 **********************************************************************/
void PWM_ConfigMatch(LPC_PWM_TypeDef *PWMx, PWM_MATCHCFG_Type *PWM_MatchConfigStruct)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_PWM1_MATCH_CHANNEL(PWM_MatchConfigStruct->MatchChannel));
    CHECK_PARAM(PARAM_FUNCTIONALSTATE(PWM_MatchConfigStruct->IntOnMatch));
    CHECK_PARAM(PARAM_FUNCTIONALSTATE(PWM_MatchConfigStruct->ResetOnMatch));
    CHECK_PARAM(PARAM_FUNCTIONALSTATE(PWM_MatchConfigStruct->StopOnMatch));

    //interrupt on MRn
    if (PWM_MatchConfigStruct->IntOnMatch == ENABLE)
    {
        PWMx->MCR |= PWM_MCR_INT_ON_MATCH(PWM_MatchConfigStruct->MatchChannel);
    }
    else
    {
        PWMx->MCR &= (~PWM_MCR_INT_ON_MATCH(PWM_MatchConfigStruct->MatchChannel)) \
                    & PWM_MCR_BITMASK;
    }

    //reset on MRn
    if (PWM_MatchConfigStruct->ResetOnMatch == ENABLE)
    {
        PWMx->MCR |= PWM_MCR_RESET_ON_MATCH(PWM_MatchConfigStruct->MatchChannel);
    }
    else
    {
        PWMx->MCR &= (~PWM_MCR_RESET_ON_MATCH(PWM_MatchConfigStruct->MatchChannel)) \
                    & PWM_MCR_BITMASK;
    }

    //stop on MRn
    if (PWM_MatchConfigStruct->StopOnMatch == ENABLE)
    {
        PWMx->MCR |= PWM_MCR_STOP_ON_MATCH(PWM_MatchConfigStruct->MatchChannel);
    }
    else
    {
        PWMx->MCR &= (~PWM_MCR_STOP_ON_MATCH(PWM_MatchConfigStruct->MatchChannel)) \
                    & PWM_MCR_BITMASK;
    }
}


/*********************************************************************//**
 * @brief         Configures capture input for PWM peripheral
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @param[in]   PWM_CaptureConfigStruct    Pointer to a PWM_CAPTURECFG_Type structure
*                    that contains the configuration information for the
*                    specified PWM capture input function.
 * @return         None
 **********************************************************************/
void PWM_ConfigCapture(LPC_PWM_TypeDef *PWMx, PWM_CAPTURECFG_Type *PWM_CaptureConfigStruct)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_PWM1_CAPTURE_CHANNEL(PWM_CaptureConfigStruct->CaptureChannel));
    CHECK_PARAM(PARAM_FUNCTIONALSTATE(PWM_CaptureConfigStruct->FallingEdge));
    CHECK_PARAM(PARAM_FUNCTIONALSTATE(PWM_CaptureConfigStruct->IntOnCaption));
    CHECK_PARAM(PARAM_FUNCTIONALSTATE(PWM_CaptureConfigStruct->RisingEdge));

    if (PWM_CaptureConfigStruct->RisingEdge == ENABLE)
    {
        PWMx->CCR |= PWM_CCR_CAP_RISING(PWM_CaptureConfigStruct->CaptureChannel);
    }
    else
    {
        PWMx->CCR &= (~PWM_CCR_CAP_RISING(PWM_CaptureConfigStruct->CaptureChannel)) \
                    & PWM_CCR_BITMASK;
    }

    if (PWM_CaptureConfigStruct->FallingEdge == ENABLE)
    {
        PWMx->CCR |= PWM_CCR_CAP_FALLING(PWM_CaptureConfigStruct->CaptureChannel);
    }
    else
    {
        PWMx->CCR &= (~PWM_CCR_CAP_FALLING(PWM_CaptureConfigStruct->CaptureChannel)) \
                    & PWM_CCR_BITMASK;
    }

    if (PWM_CaptureConfigStruct->IntOnCaption == ENABLE)
    {
        PWMx->CCR |= PWM_CCR_INT_ON_CAP(PWM_CaptureConfigStruct->CaptureChannel);
    }
    else
    {
        PWMx->CCR &= (~PWM_CCR_INT_ON_CAP(PWM_CaptureConfigStruct->CaptureChannel)) \
                    & PWM_CCR_BITMASK;
    }
}


/*********************************************************************//**
 * @brief         Read value of capture register PWM peripheral
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @param[in]    CaptureChannel: capture channel number, should be in
 *                 range 0 to 1
 * @return         Value of capture register
 **********************************************************************/
uint32_t PWM_GetCaptureValue(LPC_PWM_TypeDef *PWMx, uint8_t CaptureChannel)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_PWM1_CAPTURE_CHANNEL(CaptureChannel));

    switch (CaptureChannel)
    {
    case 0:
        return PWMx->CR0;

    case 1:
        return PWMx->CR1;

    default:
        return (0);
    }
}


/********************************************************************//**
 * @brief         Update value for each PWM channel with update type option
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @param[in]    MatchChannel Match channel
 * @param[in]    MatchValue Match value
 * @param[in]    UpdateType Type of Update, should be:
 *                 - PWM_MATCH_UPDATE_NOW: The update value will be updated for
 *                     this channel immediately
 *                 - PWM_MATCH_UPDATE_NEXT_RST: The update value will be updated for
 *                     this channel on next reset by a PWM Match event.
 * @return        None
 *********************************************************************/
void PWM_MatchUpdate(LPC_PWM_TypeDef *PWMx, uint8_t MatchChannel, \
                    uint32_t MatchValue, uint8_t UpdateType)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_PWM1_MATCH_CHANNEL(MatchChannel));
    CHECK_PARAM(PARAM_PWM_MATCH_UPDATE(UpdateType));

    switch (MatchChannel)
    {
    case 0:
        PWMx->MR0 = MatchValue;
        break;

    case 1:
        PWMx->MR1 = MatchValue;
        break;

    case 2:
        PWMx->MR2 = MatchValue;
        break;

    case 3:
        PWMx->MR3 = MatchValue;
        break;

    case 4:
        PWMx->MR4 = MatchValue;
        break;

    case 5:
        PWMx->MR5 = MatchValue;
        break;

    case 6:
        PWMx->MR6 = MatchValue;
        break;
    }

    // Write Latch register
    PWMx->LER |= PWM_LER_EN_MATCHn_LATCH(MatchChannel);

    // In case of update now
    if (UpdateType == PWM_MATCH_UPDATE_NOW)
    {
        PWMx->TCR |= PWM_TCR_COUNTER_RESET;
        PWMx->TCR &= (~PWM_TCR_COUNTER_RESET) & PWM_TCR_BITMASK;
    }
}


/********************************************************************//**
 * @brief         Configure Edge mode for each PWM channel
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @param[in]    PWMChannel PWM channel, should be in range from 2 to 6
 * @param[in]    ModeOption PWM mode option, should be:
 *                 - PWM_CHANNEL_SINGLE_EDGE: Single Edge mode
 *                 - PWM_CHANNEL_DUAL_EDGE: Dual Edge mode
 * @return         None
 * Note: PWM Channel 1 can not be selected for mode option
 *********************************************************************/
void PWM_ChannelConfig(LPC_PWM_TypeDef *PWMx, uint8_t PWMChannel, uint8_t ModeOption)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_PWM1_EDGE_MODE_CHANNEL(PWMChannel));
    CHECK_PARAM(PARAM_PWM_CHANNEL_EDGE(ModeOption));

    // Single edge mode
    if (ModeOption == PWM_CHANNEL_SINGLE_EDGE)
    {
        PWMx->PCR &= (~PWM_PCR_PWMSELn(PWMChannel)) & PWM_PCR_BITMASK;
    }
    // Double edge mode
    else if (PWM_CHANNEL_DUAL_EDGE)
    {
        PWMx->PCR |= PWM_PCR_PWMSELn(PWMChannel);
    }
}



/********************************************************************//**
 * @brief         Enable/Disable PWM channel output
 * @param[in]    PWMx    PWM peripheral selected, should be LPC_PWM1
 * @param[in]    PWMChannel PWM channel, should be in range from 1 to 6
 * @param[in]    NewState New State of this function, should be:
 *                 - ENABLE: Enable this PWM channel output
 *                 - DISABLE: Disable this PWM channel output
 * @return        None
 *********************************************************************/
void PWM_ChannelCmd(LPC_PWM_TypeDef *PWMx, uint8_t PWMChannel, FunctionalState NewState)
{
    CHECK_PARAM(PARAM_PWMx(PWMx));
    CHECK_PARAM(PARAM_PWM1_CHANNEL(PWMChannel));

    if (NewState == ENABLE)
    {
        PWMx->PCR |= PWM_PCR_PWMENAn(PWMChannel);
    }
    else
    {
        PWMx->PCR &= (~PWM_PCR_PWMENAn(PWMChannel)) & PWM_PCR_BITMASK;
    }
}

/**
 * @}
 */

#endif /* _PWM */

/**
 * @}
 */

/* --------------------------------- End Of File ------------------------------ */