1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/drivers/include/periph/pwm.h

191 lines
6.2 KiB
C
Raw Normal View History

/*
* Copyright (C) 2014-2015 Freie Universität Berlin
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @defgroup drivers_periph_pwm PWM
* @ingroup drivers_periph
2014-02-04 18:52:53 +01:00
* @brief Low-level PWM peripheral driver
*
* This interface enables access to CPU peripherals generating PWM signals. On
* most platforms, this interface will be implemented based on hardware timers,
* though some CPUs provide dedicated PWM peripherals.
*
* The characteristics of a PWM signal can be defined by three basic parameters,
* namely the frequency, the duty cycle, and the operational mode. This
* interface supports basic PWM generation in left-aligned, right-aligned, and
2017-02-13 09:37:16 +01:00
* center mode. Additionally the interface supports the definition of the used
* resolution, defining the granularity with which one can specify the duty
* cycle. This brings more flexibility to the configuration of the frequency,
* especially on systems with low system clocks.
*
* Typically, a single PWM device (e.g. hardware timer) supports PWM signal
* generation on multiple pins in parallel. While the duty cycle is selectable
* for each channel individually, the frequency and resolution are shared for
* all channels.
*
2017-02-13 09:37:16 +01:00
* The mapping/configuration of PWM devices (timers) and the used pins has to be
* done in the board configuration (the board's `periph_conf.h).
*
2017-02-13 09:37:16 +01:00
* When using the PWM interface, first thing you have to do is initialize the
* PWM device with the targeted mode, frequency, and resolution settings. Once
* the device is initialized, it will start the generation of PWM signals on all
* configured pins immediately, with an initial duty cycle of `0`. Use the
2017-02-13 09:37:16 +01:00
* pwm_set() function to change the duty cycle for a given channel. If you
* want to disable the PWM generation again, simply call pwm_poweroff().
*
* ## (Low-)Power implications
*
* After initialization, the a PWM peripheral **should** be powered on and
* active. When manually stopped using the pwm_poweroff() function, the PWM
* generation **should** be stopped for all channels and the PWM peripheral
* **should** be fully power off (e.g. through peripheral clock gating). Once
* being re-enabled by calling the pwm_poweron() function, the PWM peripheral
* **should** transparently continue its previously configured operation,
* including the last active duty cycle values.
*
* While a PWM device is active, some implementations might need to block
* certain power modes.
*
* @{
* @file
2014-02-04 18:52:53 +01:00
* @brief Low-level PWM peripheral driver interface definitions
*
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
*/
#ifndef PERIPH_PWM_H
#define PERIPH_PWM_H
#include <stdint.h>
#include <limits.h>
#include "periph_cpu.h"
#include "periph_conf.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief Default PWM access macro
*/
#ifndef PWM_DEV
#define PWM_DEV(x) (x)
#endif
/**
* @brief Default PWM undefined value
*/
#ifndef PWM_UNDEF
#define PWM_UNDEF (UINT_FAST8_MAX)
#endif
/**
* @brief Default PWM type definition
*/
#ifndef HAVE_PWM_T
typedef uint_fast8_t pwm_t;
#endif
/**
* @brief Default PWM mode definition
*/
#ifndef HAVE_PWM_MODE_T
typedef enum {
PWM_LEFT, /*< use left aligned PWM */
PWM_RIGHT, /*< use right aligned PWM */
PWM_CENTER /*< use center aligned PWM */
} pwm_mode_t;
#endif
#ifdef MODULE_ARDUINO
/**
boards,sys/arduino: major clean up - Rename all `arduino_pinmap.h` to `arduino_iomap.h` - An empty `arduino_pinmap.h` that just includes `arduino_iomap.h` is provided for backward compatibility - Move all info from `arduino_board.h` into the new file as trivial macros, so that they can also be used outside of sketches - The new name reflects the fact not just pin mappings, but also other I/O features such as PWMs are mapped - Drop all `arduino_board.h` - `arduino_board.h` and `arduino_iomap.h` now provide the exact same information, just in a different format - a generic `arduino_board.h` is provided instead that just uses the info in `arduinio_iomap.h` and provides them in the format the code in `sys/arduino` expects it - Add fine grained features to indicate for mappings - availability of mappings for analog pins, DAC pins, PWM pins, UART devices, SPI/I2C buses to the corresponding RIOT identification can now be expressed: - `arduino_pins`: `ARDUINO_PIN_0` etc. are available - `arduino_analog`: `ARDUINO_A0` etc. are available - `arduino_pwm`: `ARDUINO_PIN_13_PWM_DEV` etc. are available - `arduino_dac`: `ARDUINO_DAC0` etc. are available - `arduino_uart`: `ARDUINO_UART_D0D1` or similar are available - `arduino_spi`: `ARDUINO_SPI_ISP` or similar are available - `arduino_i2c`: `ARDUINO_I2C_UNO` or similar are available - mechanical/electrical compatibility with specific form factors can now be expressed as features: - `aruino_shield_nano`: Arduino NANO compatible headers - `aruino_shield_uno`: Arduino UNO compatible headers - `aruino_shield_mega`: Arduino MEGA compatible headers - `aruino_shield_isp`: ISP header is available This provides the groundwork to implement shield support as modules that can rely on the I/O mappings, rather than having to provide a configuration per board.
2023-06-23 15:42:08 +02:00
* @brief Mapping of an Arduino digital pin to the corresponding PWM dev and
* channel pair
*/
typedef struct {
boards,sys/arduino: major clean up - Rename all `arduino_pinmap.h` to `arduino_iomap.h` - An empty `arduino_pinmap.h` that just includes `arduino_iomap.h` is provided for backward compatibility - Move all info from `arduino_board.h` into the new file as trivial macros, so that they can also be used outside of sketches - The new name reflects the fact not just pin mappings, but also other I/O features such as PWMs are mapped - Drop all `arduino_board.h` - `arduino_board.h` and `arduino_iomap.h` now provide the exact same information, just in a different format - a generic `arduino_board.h` is provided instead that just uses the info in `arduinio_iomap.h` and provides them in the format the code in `sys/arduino` expects it - Add fine grained features to indicate for mappings - availability of mappings for analog pins, DAC pins, PWM pins, UART devices, SPI/I2C buses to the corresponding RIOT identification can now be expressed: - `arduino_pins`: `ARDUINO_PIN_0` etc. are available - `arduino_analog`: `ARDUINO_A0` etc. are available - `arduino_pwm`: `ARDUINO_PIN_13_PWM_DEV` etc. are available - `arduino_dac`: `ARDUINO_DAC0` etc. are available - `arduino_uart`: `ARDUINO_UART_D0D1` or similar are available - `arduino_spi`: `ARDUINO_SPI_ISP` or similar are available - `arduino_i2c`: `ARDUINO_I2C_UNO` or similar are available - mechanical/electrical compatibility with specific form factors can now be expressed as features: - `aruino_shield_nano`: Arduino NANO compatible headers - `aruino_shield_uno`: Arduino UNO compatible headers - `aruino_shield_mega`: Arduino MEGA compatible headers - `aruino_shield_isp`: ISP header is available This provides the groundwork to implement shield support as modules that can rely on the I/O mappings, rather than having to provide a configuration per board.
2023-06-23 15:42:08 +02:00
pwm_t dev; /**< PWM device connected to the pin */
uint8_t chan; /**< PWM channel index */
uint8_t pin; /**< Arduino pin number */
} arduino_pwm_t;
#endif
/**
* @brief Initialize a PWM device
2014-05-14 10:46:15 +02:00
*
* The PWM module is based on virtual PWM devices, which can have one or more
* channels. The PWM devices can be configured to run with a given frequency and
* resolution, which are always identical for the complete device, hence for
* every channel on a device.
2014-05-14 10:46:15 +02:00
*
* The desired frequency and resolution may not be possible on a given device
* when chosen too large. In this case the PWM driver will always keep the
* resolution and decrease the frequency if needed. To verify the correct
* settings compare the returned value which is the actually set frequency.
2014-05-14 10:46:15 +02:00
*
* @param[in] dev PWM device to initialize
* @param[in] mode PWM mode, left, right or center aligned
* @param[in] freq PWM frequency in Hz
* @param[in] res PWM resolution
*
* @return actual PWM frequency on success
* @return 0 on error
*/
uint32_t pwm_init(pwm_t dev, pwm_mode_t mode, uint32_t freq, uint16_t res);
/**
* @brief Get the number of available channels
2014-05-14 10:46:15 +02:00
*
* @param[in] dev PWM device
*
* @return Number of channels available for the given device
*/
uint8_t pwm_channels(pwm_t dev);
/**
* @brief Set the duty-cycle for a given channel of the given PWM device
*
* The duty-cycle is set in relation to the chosen resolution of the given
* device. If value > resolution, value is set to resolution.
2014-05-14 10:46:15 +02:00
*
* @param[in] dev the PWM device to set
* @param[in] channel the channel of the given device to set
* @param[in] value the desired duty-cycle to set
*/
void pwm_set(pwm_t dev, uint8_t channel, uint16_t value);
/**
* @brief Resume PWM generation on the given device
2014-05-14 10:46:15 +02:00
*
* When this function is called, the given PWM device is powered on and
2017-02-13 09:37:16 +01:00
* continues its previously configured operation. The duty cycle of each channel
* will be the value that was last set.
*
* This function must not be called before the PWM device was initialized.
*
* @param[in] dev device to start
*/
void pwm_poweron(pwm_t dev);
/**
* @brief Stop PWM generation on the given device
*
2017-02-13 09:37:16 +01:00
* This function stops the PWM generation on all configured channels for the
* given device and powers down the given PWM peripheral.
*
* @param[in] dev device to stop
*/
void pwm_poweroff(pwm_t dev);
#ifdef __cplusplus
}
#endif
#endif /* PERIPH_PWM_H */
/** @} */