1 /*
2 * Copyright (c) 2023 Intel Corporation.
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT intel_blinky_pwm
8
9 #include <errno.h>
10 #include <soc.h>
11 #include <zephyr/device.h>
12 #include <zephyr/kernel.h>
13 #include <zephyr/init.h>
14 #include <zephyr/sys/util.h>
15 #include <zephyr/devicetree.h>
16 #include <zephyr/drivers/pwm.h>
17
18 #define PWM_ENABLE 0x80000000
19 #define PWM_SWUP 0x40000000
20 #define PWM_FREQ_INT_SHIFT 8
21 #define PWM_BASE_UNIT_FRACTION 14
22 #define PWM_FREQ_MAX 0x100
23 #define PWM_DUTY_MAX 0x100
24
25 struct bk_intel_config {
26 DEVICE_MMIO_NAMED_ROM(reg_base);
27 uint32_t reg_offset;
28 uint32_t clock_freq;
29 uint32_t max_pins;
30 };
31
32 struct bk_intel_runtime {
33 DEVICE_MMIO_NAMED_RAM(reg_base);
34 };
35
bk_intel_set_cycles(const struct device * dev,uint32_t pin,uint32_t period_cycles,uint32_t pulse_cycles,pwm_flags_t flags)36 static int bk_intel_set_cycles(const struct device *dev, uint32_t pin,
37 uint32_t period_cycles, uint32_t pulse_cycles,
38 pwm_flags_t flags)
39 {
40 struct bk_intel_runtime *rt = dev->data;
41 const struct bk_intel_config *cfg = dev->config;
42 uint32_t ret = 0;
43 uint32_t val = 0;
44 uint32_t duty;
45 float period;
46 float out_freq;
47 uint32_t base_unit;
48
49 if (pin >= cfg->max_pins) {
50 ret = -EINVAL;
51 goto err;
52 }
53
54 out_freq = cfg->clock_freq / (float) period_cycles;
55 period = (out_freq * PWM_FREQ_MAX) / cfg->clock_freq;
56 base_unit = (uint32_t) (period * (1 << PWM_BASE_UNIT_FRACTION));
57 duty = (pulse_cycles * PWM_DUTY_MAX) / period_cycles;
58
59 if (duty) {
60 val = PWM_DUTY_MAX - duty;
61 val |= (base_unit << PWM_FREQ_INT_SHIFT);
62 } else {
63 val = PWM_DUTY_MAX - 1;
64 }
65
66 val |= PWM_ENABLE | PWM_SWUP;
67
68 if (period >= PWM_FREQ_MAX) {
69 ret = -EINVAL;
70 goto err;
71 }
72
73 if (duty > PWM_DUTY_MAX) {
74 ret = -EINVAL;
75 goto err;
76 }
77
78 sys_write32(val, rt->reg_base + cfg->reg_offset);
79 err:
80 return ret;
81 }
82
bk_intel_get_cycles_per_sec(const struct device * dev,uint32_t pin,uint64_t * cycles)83 static int bk_intel_get_cycles_per_sec(const struct device *dev, uint32_t pin,
84 uint64_t *cycles)
85 {
86 const struct bk_intel_config *cfg = dev->config;
87
88 if (pin >= cfg->max_pins) {
89 return -EINVAL;
90 }
91
92 *cycles = cfg->clock_freq;
93
94 return 0;
95 }
96
97 static DEVICE_API(pwm, api_funcs) = {
98 .set_cycles = bk_intel_set_cycles,
99 .get_cycles_per_sec = bk_intel_get_cycles_per_sec,
100 };
101
bk_intel_init(const struct device * dev)102 static int bk_intel_init(const struct device *dev)
103 {
104 struct bk_intel_runtime *runtime = dev->data;
105 const struct bk_intel_config *config = dev->config;
106
107 device_map(&runtime->reg_base,
108 config->reg_base.phys_addr & ~0xFFU,
109 config->reg_base.size,
110 K_MEM_CACHE_NONE);
111
112 return 0;
113 }
114
115 #define BK_INTEL_DEV_CFG(n) \
116 static const struct bk_intel_config bk_cfg_##n = { \
117 DEVICE_MMIO_NAMED_ROM_INIT(reg_base, DT_DRV_INST(n)), \
118 .reg_offset = DT_INST_PROP(n, reg_offset), \
119 .max_pins = DT_INST_PROP(n, max_pins), \
120 .clock_freq = DT_INST_PROP(n, clock_frequency), \
121 }; \
122 \
123 static struct bk_intel_runtime bk_rt_##n; \
124 DEVICE_DT_INST_DEFINE(n, &bk_intel_init, NULL, \
125 &bk_rt_##n, &bk_cfg_##n, \
126 POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEVICE, \
127 &api_funcs); \
128
129 DT_INST_FOREACH_STATUS_OKAY(BK_INTEL_DEV_CFG)
130