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