d982ea54b6
This patch adds support for PWM blink which is found in intel's PCH hardwares. Signed-off-by: Anisetti Avinash Krishna <anisetti.avinash.krishna@intel.com>
130 lines
3.1 KiB
C
130 lines
3.1 KiB
C
/*
|
|
* Copyright (c) 2023 Intel Corporation.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#define DT_DRV_COMPAT intel_blinky_pwm
|
|
|
|
#include <errno.h>
|
|
#include <soc.h>
|
|
#include <zephyr/device.h>
|
|
#include <zephyr/kernel.h>
|
|
#include <zephyr/init.h>
|
|
#include <zephyr/sys/util.h>
|
|
#include <zephyr/devicetree.h>
|
|
#include <zephyr/drivers/pwm.h>
|
|
|
|
#define PWM_ENABLE 0x80000000
|
|
#define PWM_SWUP 0x40000000
|
|
#define PWM_FREQ_INT_SHIFT 8
|
|
#define PWM_BASE_UNIT_FRACTION 14
|
|
#define PWM_FREQ_MAX 0x100
|
|
#define PWM_DUTY_MAX 0x100
|
|
|
|
struct bk_intel_config {
|
|
DEVICE_MMIO_NAMED_ROM(reg_base);
|
|
uint32_t reg_offset;
|
|
uint32_t clock_freq;
|
|
uint32_t max_pins;
|
|
};
|
|
|
|
struct bk_intel_runtime {
|
|
DEVICE_MMIO_NAMED_RAM(reg_base);
|
|
};
|
|
|
|
static int bk_intel_set_cycles(const struct device *dev, uint32_t pin,
|
|
uint32_t period_cycles, uint32_t pulse_cycles,
|
|
pwm_flags_t flags)
|
|
{
|
|
struct bk_intel_runtime *rt = dev->data;
|
|
const struct bk_intel_config *cfg = dev->config;
|
|
uint32_t ret = 0;
|
|
uint32_t val = 0;
|
|
uint32_t duty;
|
|
float period;
|
|
float out_freq;
|
|
uint32_t base_unit;
|
|
|
|
if (pin >= cfg->max_pins) {
|
|
ret = -EINVAL;
|
|
goto err;
|
|
}
|
|
|
|
out_freq = cfg->clock_freq / (float) period_cycles;
|
|
period = (out_freq * PWM_FREQ_MAX) / cfg->clock_freq;
|
|
base_unit = (uint32_t) (period * (1 << PWM_BASE_UNIT_FRACTION));
|
|
duty = (pulse_cycles * PWM_DUTY_MAX) / period_cycles;
|
|
|
|
if (duty) {
|
|
val = PWM_DUTY_MAX - duty;
|
|
val |= (base_unit << PWM_FREQ_INT_SHIFT);
|
|
} else {
|
|
val = PWM_DUTY_MAX - 1;
|
|
}
|
|
|
|
val |= PWM_ENABLE | PWM_SWUP;
|
|
|
|
if (period >= PWM_FREQ_MAX) {
|
|
ret = -EINVAL;
|
|
goto err;
|
|
}
|
|
|
|
if (duty > PWM_DUTY_MAX) {
|
|
ret = -EINVAL;
|
|
goto err;
|
|
}
|
|
|
|
sys_write32(val, rt->reg_base + cfg->reg_offset);
|
|
err:
|
|
return ret;
|
|
}
|
|
|
|
static int bk_intel_get_cycles_per_sec(const struct device *dev, uint32_t pin,
|
|
uint64_t *cycles)
|
|
{
|
|
const struct bk_intel_config *cfg = dev->config;
|
|
|
|
if (pin >= cfg->max_pins) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
*cycles = cfg->clock_freq;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct pwm_driver_api api_funcs = {
|
|
.set_cycles = bk_intel_set_cycles,
|
|
.get_cycles_per_sec = bk_intel_get_cycles_per_sec,
|
|
};
|
|
|
|
static int bk_intel_init(const struct device *dev)
|
|
{
|
|
struct bk_intel_runtime *runtime = dev->data;
|
|
const struct bk_intel_config *config = dev->config;
|
|
|
|
device_map(&runtime->reg_base,
|
|
config->reg_base.phys_addr & ~0xFFU,
|
|
config->reg_base.size,
|
|
K_MEM_CACHE_NONE);
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define BK_INTEL_DEV_CFG(n) \
|
|
static const struct bk_intel_config bk_cfg_##n = { \
|
|
DEVICE_MMIO_NAMED_ROM_INIT(reg_base, DT_DRV_INST(n)), \
|
|
.reg_offset = DT_INST_PROP(n, reg_offset), \
|
|
.max_pins = DT_INST_PROP(n, max_pins), \
|
|
.clock_freq = DT_INST_PROP(n, clock_frequency), \
|
|
}; \
|
|
\
|
|
static struct bk_intel_runtime bk_rt_##n; \
|
|
DEVICE_DT_INST_DEFINE(n, &bk_intel_init, NULL, \
|
|
&bk_rt_##n, &bk_cfg_##n, \
|
|
POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEVICE, \
|
|
&api_funcs); \
|
|
|
|
DT_INST_FOREACH_STATUS_OKAY(BK_INTEL_DEV_CFG)
|