mbed library sources. Supersedes mbed-src.

Dependents:   Nucleo_Hello_Encoder BLE_iBeaconScan AM1805_DEMO DISCO-F429ZI_ExportTemplate1 ... more

Committer:
AnnaBridge
Date:
Wed Feb 20 22:31:08 2019 +0000
Revision:
189:f392fc9709a3
Parent:
188:bcfe06ba3d64
mbed library release version 165

Who changed what in which revision?

UserRevisionLine numberNew contents of line
<> 144:ef7eb2e8f9f7 1 /***************************************************************************//**
<> 144:ef7eb2e8f9f7 2 * @file pwmout_api.c
<> 144:ef7eb2e8f9f7 3 *******************************************************************************
<> 144:ef7eb2e8f9f7 4 * @section License
<> 144:ef7eb2e8f9f7 5 * <b>(C) Copyright 2015 Silicon Labs, http://www.silabs.com</b>
<> 144:ef7eb2e8f9f7 6 *******************************************************************************
<> 144:ef7eb2e8f9f7 7 *
<> 144:ef7eb2e8f9f7 8 * SPDX-License-Identifier: Apache-2.0
<> 144:ef7eb2e8f9f7 9 *
<> 144:ef7eb2e8f9f7 10 * Licensed under the Apache License, Version 2.0 (the "License"); you may
<> 144:ef7eb2e8f9f7 11 * not use this file except in compliance with the License.
<> 144:ef7eb2e8f9f7 12 * You may obtain a copy of the License at
<> 144:ef7eb2e8f9f7 13 *
<> 144:ef7eb2e8f9f7 14 * http://www.apache.org/licenses/LICENSE-2.0
<> 144:ef7eb2e8f9f7 15 *
<> 144:ef7eb2e8f9f7 16 * Unless required by applicable law or agreed to in writing, software
<> 144:ef7eb2e8f9f7 17 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
<> 144:ef7eb2e8f9f7 18 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
<> 144:ef7eb2e8f9f7 19 * See the License for the specific language governing permissions and
<> 144:ef7eb2e8f9f7 20 * limitations under the License.
<> 144:ef7eb2e8f9f7 21 *
<> 144:ef7eb2e8f9f7 22 ******************************************************************************/
<> 144:ef7eb2e8f9f7 23
<> 144:ef7eb2e8f9f7 24 #include "device.h"
<> 144:ef7eb2e8f9f7 25 #include "clocking.h"
<> 144:ef7eb2e8f9f7 26 #if DEVICE_PWMOUT
<> 144:ef7eb2e8f9f7 27
<> 144:ef7eb2e8f9f7 28 #include "mbed_assert.h"
AnnaBridge 184:08ed48f1de7f 29 #include "mbed_power_mgmt.h"
<> 144:ef7eb2e8f9f7 30 #include "pwmout_api.h"
<> 144:ef7eb2e8f9f7 31 #include "pinmap.h"
<> 144:ef7eb2e8f9f7 32 #include "PeripheralPins.h"
<> 144:ef7eb2e8f9f7 33 #include "device_peripherals.h"
<> 144:ef7eb2e8f9f7 34
<> 144:ef7eb2e8f9f7 35 #include "em_cmu.h"
<> 144:ef7eb2e8f9f7 36 #include "em_gpio.h"
<> 144:ef7eb2e8f9f7 37 #include "em_timer.h"
<> 144:ef7eb2e8f9f7 38
<> 144:ef7eb2e8f9f7 39 static uint32_t pwm_prescaler_div;
<> 144:ef7eb2e8f9f7 40
<> 144:ef7eb2e8f9f7 41 float pwmout_calculate_duty(uint32_t width_cycles, uint32_t period_cycles);
<> 144:ef7eb2e8f9f7 42 void pwmout_write_channel(uint32_t channel, float value);
<> 144:ef7eb2e8f9f7 43
<> 144:ef7eb2e8f9f7 44 uint32_t pwmout_get_channel_route(uint32_t channel)
<> 144:ef7eb2e8f9f7 45 {
<> 144:ef7eb2e8f9f7 46 MBED_ASSERT(channel != (PWMName) NC);
<> 144:ef7eb2e8f9f7 47
<> 144:ef7eb2e8f9f7 48 switch (channel) {
<> 144:ef7eb2e8f9f7 49 #ifdef TIMER_ROUTEPEN_CC0PEN
<> 144:ef7eb2e8f9f7 50 case PWM_CH0:
<> 144:ef7eb2e8f9f7 51 return TIMER_ROUTEPEN_CC0PEN;
<> 144:ef7eb2e8f9f7 52 break;
<> 144:ef7eb2e8f9f7 53 case PWM_CH1:
<> 144:ef7eb2e8f9f7 54 return TIMER_ROUTEPEN_CC1PEN;
<> 144:ef7eb2e8f9f7 55 break;
<> 144:ef7eb2e8f9f7 56 case PWM_CH2:
<> 144:ef7eb2e8f9f7 57 return TIMER_ROUTEPEN_CC2PEN;
<> 144:ef7eb2e8f9f7 58 break;
<> 144:ef7eb2e8f9f7 59 case PWM_CH3:
<> 144:ef7eb2e8f9f7 60 return TIMER_ROUTEPEN_CC3PEN;
<> 144:ef7eb2e8f9f7 61 break;
<> 144:ef7eb2e8f9f7 62 #else
<> 144:ef7eb2e8f9f7 63 case PWM_CH0:
<> 144:ef7eb2e8f9f7 64 return TIMER_ROUTE_CC0PEN;
<> 144:ef7eb2e8f9f7 65 break;
<> 144:ef7eb2e8f9f7 66 case PWM_CH1:
<> 144:ef7eb2e8f9f7 67 return TIMER_ROUTE_CC1PEN;
<> 144:ef7eb2e8f9f7 68 break;
<> 144:ef7eb2e8f9f7 69 case PWM_CH2:
<> 144:ef7eb2e8f9f7 70 return TIMER_ROUTE_CC2PEN;
<> 144:ef7eb2e8f9f7 71 break;
<> 144:ef7eb2e8f9f7 72 #endif
<> 144:ef7eb2e8f9f7 73 default:
<> 144:ef7eb2e8f9f7 74 return 0;
<> 144:ef7eb2e8f9f7 75 }
<> 144:ef7eb2e8f9f7 76 }
<> 144:ef7eb2e8f9f7 77
<> 144:ef7eb2e8f9f7 78 /*
<> 144:ef7eb2e8f9f7 79 * Disables the route location given. Returns true if it was enabled, false if it wasn't.
<> 144:ef7eb2e8f9f7 80 */
<> 144:ef7eb2e8f9f7 81 bool pwmout_disable_channel_route(uint32_t routeloc) {
<> 144:ef7eb2e8f9f7 82 #ifdef TIMER_ROUTEPEN_CC0PEN
<> 144:ef7eb2e8f9f7 83 if(PWM_TIMER->ROUTEPEN & routeloc) {
<> 144:ef7eb2e8f9f7 84 //This channel was in use, so disable
<> 144:ef7eb2e8f9f7 85 PWM_TIMER->ROUTEPEN &= ~routeloc;
<> 144:ef7eb2e8f9f7 86 return true;
<> 144:ef7eb2e8f9f7 87 }
<> 144:ef7eb2e8f9f7 88 #else
<> 144:ef7eb2e8f9f7 89 if(PWM_TIMER->ROUTE & routeloc) {
<> 144:ef7eb2e8f9f7 90 //This channel was in use, so disable
<> 144:ef7eb2e8f9f7 91 PWM_TIMER->ROUTE &= ~routeloc;
<> 144:ef7eb2e8f9f7 92 return true;
<> 144:ef7eb2e8f9f7 93 }
<> 144:ef7eb2e8f9f7 94 #endif
<> 144:ef7eb2e8f9f7 95 return false;
<> 144:ef7eb2e8f9f7 96 }
<> 144:ef7eb2e8f9f7 97
<> 144:ef7eb2e8f9f7 98 /*
<> 144:ef7eb2e8f9f7 99 * Check if a channel is active
<> 144:ef7eb2e8f9f7 100 */
<> 144:ef7eb2e8f9f7 101 bool pwmout_channel_route_active(uint32_t routeloc) {
<> 144:ef7eb2e8f9f7 102 #ifdef TIMER_ROUTEPEN_CC0PEN
<> 144:ef7eb2e8f9f7 103 if(PWM_TIMER->ROUTEPEN & routeloc) {
<> 144:ef7eb2e8f9f7 104 return true;
<> 144:ef7eb2e8f9f7 105 }
<> 144:ef7eb2e8f9f7 106 #else
<> 144:ef7eb2e8f9f7 107 if(PWM_TIMER->ROUTE & routeloc) {
<> 144:ef7eb2e8f9f7 108 return true;
<> 144:ef7eb2e8f9f7 109 }
<> 144:ef7eb2e8f9f7 110 #endif
<> 144:ef7eb2e8f9f7 111 return false;
<> 144:ef7eb2e8f9f7 112 }
<> 144:ef7eb2e8f9f7 113
<> 144:ef7eb2e8f9f7 114 /*
<> 144:ef7eb2e8f9f7 115 * Set the given route PEN flag
<> 144:ef7eb2e8f9f7 116 */
<> 144:ef7eb2e8f9f7 117 void pwmout_set_channel_route(uint32_t routeloc) {
<> 144:ef7eb2e8f9f7 118 #ifdef TIMER_ROUTEPEN_CC0PEN
<> 144:ef7eb2e8f9f7 119 PWM_TIMER->ROUTEPEN |= routeloc;
<> 144:ef7eb2e8f9f7 120 #else
<> 144:ef7eb2e8f9f7 121 PWM_TIMER->ROUTE |= routeloc;
<> 144:ef7eb2e8f9f7 122 #endif
<> 144:ef7eb2e8f9f7 123 }
<> 144:ef7eb2e8f9f7 124
<> 144:ef7eb2e8f9f7 125 /*
<> 144:ef7eb2e8f9f7 126 * Check if all routes are disabled
<> 144:ef7eb2e8f9f7 127 */
<> 144:ef7eb2e8f9f7 128 bool pwmout_all_inactive(void) {
<> 144:ef7eb2e8f9f7 129 #ifdef TIMER_ROUTEPEN_CC0PEN
<> 144:ef7eb2e8f9f7 130 if(PWM_TIMER->ROUTEPEN == _TIMER_ROUTEPEN_RESETVALUE) {
<> 144:ef7eb2e8f9f7 131 return true;
<> 144:ef7eb2e8f9f7 132 }
<> 144:ef7eb2e8f9f7 133 #else
Anna Bridge 163:74e0ce7f98e8 134 if (!(PWM_TIMER->ROUTE & (TIMER_ROUTE_CC0PEN | TIMER_ROUTE_CC1PEN | TIMER_ROUTE_CC2PEN))) {
<> 144:ef7eb2e8f9f7 135 return true;
<> 144:ef7eb2e8f9f7 136 }
<> 144:ef7eb2e8f9f7 137 #endif
<> 144:ef7eb2e8f9f7 138 return false;
<> 144:ef7eb2e8f9f7 139 }
<> 144:ef7eb2e8f9f7 140
<> 144:ef7eb2e8f9f7 141 void pwmout_enable_pins(pwmout_t *obj, uint8_t enable)
<> 144:ef7eb2e8f9f7 142 {
<> 144:ef7eb2e8f9f7 143 if (enable) {
<> 144:ef7eb2e8f9f7 144 pin_mode(obj->pin, PushPull);
<> 144:ef7eb2e8f9f7 145 } else {
<> 144:ef7eb2e8f9f7 146 // TODO_LP return PinMode to the previous state
<> 144:ef7eb2e8f9f7 147 pin_mode(obj->pin, Disabled);
<> 144:ef7eb2e8f9f7 148 }
<> 144:ef7eb2e8f9f7 149 }
<> 144:ef7eb2e8f9f7 150
<> 144:ef7eb2e8f9f7 151 void pwmout_enable(pwmout_t *obj, uint8_t enable){
<> 144:ef7eb2e8f9f7 152 if (enable) {
<> 144:ef7eb2e8f9f7 153 // Set mode to PWM
<> 144:ef7eb2e8f9f7 154 PWM_TIMER->CC[obj->channel].CTRL = TIMER_CC_CTRL_MODE_PWM;
<> 144:ef7eb2e8f9f7 155 } else {
<> 144:ef7eb2e8f9f7 156 // Set mode to default (== disabled)
<> 144:ef7eb2e8f9f7 157 PWM_TIMER->CC[obj->channel].CTRL = _TIMER_CC_CTRL_MODE_DEFAULT;
<> 144:ef7eb2e8f9f7 158 }
<> 144:ef7eb2e8f9f7 159 }
<> 144:ef7eb2e8f9f7 160
<> 144:ef7eb2e8f9f7 161 void pwmout_init(pwmout_t *obj, PinName pin)
<> 144:ef7eb2e8f9f7 162 {
<> 144:ef7eb2e8f9f7 163 obj->channel = (PWMName) pinmap_peripheral(pin, PinMap_PWM);
<> 144:ef7eb2e8f9f7 164 obj->pin = pin;
<> 144:ef7eb2e8f9f7 165 MBED_ASSERT(obj->channel != (PWMName) NC);
<> 144:ef7eb2e8f9f7 166
<> 144:ef7eb2e8f9f7 167 /* Turn on clock */
<> 144:ef7eb2e8f9f7 168 CMU_ClockEnable(PWM_TIMER_CLOCK, true);
<> 144:ef7eb2e8f9f7 169
<> 144:ef7eb2e8f9f7 170 /* Turn on timer */
<> 144:ef7eb2e8f9f7 171 if(!(PWM_TIMER->STATUS & TIMER_STATUS_RUNNING)) {
<> 144:ef7eb2e8f9f7 172 TIMER_Init_TypeDef timerInit = TIMER_INIT_DEFAULT;
<> 144:ef7eb2e8f9f7 173 TIMER_Init(PWM_TIMER, &timerInit);
<> 144:ef7eb2e8f9f7 174 }
<> 144:ef7eb2e8f9f7 175
<> 144:ef7eb2e8f9f7 176 // Set route enable
<> 144:ef7eb2e8f9f7 177 if(pwmout_channel_route_active(pwmout_get_channel_route(obj->channel))) {
<> 144:ef7eb2e8f9f7 178 //This channel was already in use
<> 144:ef7eb2e8f9f7 179 //TODO: gracefully handle this case. mbed_error?
<> 144:ef7eb2e8f9f7 180 return;
<> 144:ef7eb2e8f9f7 181 } else {
<> 144:ef7eb2e8f9f7 182 pwmout_set_channel_route(pwmout_get_channel_route(obj->channel));
Anna Bridge 180:96ed750bd169 183 sleep_manager_lock_deep_sleep();
<> 144:ef7eb2e8f9f7 184 pwmout_enable(obj, true);
<> 144:ef7eb2e8f9f7 185 pwmout_enable_pins(obj, true);
<> 144:ef7eb2e8f9f7 186 }
<> 144:ef7eb2e8f9f7 187
<> 144:ef7eb2e8f9f7 188 // Set route location
<> 144:ef7eb2e8f9f7 189 #ifdef _TIMER_ROUTELOC0_CC0LOC_LOC0
<> 144:ef7eb2e8f9f7 190 switch (obj->channel) {
<> 144:ef7eb2e8f9f7 191 case PWM_CH0:
<> 144:ef7eb2e8f9f7 192 PWM_TIMER->ROUTELOC0 &= ~_TIMER_ROUTELOC0_CC0LOC_MASK;
<> 144:ef7eb2e8f9f7 193 PWM_TIMER->ROUTELOC0 |= pinmap_find_function(pin,PinMap_PWM) << _TIMER_ROUTELOC0_CC0LOC_SHIFT;
<> 144:ef7eb2e8f9f7 194 break;
<> 144:ef7eb2e8f9f7 195 case PWM_CH1:
<> 144:ef7eb2e8f9f7 196 PWM_TIMER->ROUTELOC0 &= ~_TIMER_ROUTELOC0_CC1LOC_MASK;
<> 144:ef7eb2e8f9f7 197 PWM_TIMER->ROUTELOC0 |= pinmap_find_function(pin,PinMap_PWM)<< _TIMER_ROUTELOC0_CC1LOC_SHIFT;
<> 144:ef7eb2e8f9f7 198 break;
<> 144:ef7eb2e8f9f7 199 case PWM_CH2:
<> 144:ef7eb2e8f9f7 200 PWM_TIMER->ROUTELOC0 &= ~_TIMER_ROUTELOC0_CC2LOC_MASK;
<> 144:ef7eb2e8f9f7 201 PWM_TIMER->ROUTELOC0 |= pinmap_find_function(pin,PinMap_PWM) << _TIMER_ROUTELOC0_CC2LOC_SHIFT;
<> 144:ef7eb2e8f9f7 202 break;
<> 144:ef7eb2e8f9f7 203 case PWM_CH3:
<> 144:ef7eb2e8f9f7 204 PWM_TIMER->ROUTELOC0 &= ~_TIMER_ROUTELOC0_CC3LOC_MASK;
<> 144:ef7eb2e8f9f7 205 PWM_TIMER->ROUTELOC0 |= pinmap_find_function(pin,PinMap_PWM) << _TIMER_ROUTELOC0_CC3LOC_SHIFT;
<> 144:ef7eb2e8f9f7 206 break;
<> 144:ef7eb2e8f9f7 207 default:
<> 144:ef7eb2e8f9f7 208 MBED_ASSERT(false);
<> 144:ef7eb2e8f9f7 209 }
<> 144:ef7eb2e8f9f7 210 #else
<> 144:ef7eb2e8f9f7 211 // On P1, the route location is statically defined for the entire timer.
<> 144:ef7eb2e8f9f7 212 PWM_TIMER->ROUTE &= ~_TIMER_ROUTE_LOCATION_MASK;
Anna Bridge 163:74e0ce7f98e8 213 // Make sure the route location is not overwritten
Anna Bridge 163:74e0ce7f98e8 214 if(pwmout_all_inactive()) {
<> 144:ef7eb2e8f9f7 215 PWM_TIMER->ROUTE |= pinmap_find_function(pin,PinMap_PWM) << _TIMER_ROUTE_LOCATION_SHIFT;
<> 144:ef7eb2e8f9f7 216 } else {
AnnaBridge 179:b0033dcd6934 217 MBED_ASSERT((PWM_TIMER->ROUTE & _TIMER_ROUTE_LOCATION_MASK) == pinmap_find_function(pin,PinMap_PWM) << _TIMER_ROUTE_LOCATION_SHIFT);
<> 144:ef7eb2e8f9f7 218 }
<> 144:ef7eb2e8f9f7 219 #endif
<> 144:ef7eb2e8f9f7 220
<> 144:ef7eb2e8f9f7 221 // Set default 20ms frequency and 0ms pulse width
<> 144:ef7eb2e8f9f7 222 pwmout_period(obj, 0.02);
<> 144:ef7eb2e8f9f7 223 }
<> 144:ef7eb2e8f9f7 224
<> 144:ef7eb2e8f9f7 225 void pwmout_free(pwmout_t *obj)
<> 144:ef7eb2e8f9f7 226 {
<> 144:ef7eb2e8f9f7 227 if(pwmout_disable_channel_route(pwmout_get_channel_route(obj->channel))) {
<> 144:ef7eb2e8f9f7 228 //Channel was previously enabled, so do housekeeping
Anna Bridge 180:96ed750bd169 229 sleep_manager_unlock_deep_sleep();
<> 144:ef7eb2e8f9f7 230 } else {
<> 144:ef7eb2e8f9f7 231 //This channel was disabled already
<> 144:ef7eb2e8f9f7 232 }
AnnaBridge 179:b0033dcd6934 233
<> 144:ef7eb2e8f9f7 234 pwmout_enable_pins(obj, false);
AnnaBridge 179:b0033dcd6934 235
<> 144:ef7eb2e8f9f7 236 if(pwmout_all_inactive()) {
<> 144:ef7eb2e8f9f7 237 //Stop timer
<> 144:ef7eb2e8f9f7 238 PWM_TIMER->CMD = TIMER_CMD_STOP;
<> 144:ef7eb2e8f9f7 239 while(PWM_TIMER->STATUS & TIMER_STATUS_RUNNING);
AnnaBridge 179:b0033dcd6934 240
<> 144:ef7eb2e8f9f7 241 //Disable clock
<> 144:ef7eb2e8f9f7 242 CMU_ClockEnable(PWM_TIMER_CLOCK, false);
<> 144:ef7eb2e8f9f7 243 }
<> 144:ef7eb2e8f9f7 244 }
<> 144:ef7eb2e8f9f7 245
<> 144:ef7eb2e8f9f7 246 void pwmout_write(pwmout_t *obj, float value)
<> 144:ef7eb2e8f9f7 247 {
<> 144:ef7eb2e8f9f7 248 pwmout_write_channel(obj->channel, value);
<> 144:ef7eb2e8f9f7 249 }
<> 144:ef7eb2e8f9f7 250
<> 144:ef7eb2e8f9f7 251 void pwmout_write_channel(uint32_t channel, float value) {
<> 144:ef7eb2e8f9f7 252 uint32_t width_cycles = 0;
<> 144:ef7eb2e8f9f7 253 if (value < 0.0f) {
<> 144:ef7eb2e8f9f7 254 width_cycles = 0;
<> 144:ef7eb2e8f9f7 255 } else if (value >= 1.0f) {
<> 144:ef7eb2e8f9f7 256 width_cycles = PWM_TIMER->TOPB + 1;
<> 144:ef7eb2e8f9f7 257 } else {
<> 144:ef7eb2e8f9f7 258 width_cycles = (uint16_t)((float)PWM_TIMER->TOPB * value);
<> 144:ef7eb2e8f9f7 259 }
<> 144:ef7eb2e8f9f7 260
<> 144:ef7eb2e8f9f7 261 TIMER_CompareBufSet(PWM_TIMER, channel, width_cycles);
<> 144:ef7eb2e8f9f7 262 }
<> 144:ef7eb2e8f9f7 263
<> 144:ef7eb2e8f9f7 264 float pwmout_read(pwmout_t *obj)
<> 144:ef7eb2e8f9f7 265 {
<> 144:ef7eb2e8f9f7 266 return pwmout_calculate_duty(TIMER_CaptureGet(PWM_TIMER, obj->channel), TIMER_TopGet(PWM_TIMER));
<> 144:ef7eb2e8f9f7 267 }
<> 144:ef7eb2e8f9f7 268
<> 144:ef7eb2e8f9f7 269 float pwmout_calculate_duty(uint32_t width_cycles, uint32_t period_cycles) {
<> 144:ef7eb2e8f9f7 270 if(width_cycles > period_cycles) {
<> 144:ef7eb2e8f9f7 271 return 1.0f;
<> 144:ef7eb2e8f9f7 272 }
<> 144:ef7eb2e8f9f7 273 else if (width_cycles == 0) {
<> 144:ef7eb2e8f9f7 274 return 0.0f;
<> 144:ef7eb2e8f9f7 275 }
<> 144:ef7eb2e8f9f7 276 else {
<> 144:ef7eb2e8f9f7 277 return (float) width_cycles / (float) period_cycles;
<> 144:ef7eb2e8f9f7 278 }
<> 144:ef7eb2e8f9f7 279 }
<> 144:ef7eb2e8f9f7 280
<> 144:ef7eb2e8f9f7 281 // Set the PWM period, keeping the duty cycle the same.
<> 144:ef7eb2e8f9f7 282 void pwmout_period(pwmout_t *obj, float seconds)
<> 144:ef7eb2e8f9f7 283 {
<> 144:ef7eb2e8f9f7 284 // Find the lowest prescaler divider possible.
<> 144:ef7eb2e8f9f7 285 // This gives us max resolution for a given period
<> 144:ef7eb2e8f9f7 286
<> 144:ef7eb2e8f9f7 287 //The value of the top register if prescaler is set to 0
<> 144:ef7eb2e8f9f7 288 uint32_t cycles = (uint32_t)REFERENCE_FREQUENCY * seconds;
<> 144:ef7eb2e8f9f7 289 pwm_prescaler_div = 0;
<> 144:ef7eb2e8f9f7 290
<> 144:ef7eb2e8f9f7 291 //The top register is only 16 bits, so we keep dividing till we are below 0xFFFF
<> 144:ef7eb2e8f9f7 292 while (cycles > 0xFFFF) {
<> 144:ef7eb2e8f9f7 293 cycles /= 2;
<> 144:ef7eb2e8f9f7 294 pwm_prescaler_div++;
<> 144:ef7eb2e8f9f7 295
<> 144:ef7eb2e8f9f7 296 //Max pwm_prescaler_div supported is 10
<> 144:ef7eb2e8f9f7 297 if (pwm_prescaler_div > 10) {
<> 144:ef7eb2e8f9f7 298 pwm_prescaler_div = 10;
<> 144:ef7eb2e8f9f7 299 cycles = 0xFFFF; //Set it to max possible value;
<> 144:ef7eb2e8f9f7 300 break;
<> 144:ef7eb2e8f9f7 301 }
<> 144:ef7eb2e8f9f7 302 }
<> 144:ef7eb2e8f9f7 303
<> 144:ef7eb2e8f9f7 304 //Check if anything changed
AnnaBridge 188:bcfe06ba3d64 305 if(((PWM_TIMER->CTRL & _TIMER_CTRL_PRESC_MASK) == (pwm_prescaler_div << _TIMER_CTRL_PRESC_SHIFT)) && (TIMER_TopGet(PWM_TIMER) == cycles)) return;
<> 144:ef7eb2e8f9f7 306
<> 144:ef7eb2e8f9f7 307 //Save previous period for recalculation of duty cycles
<> 144:ef7eb2e8f9f7 308 uint32_t previous_period_cycles = PWM_TIMER->TOPB;
<> 144:ef7eb2e8f9f7 309
<> 144:ef7eb2e8f9f7 310 //Set prescaler
<> 144:ef7eb2e8f9f7 311 PWM_TIMER->CTRL = (PWM_TIMER->CTRL & ~_TIMER_CTRL_PRESC_MASK) | (pwm_prescaler_div << _TIMER_CTRL_PRESC_SHIFT);
<> 144:ef7eb2e8f9f7 312
<> 144:ef7eb2e8f9f7 313 //Set Top Value, which controls the PWM period
<> 144:ef7eb2e8f9f7 314 TIMER_TopBufSet(PWM_TIMER, cycles);
<> 144:ef7eb2e8f9f7 315
<> 144:ef7eb2e8f9f7 316 //For each active channel, re-calculate the compare value
<> 144:ef7eb2e8f9f7 317 uint32_t channel = 0;
<> 144:ef7eb2e8f9f7 318 while(pwmout_get_channel_route(channel) != 0) {
<> 144:ef7eb2e8f9f7 319 if(pwmout_channel_route_active(channel)) {
<> 144:ef7eb2e8f9f7 320 //recalc and reset compare value
<> 144:ef7eb2e8f9f7 321 pwmout_write_channel(channel, pwmout_calculate_duty(PWM_TIMER->CC[channel].CCVB, previous_period_cycles));
<> 144:ef7eb2e8f9f7 322 }
<> 144:ef7eb2e8f9f7 323 channel++;
<> 144:ef7eb2e8f9f7 324 }
<> 144:ef7eb2e8f9f7 325 }
<> 144:ef7eb2e8f9f7 326
<> 144:ef7eb2e8f9f7 327 void pwmout_period_ms(pwmout_t *obj, int ms)
<> 144:ef7eb2e8f9f7 328 {
<> 144:ef7eb2e8f9f7 329 pwmout_period(obj, ms / 1000.0f);
<> 144:ef7eb2e8f9f7 330 }
<> 144:ef7eb2e8f9f7 331
<> 144:ef7eb2e8f9f7 332 void pwmout_period_us(pwmout_t *obj, int us)
<> 144:ef7eb2e8f9f7 333 {
<> 144:ef7eb2e8f9f7 334 pwmout_period(obj, us / 1000000.0f);
<> 144:ef7eb2e8f9f7 335 }
<> 144:ef7eb2e8f9f7 336
<> 144:ef7eb2e8f9f7 337 void pwmout_pulsewidth(pwmout_t *obj, float seconds)
<> 144:ef7eb2e8f9f7 338 {
<> 144:ef7eb2e8f9f7 339 uint16_t width_cycles = (uint16_t) (((float) (REFERENCE_FREQUENCY >> pwm_prescaler_div)) * seconds);
<> 144:ef7eb2e8f9f7 340 TIMER_CompareBufSet(PWM_TIMER, obj->channel, width_cycles);
<> 144:ef7eb2e8f9f7 341 }
<> 144:ef7eb2e8f9f7 342
<> 144:ef7eb2e8f9f7 343 void pwmout_pulsewidth_ms(pwmout_t *obj, int ms)
<> 144:ef7eb2e8f9f7 344 {
<> 144:ef7eb2e8f9f7 345 uint16_t width_cycles = (uint16_t) ((REFERENCE_FREQUENCY >> pwm_prescaler_div) * ms) / 1000;
<> 144:ef7eb2e8f9f7 346 TIMER_CompareBufSet(PWM_TIMER, obj->channel, width_cycles);
<> 144:ef7eb2e8f9f7 347 }
<> 144:ef7eb2e8f9f7 348
<> 144:ef7eb2e8f9f7 349 void pwmout_pulsewidth_us(pwmout_t *obj, int us)
<> 144:ef7eb2e8f9f7 350 {
<> 144:ef7eb2e8f9f7 351 uint16_t width_cycles = (uint16_t) ((REFERENCE_FREQUENCY >> pwm_prescaler_div) * us) / 1000000;
<> 144:ef7eb2e8f9f7 352 TIMER_CompareBufSet(PWM_TIMER, obj->channel, width_cycles);
<> 144:ef7eb2e8f9f7 353 }
<> 144:ef7eb2e8f9f7 354
<> 144:ef7eb2e8f9f7 355 #endif