Skip to content

Commit 6fb3481

Browse files
Fabrice Gasnierjic23
authored andcommitted
iio: stm32 trigger: Add support for TRGO2 triggers
Add support for TRGO2 trigger that can be found on STM32F7. Add additional master modes supported by TRGO2. Register additional "tim[1/8]_trgo2" triggers for timer1 & timer8. Detect TRGO2 timer capability (master mode selection 2). Signed-off-by: Fabrice Gasnier <[email protected]> Acked-by: Benjamin Gaignard <[email protected]> Signed-off-by: Jonathan Cameron <[email protected]>
1 parent f80ac40 commit 6fb3481

File tree

4 files changed

+151
-14
lines changed

4 files changed

+151
-14
lines changed

Documentation/ABI/testing/sysfs-bus-iio-timer-stm32

Lines changed: 48 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,54 @@ Description:
1616
- "OC2REF" : OC2REF signal is used as trigger output.
1717
- "OC3REF" : OC3REF signal is used as trigger output.
1818
- "OC4REF" : OC4REF signal is used as trigger output.
19+
Additional modes (on TRGO2 only):
20+
- "OC5REF" : OC5REF signal is used as trigger output.
21+
- "OC6REF" : OC6REF signal is used as trigger output.
22+
- "compare_pulse_OC4REF":
23+
OC4REF rising or falling edges generate pulses.
24+
- "compare_pulse_OC6REF":
25+
OC6REF rising or falling edges generate pulses.
26+
- "compare_pulse_OC4REF_r_or_OC6REF_r":
27+
OC4REF or OC6REF rising edges generate pulses.
28+
- "compare_pulse_OC4REF_r_or_OC6REF_f":
29+
OC4REF rising or OC6REF falling edges generate pulses.
30+
- "compare_pulse_OC5REF_r_or_OC6REF_r":
31+
OC5REF or OC6REF rising edges generate pulses.
32+
- "compare_pulse_OC5REF_r_or_OC6REF_f":
33+
OC5REF rising or OC6REF falling edges generate pulses.
34+
35+
+-----------+ +-------------+ +---------+
36+
| Prescaler +-> | Counter | +-> | Master | TRGO(2)
37+
+-----------+ +--+--------+-+ |-> | Control +-->
38+
| | || +---------+
39+
+--v--------+-+ OCxREF || +---------+
40+
| Chx compare +----------> | Output | ChX
41+
+-----------+-+ | | Control +-->
42+
. | | +---------+
43+
. | | .
44+
+-----------v-+ OC6REF | .
45+
| Ch6 compare +---------+>
46+
+-------------+
47+
48+
Example with: "compare_pulse_OC4REF_r_or_OC6REF_r":
49+
50+
X
51+
X X
52+
X . . X
53+
X . . X
54+
X . . X
55+
count X . . . . X
56+
. . . .
57+
. . . .
58+
+---------------+
59+
OC4REF | . . |
60+
+-+ . . +-+
61+
. +---+ .
62+
OC6REF . | | .
63+
+-------+ +-------+
64+
+-+ +-+
65+
TRGO2 | | | |
66+
+-+ +---+ +---------+
1967

2068
What: /sys/bus/iio/devices/triggerX/master_mode
2169
KernelVersion: 4.11

drivers/iio/trigger/stm32-timer-trigger.c

Lines changed: 99 additions & 14 deletions
Original file line numberDiff line numberDiff line change
@@ -14,19 +14,19 @@
1414
#include <linux/module.h>
1515
#include <linux/platform_device.h>
1616

17-
#define MAX_TRIGGERS 6
17+
#define MAX_TRIGGERS 7
1818
#define MAX_VALIDS 5
1919

