| // SPDX-License-Identifier: GPL-2.0-only |
| // Copyright(c) 2015-2020 Intel Corporation. |
| |
| #include <linux/device.h> |
| #include <linux/mod_devicetable.h> |
| #include <linux/slab.h> |
| #include <linux/sysfs.h> |
| #include <linux/soundwire/sdw.h> |
| #include <linux/soundwire/sdw_type.h> |
| #include "bus.h" |
| #include "sysfs_local.h" |
| |
| /* |
| * Slave sysfs |
| */ |
| |
| /* |
| * The sysfs for Slave reflects the MIPI description as given |
| * in the MIPI DisCo spec. |
| * status and device_number come directly from the MIPI SoundWire |
| * 1.x specification. |
| * |
| * Base file is device |
| * |---- status |
| * |---- device_number |
| * |---- modalias |
| * |---- dev-properties |
| * |---- mipi_revision |
| * |---- wake_capable |
| * |---- test_mode_capable |
| * |---- clk_stop_mode1 |
| * |---- simple_clk_stop_capable |
| * |---- clk_stop_timeout |
| * |---- ch_prep_timeout |
| * |---- reset_behave |
| * |---- high_PHY_capable |
| * |---- paging_support |
| * |---- bank_delay_support |
| * |---- p15_behave |
| * |---- master_count |
| * |---- source_ports |
| * |---- sink_ports |
| * |---- dp0 |
| * |---- max_word |
| * |---- min_word |
| * |---- words |
| * |---- BRA_flow_controlled |
| * |---- simple_ch_prep_sm |
| * |---- imp_def_interrupts |
| * |---- dpN_<sink/src> |
| * |---- max_word |
| * |---- min_word |
| * |---- words |
| * |---- type |
| * |---- max_grouping |
| * |---- simple_ch_prep_sm |
| * |---- ch_prep_timeout |
| * |---- imp_def_interrupts |
| * |---- min_ch |
| * |---- max_ch |
| * |---- channels |
| * |---- ch_combinations |
| * |---- max_async_buffer |
| * |---- block_pack_mode |
| * |---- port_encoding |
| * |
| */ |
| |
| #define sdw_slave_attr(field, format_string) \ |
| static ssize_t field##_show(struct device *dev, \ |
| struct device_attribute *attr, \ |
| char *buf) \ |
| { \ |
| struct sdw_slave *slave = dev_to_sdw_dev(dev); \ |
| return sprintf(buf, format_string, slave->prop.field); \ |
| } \ |
| static DEVICE_ATTR_RO(field) |
| |
| sdw_slave_attr(mipi_revision, "0x%x\n"); |
| sdw_slave_attr(wake_capable, "%d\n"); |
| sdw_slave_attr(test_mode_capable, "%d\n"); |
| sdw_slave_attr(clk_stop_mode1, "%d\n"); |
| sdw_slave_attr(simple_clk_stop_capable, "%d\n"); |
| sdw_slave_attr(clk_stop_timeout, "%d\n"); |
| sdw_slave_attr(ch_prep_timeout, "%d\n"); |
| sdw_slave_attr(reset_behave, "%d\n"); |
| sdw_slave_attr(high_PHY_capable, "%d\n"); |
| sdw_slave_attr(paging_support, "%d\n"); |
| sdw_slave_attr(bank_delay_support, "%d\n"); |
| sdw_slave_attr(p15_behave, "%d\n"); |
| sdw_slave_attr(master_count, "%d\n"); |
| sdw_slave_attr(source_ports, "0x%x\n"); |
| sdw_slave_attr(sink_ports, "0x%x\n"); |
| |
| static ssize_t modalias_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct sdw_slave *slave = dev_to_sdw_dev(dev); |
| |
| return sdw_slave_modalias(slave, buf, 256); |
| } |
| static DEVICE_ATTR_RO(modalias); |
| |
| static struct attribute *slave_attrs[] = { |
| &dev_attr_modalias.attr, |
| NULL, |
| }; |
| ATTRIBUTE_GROUPS(slave); |
| |
| static struct attribute *slave_dev_attrs[] = { |
| &dev_attr_mipi_revision.attr, |
| &dev_attr_wake_capable.attr, |
| &dev_attr_test_mode_capable.attr, |
| &dev_attr_clk_stop_mode1.attr, |
| &dev_attr_simple_clk_stop_capable.attr, |
| &dev_attr_clk_stop_timeout.attr, |
| &dev_attr_ch_prep_timeout.attr, |
| &dev_attr_reset_behave.attr, |
| &dev_attr_high_PHY_capable.attr, |
| &dev_attr_paging_support.attr, |
| &dev_attr_bank_delay_support.attr, |
| &dev_attr_p15_behave.attr, |
| &dev_attr_master_count.attr, |
| &dev_attr_source_ports.attr, |
| &dev_attr_sink_ports.attr, |
| NULL, |
| }; |
| |
| /* |
| * we don't use ATTRIBUTES_GROUP here since we want to add a subdirectory |
| * for device-level properties |
| */ |
| static const struct attribute_group sdw_slave_dev_attr_group = { |
| .attrs = slave_dev_attrs, |
| .name = "dev-properties", |
| }; |
| |
| /* |
| * DP0 sysfs |
| */ |
| |
| #define sdw_dp0_attr(field, format_string) \ |
| static ssize_t field##_show(struct device *dev, \ |
| struct device_attribute *attr, \ |
| char *buf) \ |
| { \ |
| struct sdw_slave *slave = dev_to_sdw_dev(dev); \ |
| return sprintf(buf, format_string, slave->prop.dp0_prop->field);\ |
| } \ |
| static DEVICE_ATTR_RO(field) |
| |
| sdw_dp0_attr(max_word, "%d\n"); |
| sdw_dp0_attr(min_word, "%d\n"); |
| sdw_dp0_attr(BRA_flow_controlled, "%d\n"); |
| sdw_dp0_attr(simple_ch_prep_sm, "%d\n"); |
| sdw_dp0_attr(imp_def_interrupts, "0x%x\n"); |
| |
| static ssize_t words_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct sdw_slave *slave = dev_to_sdw_dev(dev); |
| ssize_t size = 0; |
| int i; |
| |
| for (i = 0; i < slave->prop.dp0_prop->num_words; i++) |
| size += sprintf(buf + size, "%d ", |
| slave->prop.dp0_prop->words[i]); |
| size += sprintf(buf + size, "\n"); |
| |
| return size; |
| } |
| static DEVICE_ATTR_RO(words); |
| |
| static struct attribute *dp0_attrs[] = { |
| &dev_attr_max_word.attr, |
| &dev_attr_min_word.attr, |
| &dev_attr_words.attr, |
| &dev_attr_BRA_flow_controlled.attr, |
| &dev_attr_simple_ch_prep_sm.attr, |
| &dev_attr_imp_def_interrupts.attr, |
| NULL, |
| }; |
| |
| /* |
| * we don't use ATTRIBUTES_GROUP here since we want to add a subdirectory |
| * for dp0-level properties |
| */ |
| static const struct attribute_group dp0_group = { |
| .attrs = dp0_attrs, |
| .name = "dp0", |
| }; |
| |
| int sdw_slave_sysfs_init(struct sdw_slave *slave) |
| { |
| int ret; |
| |
| ret = devm_device_add_groups(&slave->dev, slave_groups); |
| if (ret < 0) |
| return ret; |
| |
| ret = devm_device_add_group(&slave->dev, &sdw_slave_dev_attr_group); |
| if (ret < 0) |
| return ret; |
| |
| if (slave->prop.dp0_prop) { |
| ret = devm_device_add_group(&slave->dev, &dp0_group); |
| if (ret < 0) |
| return ret; |
| } |
| |
| if (slave->prop.source_ports || slave->prop.sink_ports) { |
| ret = sdw_slave_sysfs_dpn_init(slave); |
| if (ret < 0) |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| /* |
| * the status is shown in capital letters for UNATTACHED and RESERVED |
| * on purpose, to highligh users to the fact that these status values |
| * are not expected. |
| */ |
| static const char *const slave_status[] = { |
| [SDW_SLAVE_UNATTACHED] = "UNATTACHED", |
| [SDW_SLAVE_ATTACHED] = "Attached", |
| [SDW_SLAVE_ALERT] = "Alert", |
| [SDW_SLAVE_RESERVED] = "RESERVED", |
| }; |
| |
| static ssize_t status_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct sdw_slave *slave = dev_to_sdw_dev(dev); |
| |
| return sprintf(buf, "%s\n", slave_status[slave->status]); |
| } |
| static DEVICE_ATTR_RO(status); |
| |
| static ssize_t device_number_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct sdw_slave *slave = dev_to_sdw_dev(dev); |
| |
| if (slave->status == SDW_SLAVE_UNATTACHED) |
| return sprintf(buf, "%s", "N/A"); |
| else |
| return sprintf(buf, "%d", slave->dev_num); |
| } |
| static DEVICE_ATTR_RO(device_number); |
| |
| static struct attribute *slave_status_attrs[] = { |
| &dev_attr_status.attr, |
| &dev_attr_device_number.attr, |
| NULL, |
| }; |
| |
| /* |
| * we don't use ATTRIBUTES_GROUP here since the group is used in a |
| * separate file and can't be handled as a static. |
| */ |
| static const struct attribute_group sdw_slave_status_attr_group = { |
| .attrs = slave_status_attrs, |
| }; |
| |
| const struct attribute_group *sdw_slave_status_attr_groups[] = { |
| &sdw_slave_status_attr_group, |
| NULL |
| }; |