86 lines
1.9 KiB
C
86 lines
1.9 KiB
C
|
/* Copyright 2013 The Chromium OS Authors. All rights reserved.
|
||
|
* Use of this source code is governed by a BSD-style license that can be
|
||
|
* found in the LICENSE file.
|
||
|
*/
|
||
|
|
||
|
/* PWM control module for MEC1322 */
|
||
|
|
||
|
#include "hooks.h"
|
||
|
#include "pwm.h"
|
||
|
#include "pwm_chip.h"
|
||
|
#include "registers.h"
|
||
|
#include "util.h"
|
||
|
|
||
|
/*
|
||
|
* PWMs that must remain active in low-power idle - MEC1322_PCR_EC_SLP_EN
|
||
|
* bit mask.
|
||
|
*/
|
||
|
static uint32_t pwm_keep_awake_mask;
|
||
|
|
||
|
void pwm_enable(enum pwm_channel ch, int enabled)
|
||
|
{
|
||
|
int id = pwm_channels[ch].channel;
|
||
|
|
||
|
if (enabled) {
|
||
|
MEC1322_PWM_CFG(id) |= 0x1;
|
||
|
if (pwm_channels[ch].flags & PWM_CONFIG_DSLEEP)
|
||
|
pwm_keep_awake_mask |=
|
||
|
MEC1322_PCR_EC_SLP_EN_PWM(id);
|
||
|
} else {
|
||
|
MEC1322_PWM_CFG(id) &= ~0x1;
|
||
|
pwm_keep_awake_mask &= ~MEC1322_PCR_EC_SLP_EN_PWM(id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int pwm_get_enabled(enum pwm_channel ch)
|
||
|
{
|
||
|
return MEC1322_PWM_CFG(pwm_channels[ch].channel) & 0x1;
|
||
|
}
|
||
|
|
||
|
void pwm_set_duty(enum pwm_channel ch, int percent)
|
||
|
{
|
||
|
int id = pwm_channels[ch].channel;
|
||
|
|
||
|
if (percent < 0)
|
||
|
percent = 0;
|
||
|
else if (percent > 100)
|
||
|
percent = 100;
|
||
|
|
||
|
MEC1322_PWM_ON(id) = percent;
|
||
|
MEC1322_PWM_OFF(id) = 100 - percent;
|
||
|
}
|
||
|
|
||
|
int pwm_get_duty(enum pwm_channel ch)
|
||
|
{
|
||
|
return MEC1322_PWM_ON(pwm_channels[ch].channel);
|
||
|
}
|
||
|
|
||
|
uint32_t pwm_get_keep_awake_mask(void)
|
||
|
{
|
||
|
return pwm_keep_awake_mask;
|
||
|
}
|
||
|
|
||
|
static void pwm_configure(int ch, int active_low, int clock_low)
|
||
|
{
|
||
|
/*
|
||
|
* clock_low=0 selects the 48MHz Ring Oscillator source
|
||
|
* clock_low=1 selects the 100kHz_Clk source
|
||
|
*/
|
||
|
MEC1322_PWM_CFG(ch) = (15 << 3) | /* Pre-divider = 16 */
|
||
|
(active_low ? BIT(2) : 0) |
|
||
|
(clock_low ? BIT(1) : 0);
|
||
|
}
|
||
|
|
||
|
static void pwm_init(void)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
for (i = 0; i < PWM_CH_COUNT; ++i) {
|
||
|
pwm_configure(pwm_channels[i].channel,
|
||
|
pwm_channels[i].flags & PWM_CONFIG_ACTIVE_LOW,
|
||
|
pwm_channels[i].flags & PWM_CONFIG_ALT_CLOCK);
|
||
|
pwm_set_duty(i, 0);
|
||
|
}
|
||
|
}
|
||
|
DECLARE_HOOK(HOOK_INIT, pwm_init, HOOK_PRIO_INIT_PWM);
|