xref: /linux/drivers/soundwire/sysfs_slave.c (revision fc7e56017b51482f1b9da2e778eedb4bd1deb6b3)
1bcac5902SPierre-Louis Bossart // SPDX-License-Identifier: GPL-2.0-only
2bcac5902SPierre-Louis Bossart // Copyright(c) 2015-2020 Intel Corporation.
3bcac5902SPierre-Louis Bossart 
4bcac5902SPierre-Louis Bossart #include <linux/device.h>
5bcac5902SPierre-Louis Bossart #include <linux/mod_devicetable.h>
6bcac5902SPierre-Louis Bossart #include <linux/slab.h>
7bcac5902SPierre-Louis Bossart #include <linux/sysfs.h>
8bcac5902SPierre-Louis Bossart #include <linux/soundwire/sdw.h>
9bcac5902SPierre-Louis Bossart #include <linux/soundwire/sdw_type.h>
10bcac5902SPierre-Louis Bossart #include "bus.h"
11bcac5902SPierre-Louis Bossart #include "sysfs_local.h"
12bcac5902SPierre-Louis Bossart 
13bcac5902SPierre-Louis Bossart /*
14bcac5902SPierre-Louis Bossart  * Slave sysfs
15bcac5902SPierre-Louis Bossart  */
16bcac5902SPierre-Louis Bossart 
17bcac5902SPierre-Louis Bossart /*
18bcac5902SPierre-Louis Bossart  * The sysfs for Slave reflects the MIPI description as given
190173f525SPierre-Louis Bossart  * in the MIPI DisCo spec.
200173f525SPierre-Louis Bossart  * status and device_number come directly from the MIPI SoundWire
210173f525SPierre-Louis Bossart  * 1.x specification.
22bcac5902SPierre-Louis Bossart  *
23bcac5902SPierre-Louis Bossart  * Base file is device
240173f525SPierre-Louis Bossart  *	|---- status
250173f525SPierre-Louis Bossart  *	|---- device_number
26bcac5902SPierre-Louis Bossart  *	|---- modalias
27bcac5902SPierre-Louis Bossart  *	|---- dev-properties
28bcac5902SPierre-Louis Bossart  *		|---- mipi_revision
29bcac5902SPierre-Louis Bossart  *		|---- wake_capable
30bcac5902SPierre-Louis Bossart  *		|---- test_mode_capable
31bcac5902SPierre-Louis Bossart  *		|---- clk_stop_mode1
32bcac5902SPierre-Louis Bossart  *		|---- simple_clk_stop_capable
33bcac5902SPierre-Louis Bossart  *		|---- clk_stop_timeout
34bcac5902SPierre-Louis Bossart  *		|---- ch_prep_timeout
35bcac5902SPierre-Louis Bossart  *		|---- reset_behave
36bcac5902SPierre-Louis Bossart  *		|---- high_PHY_capable
37bcac5902SPierre-Louis Bossart  *		|---- paging_support
38bcac5902SPierre-Louis Bossart  *		|---- bank_delay_support
39bcac5902SPierre-Louis Bossart  *		|---- p15_behave
40bcac5902SPierre-Louis Bossart  *		|---- master_count
41bcac5902SPierre-Louis Bossart  *		|---- source_ports
42bcac5902SPierre-Louis Bossart  *		|---- sink_ports
43bcac5902SPierre-Louis Bossart  *	|---- dp0
44bcac5902SPierre-Louis Bossart  *		|---- max_word
45bcac5902SPierre-Louis Bossart  *		|---- min_word
46bcac5902SPierre-Louis Bossart  *		|---- words
47bcac5902SPierre-Louis Bossart  *		|---- BRA_flow_controlled
48bcac5902SPierre-Louis Bossart  *		|---- simple_ch_prep_sm
49bcac5902SPierre-Louis Bossart  *		|---- imp_def_interrupts
50bcac5902SPierre-Louis Bossart  *	|---- dpN_<sink/src>
51bcac5902SPierre-Louis Bossart  *		|---- max_word
52bcac5902SPierre-Louis Bossart  *		|---- min_word
53bcac5902SPierre-Louis Bossart  *		|---- words
54bcac5902SPierre-Louis Bossart  *		|---- type
55bcac5902SPierre-Louis Bossart  *		|---- max_grouping
56bcac5902SPierre-Louis Bossart  *		|---- simple_ch_prep_sm
57bcac5902SPierre-Louis Bossart  *		|---- ch_prep_timeout
58bcac5902SPierre-Louis Bossart  *		|---- imp_def_interrupts
59bcac5902SPierre-Louis Bossart  *		|---- min_ch
60bcac5902SPierre-Louis Bossart  *		|---- max_ch
61bcac5902SPierre-Louis Bossart  *		|---- channels
62bcac5902SPierre-Louis Bossart  *		|---- ch_combinations
63bcac5902SPierre-Louis Bossart  *		|---- max_async_buffer
64bcac5902SPierre-Louis Bossart  *		|---- block_pack_mode
65bcac5902SPierre-Louis Bossart  *		|---- port_encoding
66bcac5902SPierre-Louis Bossart  *
67bcac5902SPierre-Louis Bossart  */
68bcac5902SPierre-Louis Bossart 
69bcac5902SPierre-Louis Bossart #define sdw_slave_attr(field, format_string)			\
70bcac5902SPierre-Louis Bossart static ssize_t field##_show(struct device *dev,			\
71bcac5902SPierre-Louis Bossart 			    struct device_attribute *attr,	\
72bcac5902SPierre-Louis Bossart 			    char *buf)				\
73bcac5902SPierre-Louis Bossart {								\
74bcac5902SPierre-Louis Bossart 	struct sdw_slave *slave = dev_to_sdw_dev(dev);		\
75bcac5902SPierre-Louis Bossart 	return sprintf(buf, format_string, slave->prop.field);	\
76bcac5902SPierre-Louis Bossart }								\
77bcac5902SPierre-Louis Bossart static DEVICE_ATTR_RO(field)
78bcac5902SPierre-Louis Bossart 
79bcac5902SPierre-Louis Bossart sdw_slave_attr(mipi_revision, "0x%x\n");
80bcac5902SPierre-Louis Bossart sdw_slave_attr(wake_capable, "%d\n");
81bcac5902SPierre-Louis Bossart sdw_slave_attr(test_mode_capable, "%d\n");
82bcac5902SPierre-Louis Bossart sdw_slave_attr(clk_stop_mode1, "%d\n");
83bcac5902SPierre-Louis Bossart sdw_slave_attr(simple_clk_stop_capable, "%d\n");
84bcac5902SPierre-Louis Bossart sdw_slave_attr(clk_stop_timeout, "%d\n");
85bcac5902SPierre-Louis Bossart sdw_slave_attr(ch_prep_timeout, "%d\n");
86bcac5902SPierre-Louis Bossart sdw_slave_attr(reset_behave, "%d\n");
87bcac5902SPierre-Louis Bossart sdw_slave_attr(high_PHY_capable, "%d\n");
88bcac5902SPierre-Louis Bossart sdw_slave_attr(paging_support, "%d\n");
89bcac5902SPierre-Louis Bossart sdw_slave_attr(bank_delay_support, "%d\n");
90bcac5902SPierre-Louis Bossart sdw_slave_attr(p15_behave, "%d\n");
91bcac5902SPierre-Louis Bossart sdw_slave_attr(master_count, "%d\n");
92bcac5902SPierre-Louis Bossart sdw_slave_attr(source_ports, "0x%x\n");
93bcac5902SPierre-Louis Bossart sdw_slave_attr(sink_ports, "0x%x\n");
94bcac5902SPierre-Louis Bossart 
95bcac5902SPierre-Louis Bossart static ssize_t modalias_show(struct device *dev,
96bcac5902SPierre-Louis Bossart 			     struct device_attribute *attr, char *buf)
97bcac5902SPierre-Louis Bossart {
98bcac5902SPierre-Louis Bossart 	struct sdw_slave *slave = dev_to_sdw_dev(dev);
99bcac5902SPierre-Louis Bossart 
100bcac5902SPierre-Louis Bossart 	return sdw_slave_modalias(slave, buf, 256);
101bcac5902SPierre-Louis Bossart }
102bcac5902SPierre-Louis Bossart static DEVICE_ATTR_RO(modalias);
103bcac5902SPierre-Louis Bossart 
104bcac5902SPierre-Louis Bossart static struct attribute *slave_attrs[] = {
105bcac5902SPierre-Louis Bossart 	&dev_attr_modalias.attr,
106bcac5902SPierre-Louis Bossart 	NULL,
107bcac5902SPierre-Louis Bossart };
108b1b11bb0SGreg Kroah-Hartman 
109b1b11bb0SGreg Kroah-Hartman static const struct attribute_group slave_attr_group = {
110b1b11bb0SGreg Kroah-Hartman 	.attrs = slave_attrs,
111b1b11bb0SGreg Kroah-Hartman };
112bcac5902SPierre-Louis Bossart 
113bcac5902SPierre-Louis Bossart static struct attribute *slave_dev_attrs[] = {
114bcac5902SPierre-Louis Bossart 	&dev_attr_mipi_revision.attr,
115bcac5902SPierre-Louis Bossart 	&dev_attr_wake_capable.attr,
116bcac5902SPierre-Louis Bossart 	&dev_attr_test_mode_capable.attr,
117bcac5902SPierre-Louis Bossart 	&dev_attr_clk_stop_mode1.attr,
118bcac5902SPierre-Louis Bossart 	&dev_attr_simple_clk_stop_capable.attr,
119bcac5902SPierre-Louis Bossart 	&dev_attr_clk_stop_timeout.attr,
120bcac5902SPierre-Louis Bossart 	&dev_attr_ch_prep_timeout.attr,
121bcac5902SPierre-Louis Bossart 	&dev_attr_reset_behave.attr,
122bcac5902SPierre-Louis Bossart 	&dev_attr_high_PHY_capable.attr,
123bcac5902SPierre-Louis Bossart 	&dev_attr_paging_support.attr,
124bcac5902SPierre-Louis Bossart 	&dev_attr_bank_delay_support.attr,
125bcac5902SPierre-Louis Bossart 	&dev_attr_p15_behave.attr,
126bcac5902SPierre-Louis Bossart 	&dev_attr_master_count.attr,
127bcac5902SPierre-Louis Bossart 	&dev_attr_source_ports.attr,
128bcac5902SPierre-Louis Bossart 	&dev_attr_sink_ports.attr,
129bcac5902SPierre-Louis Bossart 	NULL,
130bcac5902SPierre-Louis Bossart };
131bcac5902SPierre-Louis Bossart 
132bcac5902SPierre-Louis Bossart /*
133bcac5902SPierre-Louis Bossart  * we don't use ATTRIBUTES_GROUP here since we want to add a subdirectory
134bcac5902SPierre-Louis Bossart  * for device-level properties
135bcac5902SPierre-Louis Bossart  */
136565e3afaSRikard Falkeborn static const struct attribute_group sdw_slave_dev_attr_group = {
137bcac5902SPierre-Louis Bossart 	.attrs	= slave_dev_attrs,
138bcac5902SPierre-Louis Bossart 	.name = "dev-properties",
139bcac5902SPierre-Louis Bossart };
140bcac5902SPierre-Louis Bossart 
141bcac5902SPierre-Louis Bossart /*
142bcac5902SPierre-Louis Bossart  * DP0 sysfs
143bcac5902SPierre-Louis Bossart  */
144bcac5902SPierre-Louis Bossart 
145bcac5902SPierre-Louis Bossart #define sdw_dp0_attr(field, format_string)				\
146bcac5902SPierre-Louis Bossart static ssize_t field##_show(struct device *dev,				\
147bcac5902SPierre-Louis Bossart 			    struct device_attribute *attr,		\
148bcac5902SPierre-Louis Bossart 			    char *buf)					\
149bcac5902SPierre-Louis Bossart {									\
150bcac5902SPierre-Louis Bossart 	struct sdw_slave *slave = dev_to_sdw_dev(dev);			\
151bcac5902SPierre-Louis Bossart 	return sprintf(buf, format_string, slave->prop.dp0_prop->field);\
152bcac5902SPierre-Louis Bossart }									\
153bcac5902SPierre-Louis Bossart static DEVICE_ATTR_RO(field)
154bcac5902SPierre-Louis Bossart 
155bcac5902SPierre-Louis Bossart sdw_dp0_attr(max_word, "%d\n");
156bcac5902SPierre-Louis Bossart sdw_dp0_attr(min_word, "%d\n");
157bcac5902SPierre-Louis Bossart sdw_dp0_attr(BRA_flow_controlled, "%d\n");
158bcac5902SPierre-Louis Bossart sdw_dp0_attr(simple_ch_prep_sm, "%d\n");
159bcac5902SPierre-Louis Bossart sdw_dp0_attr(imp_def_interrupts, "0x%x\n");
160bcac5902SPierre-Louis Bossart 
161bcac5902SPierre-Louis Bossart static ssize_t words_show(struct device *dev,
162bcac5902SPierre-Louis Bossart 			  struct device_attribute *attr, char *buf)
163bcac5902SPierre-Louis Bossart {
164bcac5902SPierre-Louis Bossart 	struct sdw_slave *slave = dev_to_sdw_dev(dev);
165bcac5902SPierre-Louis Bossart 	ssize_t size = 0;
166bcac5902SPierre-Louis Bossart 	int i;
167bcac5902SPierre-Louis Bossart 
168bcac5902SPierre-Louis Bossart 	for (i = 0; i < slave->prop.dp0_prop->num_words; i++)
169bcac5902SPierre-Louis Bossart 		size += sprintf(buf + size, "%d ",
170bcac5902SPierre-Louis Bossart 				slave->prop.dp0_prop->words[i]);
171bcac5902SPierre-Louis Bossart 	size += sprintf(buf + size, "\n");
172bcac5902SPierre-Louis Bossart 
173bcac5902SPierre-Louis Bossart 	return size;
174bcac5902SPierre-Louis Bossart }
175bcac5902SPierre-Louis Bossart static DEVICE_ATTR_RO(words);
176bcac5902SPierre-Louis Bossart 
177bcac5902SPierre-Louis Bossart static struct attribute *dp0_attrs[] = {
178bcac5902SPierre-Louis Bossart 	&dev_attr_max_word.attr,
179bcac5902SPierre-Louis Bossart 	&dev_attr_min_word.attr,
180bcac5902SPierre-Louis Bossart 	&dev_attr_words.attr,
181bcac5902SPierre-Louis Bossart 	&dev_attr_BRA_flow_controlled.attr,
182bcac5902SPierre-Louis Bossart 	&dev_attr_simple_ch_prep_sm.attr,
183bcac5902SPierre-Louis Bossart 	&dev_attr_imp_def_interrupts.attr,
184bcac5902SPierre-Louis Bossart 	NULL,
185bcac5902SPierre-Louis Bossart };
186bcac5902SPierre-Louis Bossart 
1873ee43f7cSGreg Kroah-Hartman static umode_t dp0_attr_visible(struct kobject *kobj, struct attribute *attr,
1883ee43f7cSGreg Kroah-Hartman 			      int n)
1893ee43f7cSGreg Kroah-Hartman {
1903ee43f7cSGreg Kroah-Hartman 	struct sdw_slave *slave = dev_to_sdw_dev(kobj_to_dev(kobj));
1913ee43f7cSGreg Kroah-Hartman 
1923ee43f7cSGreg Kroah-Hartman 	if (slave->prop.dp0_prop)
1933ee43f7cSGreg Kroah-Hartman 		return attr->mode;
1943ee43f7cSGreg Kroah-Hartman 	return 0;
1953ee43f7cSGreg Kroah-Hartman }
1963ee43f7cSGreg Kroah-Hartman 
1973ee43f7cSGreg Kroah-Hartman static bool dp0_group_visible(struct kobject *kobj)
1983ee43f7cSGreg Kroah-Hartman {
1993ee43f7cSGreg Kroah-Hartman 	struct sdw_slave *slave = dev_to_sdw_dev(kobj_to_dev(kobj));
2003ee43f7cSGreg Kroah-Hartman 
2013ee43f7cSGreg Kroah-Hartman 	if (slave->prop.dp0_prop)
2023ee43f7cSGreg Kroah-Hartman 		return true;
2033ee43f7cSGreg Kroah-Hartman 	return false;
2043ee43f7cSGreg Kroah-Hartman }
2053ee43f7cSGreg Kroah-Hartman DEFINE_SYSFS_GROUP_VISIBLE(dp0);
2063ee43f7cSGreg Kroah-Hartman 
207bcac5902SPierre-Louis Bossart /*
208bcac5902SPierre-Louis Bossart  * we don't use ATTRIBUTES_GROUP here since we want to add a subdirectory
209bcac5902SPierre-Louis Bossart  * for dp0-level properties
210bcac5902SPierre-Louis Bossart  */
211bcac5902SPierre-Louis Bossart static const struct attribute_group dp0_group = {
212bcac5902SPierre-Louis Bossart 	.attrs = dp0_attrs,
2133ee43f7cSGreg Kroah-Hartman 	.is_visible = SYSFS_GROUP_VISIBLE(dp0),
214bcac5902SPierre-Louis Bossart 	.name = "dp0",
215bcac5902SPierre-Louis Bossart };
216bcac5902SPierre-Louis Bossart 
217*fc7e5601SGreg Kroah-Hartman const struct attribute_group *sdw_attr_groups[] = {
218b1b11bb0SGreg Kroah-Hartman 	&slave_attr_group,
219b1b11bb0SGreg Kroah-Hartman 	&sdw_slave_dev_attr_group,
2203ee43f7cSGreg Kroah-Hartman 	&dp0_group,
221b1b11bb0SGreg Kroah-Hartman 	NULL,
222b1b11bb0SGreg Kroah-Hartman };
223b1b11bb0SGreg Kroah-Hartman 
224bcac5902SPierre-Louis Bossart int sdw_slave_sysfs_init(struct sdw_slave *slave)
225bcac5902SPierre-Louis Bossart {
226bcac5902SPierre-Louis Bossart 	int ret;
227bcac5902SPierre-Louis Bossart 
228bcac5902SPierre-Louis Bossart 	if (slave->prop.source_ports || slave->prop.sink_ports) {
229bcac5902SPierre-Louis Bossart 		ret = sdw_slave_sysfs_dpn_init(slave);
230bcac5902SPierre-Louis Bossart 		if (ret < 0)
231bcac5902SPierre-Louis Bossart 			return ret;
232bcac5902SPierre-Louis Bossart 	}
233bcac5902SPierre-Louis Bossart 
234bcac5902SPierre-Louis Bossart 	return 0;
235bcac5902SPierre-Louis Bossart }
2360173f525SPierre-Louis Bossart 
2370173f525SPierre-Louis Bossart /*
2380173f525SPierre-Louis Bossart  * the status is shown in capital letters for UNATTACHED and RESERVED
2390173f525SPierre-Louis Bossart  * on purpose, to highligh users to the fact that these status values
2400173f525SPierre-Louis Bossart  * are not expected.
2410173f525SPierre-Louis Bossart  */
2420173f525SPierre-Louis Bossart static const char *const slave_status[] = {
2430173f525SPierre-Louis Bossart 	[SDW_SLAVE_UNATTACHED] =  "UNATTACHED",
2440173f525SPierre-Louis Bossart 	[SDW_SLAVE_ATTACHED] = "Attached",
2450173f525SPierre-Louis Bossart 	[SDW_SLAVE_ALERT] = "Alert",
2460173f525SPierre-Louis Bossart 	[SDW_SLAVE_RESERVED] = "RESERVED",
2470173f525SPierre-Louis Bossart };
2480173f525SPierre-Louis Bossart 
2490173f525SPierre-Louis Bossart static ssize_t status_show(struct device *dev,
2500173f525SPierre-Louis Bossart 			   struct device_attribute *attr, char *buf)
2510173f525SPierre-Louis Bossart {
2520173f525SPierre-Louis Bossart 	struct sdw_slave *slave = dev_to_sdw_dev(dev);
2530173f525SPierre-Louis Bossart 
2540173f525SPierre-Louis Bossart 	return sprintf(buf, "%s\n", slave_status[slave->status]);
2550173f525SPierre-Louis Bossart }
2560173f525SPierre-Louis Bossart static DEVICE_ATTR_RO(status);
2570173f525SPierre-Louis Bossart 
2580173f525SPierre-Louis Bossart static ssize_t device_number_show(struct device *dev,
2590173f525SPierre-Louis Bossart 				  struct device_attribute *attr, char *buf)
2600173f525SPierre-Louis Bossart {
2610173f525SPierre-Louis Bossart 	struct sdw_slave *slave = dev_to_sdw_dev(dev);
2620173f525SPierre-Louis Bossart 
2630173f525SPierre-Louis Bossart 	if (slave->status == SDW_SLAVE_UNATTACHED)
2640173f525SPierre-Louis Bossart 		return sprintf(buf, "%s", "N/A");
2650173f525SPierre-Louis Bossart 	else
2660173f525SPierre-Louis Bossart 		return sprintf(buf, "%d", slave->dev_num);
2670173f525SPierre-Louis Bossart }
2680173f525SPierre-Louis Bossart static DEVICE_ATTR_RO(device_number);
2690173f525SPierre-Louis Bossart 
2700173f525SPierre-Louis Bossart static struct attribute *slave_status_attrs[] = {
2710173f525SPierre-Louis Bossart 	&dev_attr_status.attr,
2720173f525SPierre-Louis Bossart 	&dev_attr_device_number.attr,
2730173f525SPierre-Louis Bossart 	NULL,
2740173f525SPierre-Louis Bossart };
2750173f525SPierre-Louis Bossart 
2760173f525SPierre-Louis Bossart /*
2770173f525SPierre-Louis Bossart  * we don't use ATTRIBUTES_GROUP here since the group is used in a
2780173f525SPierre-Louis Bossart  * separate file and can't be handled as a static.
2790173f525SPierre-Louis Bossart  */
2800173f525SPierre-Louis Bossart static const struct attribute_group sdw_slave_status_attr_group = {
2810173f525SPierre-Louis Bossart 	.attrs	= slave_status_attrs,
2820173f525SPierre-Louis Bossart };
2830173f525SPierre-Louis Bossart 
2840173f525SPierre-Louis Bossart const struct attribute_group *sdw_slave_status_attr_groups[] = {
2850173f525SPierre-Louis Bossart 	&sdw_slave_status_attr_group,
2860173f525SPierre-Louis Bossart 	NULL
2870173f525SPierre-Louis Bossart };
288