xref: /openbmc/linux/drivers/soundwire/sysfs_slave.c (revision e65e175b07bef5974045cc42238de99057669ca7)
1 // SPDX-License-Identifier: GPL-2.0-only
2 // Copyright(c) 2015-2020 Intel Corporation.
3 
4 #include <linux/device.h>
5 #include <linux/mod_devicetable.h>
6 #include <linux/slab.h>
7 #include <linux/sysfs.h>
8 #include <linux/soundwire/sdw.h>
9 #include <linux/soundwire/sdw_type.h>
10 #include "bus.h"
11 #include "sysfs_local.h"
12 
13 /*
14  * Slave sysfs
15  */
16 
17 /*
18  * The sysfs for Slave reflects the MIPI description as given
19  * in the MIPI DisCo spec.
20  * status and device_number come directly from the MIPI SoundWire
21  * 1.x specification.
22  *
23  * Base file is device
24  *	|---- status
25  *	|---- device_number
26  *	|---- modalias
27  *	|---- dev-properties
28  *		|---- mipi_revision
29  *		|---- wake_capable
30  *		|---- test_mode_capable
31  *		|---- clk_stop_mode1
32  *		|---- simple_clk_stop_capable
33  *		|---- clk_stop_timeout
34  *		|---- ch_prep_timeout
35  *		|---- reset_behave
36  *		|---- high_PHY_capable
37  *		|---- paging_support
38  *		|---- bank_delay_support
39  *		|---- p15_behave
40  *		|---- master_count
41  *		|---- source_ports
42  *		|---- sink_ports
43  *	|---- dp0
44  *		|---- max_word
45  *		|---- min_word
46  *		|---- words
47  *		|---- BRA_flow_controlled
48  *		|---- simple_ch_prep_sm
49  *		|---- imp_def_interrupts
50  *	|---- dpN_<sink/src>
51  *		|---- max_word
52  *		|---- min_word
53  *		|---- words
54  *		|---- type
55  *		|---- max_grouping
56  *		|---- simple_ch_prep_sm
57  *		|---- ch_prep_timeout
58  *		|---- imp_def_interrupts
59  *		|---- min_ch
60  *		|---- max_ch
61  *		|---- channels
62  *		|---- ch_combinations
63  *		|---- max_async_buffer
64  *		|---- block_pack_mode
65  *		|---- port_encoding
66  *
67  */
68 
69 #define sdw_slave_attr(field, format_string)			\
70 static ssize_t field##_show(struct device *dev,			\
71 			    struct device_attribute *attr,	\
72 			    char *buf)				\
73 {								\
74 	struct sdw_slave *slave = dev_to_sdw_dev(dev);		\
75 	return sprintf(buf, format_string, slave->prop.field);	\
76 }								\
77 static DEVICE_ATTR_RO(field)
78 
79 sdw_slave_attr(mipi_revision, "0x%x\n");
80 sdw_slave_attr(wake_capable, "%d\n");
81 sdw_slave_attr(test_mode_capable, "%d\n");
82 sdw_slave_attr(clk_stop_mode1, "%d\n");
83 sdw_slave_attr(simple_clk_stop_capable, "%d\n");
84 sdw_slave_attr(clk_stop_timeout, "%d\n");
85 sdw_slave_attr(ch_prep_timeout, "%d\n");
86 sdw_slave_attr(reset_behave, "%d\n");
87 sdw_slave_attr(high_PHY_capable, "%d\n");
88 sdw_slave_attr(paging_support, "%d\n");
89 sdw_slave_attr(bank_delay_support, "%d\n");
90 sdw_slave_attr(p15_behave, "%d\n");
91 sdw_slave_attr(master_count, "%d\n");
92 sdw_slave_attr(source_ports, "0x%x\n");
93 sdw_slave_attr(sink_ports, "0x%x\n");
94 
95 static ssize_t modalias_show(struct device *dev,
96 			     struct device_attribute *attr, char *buf)
97 {
98 	struct sdw_slave *slave = dev_to_sdw_dev(dev);
99 
100 	return sdw_slave_modalias(slave, buf, 256);
101 }
102 static DEVICE_ATTR_RO(modalias);
103 
104 static struct attribute *slave_attrs[] = {
105 	&dev_attr_modalias.attr,
106 	NULL,
107 };
108 ATTRIBUTE_GROUPS(slave);
109 
110 static struct attribute *slave_dev_attrs[] = {
111 	&dev_attr_mipi_revision.attr,
112 	&dev_attr_wake_capable.attr,
113 	&dev_attr_test_mode_capable.attr,
114 	&dev_attr_clk_stop_mode1.attr,
115 	&dev_attr_simple_clk_stop_capable.attr,
116 	&dev_attr_clk_stop_timeout.attr,
117 	&dev_attr_ch_prep_timeout.attr,
118 	&dev_attr_reset_behave.attr,
119 	&dev_attr_high_PHY_capable.attr,
120 	&dev_attr_paging_support.attr,
121 	&dev_attr_bank_delay_support.attr,
122 	&dev_attr_p15_behave.attr,
123 	&dev_attr_master_count.attr,
124 	&dev_attr_source_ports.attr,
125 	&dev_attr_sink_ports.attr,
126 	NULL,
127 };
128 
129 /*
130  * we don't use ATTRIBUTES_GROUP here since we want to add a subdirectory
131  * for device-level properties
132  */
133 static const struct attribute_group sdw_slave_dev_attr_group = {
134 	.attrs	= slave_dev_attrs,
135 	.name = "dev-properties",
136 };
137 
138 /*
139  * DP0 sysfs
140  */
141 
142 #define sdw_dp0_attr(field, format_string)				\
143 static ssize_t field##_show(struct device *dev,				\
144 			    struct device_attribute *attr,		\
145 			    char *buf)					\
146 {									\
147 	struct sdw_slave *slave = dev_to_sdw_dev(dev);			\
148 	return sprintf(buf, format_string, slave->prop.dp0_prop->field);\
149 }									\
150 static DEVICE_ATTR_RO(field)
151 
152 sdw_dp0_attr(max_word, "%d\n");
153 sdw_dp0_attr(min_word, "%d\n");
154 sdw_dp0_attr(BRA_flow_controlled, "%d\n");
155 sdw_dp0_attr(simple_ch_prep_sm, "%d\n");
156 sdw_dp0_attr(imp_def_interrupts, "0x%x\n");
157 
158 static ssize_t words_show(struct device *dev,
159 			  struct device_attribute *attr, char *buf)
160 {
161 	struct sdw_slave *slave = dev_to_sdw_dev(dev);
162 	ssize_t size = 0;
163 	int i;
164 
165 	for (i = 0; i < slave->prop.dp0_prop->num_words; i++)
166 		size += sprintf(buf + size, "%d ",
167 				slave->prop.dp0_prop->words[i]);
168 	size += sprintf(buf + size, "\n");
169 
170 	return size;
171 }
172 static DEVICE_ATTR_RO(words);
173 
174 static struct attribute *dp0_attrs[] = {
175 	&dev_attr_max_word.attr,
176 	&dev_attr_min_word.attr,
177 	&dev_attr_words.attr,
178 	&dev_attr_BRA_flow_controlled.attr,
179 	&dev_attr_simple_ch_prep_sm.attr,
180 	&dev_attr_imp_def_interrupts.attr,
181 	NULL,
182 };
183 
184 /*
185  * we don't use ATTRIBUTES_GROUP here since we want to add a subdirectory
186  * for dp0-level properties
187  */
188 static const struct attribute_group dp0_group = {
189 	.attrs = dp0_attrs,
190 	.name = "dp0",
191 };
192 
193 int sdw_slave_sysfs_init(struct sdw_slave *slave)
194 {
195 	int ret;
196 
197 	ret = devm_device_add_groups(&slave->dev, slave_groups);
198 	if (ret < 0)
199 		return ret;
200 
201 	ret = devm_device_add_group(&slave->dev, &sdw_slave_dev_attr_group);
202 	if (ret < 0)
203 		return ret;
204 
205 	if (slave->prop.dp0_prop) {
206 		ret = devm_device_add_group(&slave->dev, &dp0_group);
207 		if (ret < 0)
208 			return ret;
209 	}
210 
211 	if (slave->prop.source_ports || slave->prop.sink_ports) {
212 		ret = sdw_slave_sysfs_dpn_init(slave);
213 		if (ret < 0)
214 			return ret;
215 	}
216 
217 	return 0;
218 }
219 
220 /*
221  * the status is shown in capital letters for UNATTACHED and RESERVED
222  * on purpose, to highligh users to the fact that these status values
223  * are not expected.
224  */
225 static const char *const slave_status[] = {
226 	[SDW_SLAVE_UNATTACHED] =  "UNATTACHED",
227 	[SDW_SLAVE_ATTACHED] = "Attached",
228 	[SDW_SLAVE_ALERT] = "Alert",
229 	[SDW_SLAVE_RESERVED] = "RESERVED",
230 };
231 
232 static ssize_t status_show(struct device *dev,
233 			   struct device_attribute *attr, char *buf)
234 {
235 	struct sdw_slave *slave = dev_to_sdw_dev(dev);
236 
237 	return sprintf(buf, "%s\n", slave_status[slave->status]);
238 }
239 static DEVICE_ATTR_RO(status);
240 
241 static ssize_t device_number_show(struct device *dev,
242 				  struct device_attribute *attr, char *buf)
243 {
244 	struct sdw_slave *slave = dev_to_sdw_dev(dev);
245 
246 	if (slave->status == SDW_SLAVE_UNATTACHED)
247 		return sprintf(buf, "%s", "N/A");
248 	else
249 		return sprintf(buf, "%d", slave->dev_num);
250 }
251 static DEVICE_ATTR_RO(device_number);
252 
253 static struct attribute *slave_status_attrs[] = {
254 	&dev_attr_status.attr,
255 	&dev_attr_device_number.attr,
256 	NULL,
257 };
258 
259 /*
260  * we don't use ATTRIBUTES_GROUP here since the group is used in a
261  * separate file and can't be handled as a static.
262  */
263 static const struct attribute_group sdw_slave_status_attr_group = {
264 	.attrs	= slave_status_attrs,
265 };
266 
267 const struct attribute_group *sdw_slave_status_attr_groups[] = {
268 	&sdw_slave_status_attr_group,
269 	NULL
270 };
271