2 * Copyright IBM Corp. 2013
3 * Author(s): Eugene Crosser <eugene.crosser@ru.ibm.com>
6 #include <linux/slab.h>
7 #include <asm/ebcdic.h>
10 #define QETH_DEVICE_ATTR(_id, _name, _mode, _show, _store) \
11 struct device_attribute dev_attr_##_id = __ATTR(_name, _mode, _show, _store)
13 static int qeth_card_hw_is_reachable(struct qeth_card *card)
15 return (card->state == CARD_STATE_SOFTSETUP) ||
16 (card->state == CARD_STATE_UP);
19 static ssize_t qeth_bridge_port_role_state_show(struct device *dev,
20 struct device_attribute *attr, char *buf,
23 struct qeth_card *card = dev_get_drvdata(dev);
24 enum qeth_sbp_states state = QETH_SBP_STATE_INACTIVE;
31 mutex_lock(&card->conf_mutex);
33 if (qeth_card_hw_is_reachable(card) &&
34 card->options.sbp.supported_funcs)
35 rc = qeth_bridgeport_query_ports(card,
36 &card->options.sbp.role, &state);
40 case QETH_SBP_STATE_INACTIVE:
41 word = "inactive"; break;
42 case QETH_SBP_STATE_STANDBY:
43 word = "standby"; break;
44 case QETH_SBP_STATE_ACTIVE:
45 word = "active"; break;
50 switch (card->options.sbp.role) {
51 case QETH_SBP_ROLE_NONE:
53 case QETH_SBP_ROLE_PRIMARY:
54 word = "primary"; break;
55 case QETH_SBP_ROLE_SECONDARY:
56 word = "secondary"; break;
61 QETH_CARD_TEXT_(card, 2, "SBP%02x:%02x",
62 card->options.sbp.role, state);
64 rc = sprintf(buf, "%s\n", word);
67 mutex_unlock(&card->conf_mutex);
72 static ssize_t qeth_bridge_port_role_show(struct device *dev,
73 struct device_attribute *attr, char *buf)
75 return qeth_bridge_port_role_state_show(dev, attr, buf, 0);
78 static ssize_t qeth_bridge_port_role_store(struct device *dev,
79 struct device_attribute *attr, const char *buf, size_t count)
81 struct qeth_card *card = dev_get_drvdata(dev);
83 enum qeth_sbp_roles role;
87 if (sysfs_streq(buf, "primary"))
88 role = QETH_SBP_ROLE_PRIMARY;
89 else if (sysfs_streq(buf, "secondary"))
90 role = QETH_SBP_ROLE_SECONDARY;
91 else if (sysfs_streq(buf, "none"))
92 role = QETH_SBP_ROLE_NONE;
96 mutex_lock(&card->conf_mutex);
98 if (qeth_card_hw_is_reachable(card)) {
99 rc = qeth_bridgeport_setrole(card, role);
101 card->options.sbp.role = role;
103 card->options.sbp.role = role;
105 mutex_unlock(&card->conf_mutex);
107 return rc ? rc : count;
110 static DEVICE_ATTR(bridge_role, 0644, qeth_bridge_port_role_show,
111 qeth_bridge_port_role_store);
113 static ssize_t qeth_bridge_port_state_show(struct device *dev,
114 struct device_attribute *attr, char *buf)
116 return qeth_bridge_port_role_state_show(dev, attr, buf, 1);
119 static DEVICE_ATTR(bridge_state, 0644, qeth_bridge_port_state_show,
122 static struct attribute *qeth_l2_bridgeport_attrs[] = {
123 &dev_attr_bridge_role.attr,
124 &dev_attr_bridge_state.attr,
128 static struct attribute_group qeth_l2_bridgeport_attr_group = {
129 .attrs = qeth_l2_bridgeport_attrs,
132 int qeth_l2_create_device_attributes(struct device *dev)
134 return sysfs_create_group(&dev->kobj, &qeth_l2_bridgeport_attr_group);
137 void qeth_l2_remove_device_attributes(struct device *dev)
139 sysfs_remove_group(&dev->kobj, &qeth_l2_bridgeport_attr_group);
143 * qeth_l2_setup_bridgeport_attrs() - set/restore attrs when turning online.
144 * @card: qeth_card structure pointer
146 * Note: this function is called with conf_mutex held by the caller
148 void qeth_l2_setup_bridgeport_attrs(struct qeth_card *card)
152 if (!card->options.sbp.supported_funcs)
154 if (card->options.sbp.role != QETH_SBP_ROLE_NONE) {
155 /* Conditional to avoid spurious error messages */
156 qeth_bridgeport_setrole(card, card->options.sbp.role);
157 /* Let the callback function refresh the stored role value. */
158 qeth_bridgeport_query_ports(card,
159 &card->options.sbp.role, NULL);