1 // SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause)
2 // Copyright(c) 2015-17 Intel Corporation.
4 #include <linux/acpi.h>
6 #include <linux/soundwire/sdw.h>
7 #include <linux/soundwire/sdw_type.h>
9 #include "sysfs_local.h"
11 static void sdw_slave_release(struct device
*dev
)
13 struct sdw_slave
*slave
= dev_to_sdw_dev(dev
);
18 struct device_type sdw_slave_type
= {
20 .release
= sdw_slave_release
,
21 .uevent
= sdw_slave_uevent
,
24 int sdw_slave_add(struct sdw_bus
*bus
,
25 struct sdw_slave_id
*id
, struct fwnode_handle
*fwnode
)
27 struct sdw_slave
*slave
;
31 slave
= kzalloc(sizeof(*slave
), GFP_KERNEL
);
35 /* Initialize data structure */
36 memcpy(&slave
->id
, id
, sizeof(*id
));
37 slave
->dev
.parent
= bus
->dev
;
38 slave
->dev
.fwnode
= fwnode
;
40 if (id
->unique_id
== SDW_IGNORED_UNIQUE_ID
) {
41 /* name shall be sdw:link:mfg:part:class */
42 dev_set_name(&slave
->dev
, "sdw:%x:%x:%x:%x",
43 bus
->link_id
, id
->mfg_id
, id
->part_id
,
46 /* name shall be sdw:link:mfg:part:class:unique */
47 dev_set_name(&slave
->dev
, "sdw:%x:%x:%x:%x:%x",
48 bus
->link_id
, id
->mfg_id
, id
->part_id
,
49 id
->class_id
, id
->unique_id
);
52 slave
->dev
.bus
= &sdw_bus_type
;
53 slave
->dev
.of_node
= of_node_get(to_of_node(fwnode
));
54 slave
->dev
.type
= &sdw_slave_type
;
55 slave
->dev
.groups
= sdw_slave_status_attr_groups
;
57 slave
->status
= SDW_SLAVE_UNATTACHED
;
58 init_completion(&slave
->enumeration_complete
);
59 init_completion(&slave
->initialization_complete
);
61 init_completion(&slave
->probe_complete
);
62 slave
->probed
= false;
63 slave
->first_interrupt_done
= false;
65 for (i
= 0; i
< SDW_MAX_PORTS
; i
++)
66 init_completion(&slave
->port_ready
[i
]);
68 mutex_lock(&bus
->bus_lock
);
69 list_add_tail(&slave
->node
, &bus
->slaves
);
70 mutex_unlock(&bus
->bus_lock
);
72 ret
= device_register(&slave
->dev
);
74 dev_err(bus
->dev
, "Failed to add slave: ret %d\n", ret
);
77 * On err, don't free but drop ref as this will be freed
78 * when release method is invoked.
80 mutex_lock(&bus
->bus_lock
);
81 list_del(&slave
->node
);
82 mutex_unlock(&bus
->bus_lock
);
83 put_device(&slave
->dev
);
87 sdw_slave_debugfs_init(slave
);
92 #if IS_ENABLED(CONFIG_ACPI)
94 static bool find_slave(struct sdw_bus
*bus
,
95 struct acpi_device
*adev
,
96 struct sdw_slave_id
*id
)
98 unsigned long long addr
;
102 status
= acpi_evaluate_integer(adev
->handle
,
103 METHOD_NAME__ADR
, NULL
, &addr
);
105 if (ACPI_FAILURE(status
)) {
106 dev_err(bus
->dev
, "_ADR resolution failed: %x\n",
111 /* Extract link id from ADR, Bit 51 to 48 (included) */
112 link_id
= SDW_DISCO_LINK_ID(addr
);
114 /* Check for link_id match */
115 if (link_id
!= bus
->link_id
)
118 sdw_extract_slave_id(bus
, addr
, id
);
124 * sdw_acpi_find_slaves() - Find Slave devices in Master ACPI node
125 * @bus: SDW bus instance
127 * Scans Master ACPI node for SDW child Slave devices and registers it.
129 int sdw_acpi_find_slaves(struct sdw_bus
*bus
)
131 struct acpi_device
*adev
, *parent
;
132 struct acpi_device
*adev2
, *parent2
;
134 parent
= ACPI_COMPANION(bus
->dev
);
136 dev_err(bus
->dev
, "Can't find parent for acpi bind\n");
140 list_for_each_entry(adev
, &parent
->children
, node
) {
141 struct sdw_slave_id id
;
142 struct sdw_slave_id id2
;
143 bool ignore_unique_id
= true;
145 if (!find_slave(bus
, adev
, &id
))
148 /* brute-force O(N^2) search for duplicates */
150 list_for_each_entry(adev2
, &parent2
->children
, node
) {
155 if (!find_slave(bus
, adev2
, &id2
))
158 if (id
.sdw_version
!= id2
.sdw_version
||
159 id
.mfg_id
!= id2
.mfg_id
||
160 id
.part_id
!= id2
.part_id
||
161 id
.class_id
!= id2
.class_id
)
164 if (id
.unique_id
!= id2
.unique_id
) {
166 "Valid unique IDs %x %x for Slave mfg %x part %d\n",
167 id
.unique_id
, id2
.unique_id
,
168 id
.mfg_id
, id
.part_id
);
169 ignore_unique_id
= false;
172 "Invalid unique IDs %x %x for Slave mfg %x part %d\n",
173 id
.unique_id
, id2
.unique_id
,
174 id
.mfg_id
, id
.part_id
);
179 if (ignore_unique_id
)
180 id
.unique_id
= SDW_IGNORED_UNIQUE_ID
;
183 * don't error check for sdw_slave_add as we want to continue
186 sdw_slave_add(bus
, &id
, acpi_fwnode_handle(adev
));
195 * sdw_of_find_slaves() - Find Slave devices in master device tree node
196 * @bus: SDW bus instance
198 * Scans Master DT node for SDW child Slave devices and registers it.
200 int sdw_of_find_slaves(struct sdw_bus
*bus
)
202 struct device
*dev
= bus
->dev
;
203 struct device_node
*node
;
205 for_each_child_of_node(bus
->dev
->of_node
, node
) {
206 int link_id
, ret
, len
;
207 unsigned int sdw_version
;
208 const char *compat
= NULL
;
209 struct sdw_slave_id id
;
212 compat
= of_get_property(node
, "compatible", NULL
);
216 ret
= sscanf(compat
, "sdw%01x%04hx%04hx%02hhx", &sdw_version
,
217 &id
.mfg_id
, &id
.part_id
, &id
.class_id
);
220 dev_err(dev
, "Invalid compatible string found %s\n",
225 addr
= of_get_property(node
, "reg", &len
);
226 if (!addr
|| (len
< 2 * sizeof(u32
))) {
227 dev_err(dev
, "Invalid Link and Instance ID\n");
231 link_id
= be32_to_cpup(addr
++);
232 id
.unique_id
= be32_to_cpup(addr
);
233 id
.sdw_version
= sdw_version
;
235 /* Check for link_id match */
236 if (link_id
!= bus
->link_id
)
239 sdw_slave_add(bus
, &id
, of_fwnode_handle(node
));