2020
/* List the triggers created by each timer */
2121
static const void *triggers_table[][MAX_TRIGGERS] = {
22-
{ TIM1_TRGO, TIM1_CH1, TIM1_CH2, TIM1_CH3, TIM1_CH4,},
22+
{ TIM1_TRGO, TIM1_TRGO2, TIM1_CH1, TIM1_CH2, TIM1_CH3, TIM1_CH4,},
2323
{ TIM2_TRGO, TIM2_CH1, TIM2_CH2, TIM2_CH3, TIM2_CH4,},
2424
{ TIM3_TRGO, TIM3_CH1, TIM3_CH2, TIM3_CH3, TIM3_CH4,},
2525
{ TIM4_TRGO, TIM4_CH1, TIM4_CH2, TIM4_CH3, TIM4_CH4,},
2626
{ TIM5_TRGO, TIM5_CH1, TIM5_CH2, TIM5_CH3, TIM5_CH4,},
2727
{ TIM6_TRGO,},
2828
{ TIM7_TRGO,},
29-
{ TIM8_TRGO, TIM8_CH1, TIM8_CH2, TIM8_CH3, TIM8_CH4,},
29+
{ TIM8_TRGO, TIM8_TRGO2, TIM8_CH1, TIM8_CH2, TIM8_CH3, TIM8_CH4,},
3030
{ TIM9_TRGO, TIM9_CH1, TIM9_CH2,},
3131
{ }, /* timer 10 */
3232
{ }, /* timer 11 */
@@ -56,9 +56,16 @@ struct stm32_timer_trigger {
5656
u32 max_arr;
5757
const void *triggers;
5858
const void *valids;
59+
bool has_trgo2;
5960
};
6061

62+
static bool stm32_timer_is_trgo2_name(const char *name)
63+
{
64+
return !!strstr(name, "trgo2");
65+
}
66+
6167
static int stm32_timer_start(struct stm32_timer_trigger *priv,
68+
struct iio_trigger *trig,
6269
unsigned int frequency)
6370
{
6471
unsigned long long prd, div;
@@ -102,7 +109,12 @@ static int stm32_timer_start(struct stm32_timer_trigger *priv,
102109
regmap_update_bits(priv->regmap, TIM_CR1, TIM_CR1_ARPE, TIM_CR1_ARPE);
103110

104111
/* Force master mode to update mode */
105-
regmap_update_bits(priv->regmap, TIM_CR2, TIM_CR2_MMS, 0x20);
112+
if (stm32_timer_is_trgo2_name(trig->name))
113+
regmap_update_bits(priv->regmap, TIM_CR2, TIM_CR2_MMS2,
114+
0x2 << TIM_CR2_MMS2_SHIFT);
115+
else
116+
regmap_update_bits(priv->regmap, TIM_CR2, TIM_CR2_MMS,
117+
0x2 << TIM_CR2_MMS_SHIFT);
106118

107119
/* Make sure that registers are updated */
108120
regmap_update_bits(priv->regmap, TIM_EGR, TIM_EGR_UG, TIM_EGR_UG);
@@ -150,7 +162,7 @@ static ssize_t stm32_tt_store_frequency(struct device *dev,
150162
if (freq == 0) {
151163
stm32_timer_stop(priv);
152164
} else {
153-
ret = stm32_timer_start(priv, freq);
165+
ret = stm32_timer_start(priv, trig, freq);
154166
if (ret)
155167
return ret;
156168
}
@@ -183,6 +195,9 @@ static IIO_DEV_ATTR_SAMP_FREQ(0660,
183195
stm32_tt_read_frequency,
184196
stm32_tt_store_frequency);
185197

198+
#define MASTER_MODE_MAX 7
199+
#define MASTER_MODE2_MAX 15
200+
186201
static char *master_mode_table[] = {
187202
"reset",
188203
"enable",
@@ -191,18 +206,32 @@ static char *master_mode_table[] = {
191206
"OC1REF",
192207
"OC2REF",
193208
"OC3REF",
194-
"OC4REF"
209+
"OC4REF",
210+
/* Master mode selection 2 only */
211+
"OC5REF",
212+
"OC6REF",
213+
"compare_pulse_OC4REF",
214+
"compare_pulse_OC6REF",
215+
"compare_pulse_OC4REF_r_or_OC6REF_r",
216+
"compare_pulse_OC4REF_r_or_OC6REF_f",
217+
"compare_pulse_OC5REF_r_or_OC6REF_r",
218+
"compare_pulse_OC5REF_r_or_OC6REF_f",
195219
};
196220

197221
static ssize_t stm32_tt_show_master_mode(struct device *dev,
198222
struct device_attribute *attr,
199223
char *buf)
200224
{
201225
struct stm32_timer_trigger *priv = dev_get_drvdata(dev);
226+
struct iio_trigger *trig = to_iio_trigger(dev);
202227
u32 cr2;
203228

204229
regmap_read(priv->regmap, TIM_CR2, &cr2);
205-
cr2 = (cr2 & TIM_CR2_MMS) >> TIM_CR2_MMS_SHIFT;
230+
231+
if (stm32_timer_is_trgo2_name(trig->name))
232+
cr2 = (cr2 & TIM_CR2_MMS2) >> TIM_CR2_MMS2_SHIFT;
233+
else
234+
cr2 = (cr2 & TIM_CR2_MMS) >> TIM_CR2_MMS_SHIFT;
206235

207236
return snprintf(buf, PAGE_SIZE, "%s\n", master_mode_table[cr2]);
208237
}
@@ -212,13 +241,25 @@ static ssize_t stm32_tt_store_master_mode(struct device *dev,
212241
const char *buf, size_t len)
213242
{
214243
struct stm32_timer_trigger *priv = dev_get_drvdata(dev);
244+
struct iio_trigger *trig = to_iio_trigger(dev);
245+
u32 mask, shift, master_mode_max;
215246
int i;
216247

217-
for (i = 0; i < ARRAY_SIZE(master_mode_table); i++) {
248+
if (stm32_timer_is_trgo2_name(trig->name)) {
249+
mask = TIM_CR2_MMS2;
250+
shift = TIM_CR2_MMS2_SHIFT;
251+
master_mode_max = MASTER_MODE2_MAX;
252+
} else {
253+
mask = TIM_CR2_MMS;
254+
shift = TIM_CR2_MMS_SHIFT;
255+
master_mode_max = MASTER_MODE_MAX;
256+
}
257+
258+
for (i = 0; i <= master_mode_max; i++) {
218259
if (!strncmp(master_mode_table[i], buf,
219260
strlen(master_mode_table[i]))) {
220-
regmap_update_bits(priv->regmap, TIM_CR2,
221-
TIM_CR2_MMS, i << TIM_CR2_MMS_SHIFT);
261+
regmap_update_bits(priv->regmap, TIM_CR2, mask,
262+
i << shift);
222263
/* Make sure that registers are updated */
223264
regmap_update_bits(priv->regmap, TIM_EGR,
224265
TIM_EGR_UG, TIM_EGR_UG);
@@ -229,8 +270,31 @@ static ssize_t stm32_tt_store_master_mode(struct device *dev,
229270
return -EINVAL;
230271
}
231272

232-
static IIO_CONST_ATTR(master_mode_available,
233-
"reset enable update compare_pulse OC1REF OC2REF OC3REF OC4REF");
273+
static ssize_t stm32_tt_show_master_mode_avail(struct device *dev,
274+
struct device_attribute *attr,
275+
char *buf)
276+
{
277+
struct iio_trigger *trig = to_iio_trigger(dev);
278+
unsigned int i, master_mode_max;
279+
size_t len = 0;
280+
281+
if (stm32_timer_is_trgo2_name(trig->name))
282+
master_mode_max = MASTER_MODE2_MAX;
283+
else
284+
master_mode_max = MASTER_MODE_MAX;
285+
286+
for (i = 0; i <= master_mode_max; i++)
287+
len += scnprintf(buf + len, PAGE_SIZE - len,
288+
"%s ", master_mode_table[i]);
289+
290+
/* replace trailing space by newline */
291+
buf[len - 1] = '\n';
292+
293+
return len;
294+
}
295+
296+
static IIO_DEVICE_ATTR(master_mode_available, 0444,
297+
stm32_tt_show_master_mode_avail, NULL, 0);
234298

235299
static IIO_DEVICE_ATTR(master_mode, 0660,
236300
stm32_tt_show_master_mode,
@@ -240,7 +304,7 @@ static IIO_DEVICE_ATTR(master_mode, 0660,
240304
static struct attribute *stm32_trigger_attrs[] = {
241305
&iio_dev_attr_sampling_frequency.dev_attr.attr,
242306
&iio_dev_attr_master_mode.dev_attr.attr,
243-
&iio_const_attr_master_mode_available.dev_attr.attr,
307+
&iio_dev_attr_master_mode_available.dev_attr.attr,
244308
NULL,
245309
};
246310

@@ -264,6 +328,12 @@ static int stm32_setup_iio_triggers(struct stm32_timer_trigger *priv)
264328

265329
while (cur && *cur) {
266330
struct iio_trigger *trig;
331+
bool cur_is_trgo2 = stm32_timer_is_trgo2_name(*cur);
332+
333+
if (cur_is_trgo2 && !priv->has_trgo2) {
334+
cur++;
335+
continue;
336+
}
267337

268338
trig = devm_iio_trigger_alloc(priv->dev, "%s", *cur);
269339
if (!trig)
@@ -277,7 +347,7 @@ static int stm32_setup_iio_triggers(struct stm32_timer_trigger *priv)
277347
* should only be available on trgo trigger which
278348
* is always the first in the list.
279349
*/
280-
if (cur == priv->triggers)
350+
if (cur == priv->triggers || cur_is_trgo2)
281351
trig->dev.groups = stm32_trigger_attr_groups;
282352

283353
iio_trigger_set_drvdata(trig, priv);
@@ -584,6 +654,20 @@ bool is_stm32_timer_trigger(struct iio_trigger *trig)
584654
}
585655
EXPORT_SYMBOL(is_stm32_timer_trigger);
586656

657+
static void stm32_timer_detect_trgo2(struct stm32_timer_trigger *priv)
658+
{
659+
u32 val;
660+
661+
/*
662+
* Master mode selection 2 bits can only be written and read back when
663+
* timer supports it.
664+
*/
665+
regmap_update_bits(priv->regmap, TIM_CR2, TIM_CR2_MMS2, TIM_CR2_MMS2);
666+
regmap_read(priv->regmap, TIM_CR2, &val);
667+
regmap_update_bits(priv->regmap, TIM_CR2, TIM_CR2_MMS2, 0);
668+
priv->has_trgo2 = !!val;
669+
}
670+
587671
static int stm32_timer_trigger_probe(struct platform_device *pdev)
588672
{
589673
struct device *dev = &pdev->dev;
@@ -614,6 +698,7 @@ static int stm32_timer_trigger_probe(struct platform_device *pdev)
614698
priv->max_arr = ddata->max_arr;
615699
priv->triggers = triggers_table[index];
616700
priv->valids = valids_table[index];
701+
stm32_timer_detect_trgo2(priv);
617702

618703
ret = stm32_setup_iio_triggers(priv);
619704
if (ret)

include/linux/iio/timer/stm32-timer-trigger.h

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
#define _STM32_TIMER_TRIGGER_H_
1111

1212
#define TIM1_TRGO "tim1_trgo"
13+
#define TIM1_TRGO2 "tim1_trgo2"
1314
#define TIM1_CH1 "tim1_ch1"
1415
#define TIM1_CH2 "tim1_ch2"
1516
#define TIM1_CH3 "tim1_ch3"
@@ -44,6 +45,7 @@
4445
#define TIM7_TRGO "tim7_trgo"
4546

4647
#define TIM8_TRGO "tim8_trgo"
48+
#define TIM8_TRGO2 "tim8_trgo2"
4749
#define TIM8_CH1 "tim8_ch1"
4850
#define TIM8_CH2 "tim8_ch2"
4951
#define TIM8_CH3 "tim8_ch3"

include/linux/mfd/stm32-timers.h

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@
3434
#define TIM_CR1_DIR BIT(4) /* Counter Direction */
3535
#define TIM_CR1_ARPE BIT(7) /* Auto-reload Preload Ena */
3636
#define TIM_CR2_MMS (BIT(4) | BIT(5) | BIT(6)) /* Master mode selection */
37+
#define TIM_CR2_MMS2 GENMASK(23, 20) /* Master mode selection 2 */
3738
#define TIM_SMCR_SMS (BIT(0) | BIT(1) | BIT(2)) /* Slave mode selection */
3839
#define TIM_SMCR_TS (BIT(4) | BIT(5) | BIT(6)) /* Trigger selection */
3940
#define TIM_DIER_UIE BIT(0) /* Update interrupt */
@@ -60,6 +61,7 @@
6061

6162
#define MAX_TIM_PSC 0xFFFF
6263
#define TIM_CR2_MMS_SHIFT 4
64+
#define TIM_CR2_MMS2_SHIFT 20
6365
#define TIM_SMCR_TS_SHIFT 4
6466
#define TIM_BDTR_BKF_MASK 0xF
6567
#define TIM_BDTR_BKF_SHIFT 16

0 commit comments

Comments
 (0)