blob: 59e3aa538b4da4594456b0965bc0fe42a49379f3 [file] [log] [blame]
Eugene Crosserb4d72c02014-01-14 15:54:11 +01001/*
2 * Copyright IBM Corp. 2013
3 * Author(s): Eugene Crosser <eugene.crosser@ru.ibm.com>
4 */
5
6#include <linux/slab.h>
7#include <asm/ebcdic.h>
Eugene Crosser511c2442014-09-02 08:20:17 +02008#include "qeth_core.h"
Eugene Crosserb4d72c02014-01-14 15:54:11 +01009#include "qeth_l2.h"
10
11#define QETH_DEVICE_ATTR(_id, _name, _mode, _show, _store) \
12struct device_attribute dev_attr_##_id = __ATTR(_name, _mode, _show, _store)
13
Eugene Crosserb4d72c02014-01-14 15:54:11 +010014static ssize_t qeth_bridge_port_role_state_show(struct device *dev,
15 struct device_attribute *attr, char *buf,
16 int show_state)
17{
18 struct qeth_card *card = dev_get_drvdata(dev);
19 enum qeth_sbp_states state = QETH_SBP_STATE_INACTIVE;
20 int rc = 0;
21 char *word;
22
23 if (!card)
24 return -EINVAL;
25
26 mutex_lock(&card->conf_mutex);
27
28 if (qeth_card_hw_is_reachable(card) &&
29 card->options.sbp.supported_funcs)
30 rc = qeth_bridgeport_query_ports(card,
31 &card->options.sbp.role, &state);
32 if (!rc) {
33 if (show_state)
34 switch (state) {
35 case QETH_SBP_STATE_INACTIVE:
36 word = "inactive"; break;
37 case QETH_SBP_STATE_STANDBY:
38 word = "standby"; break;
39 case QETH_SBP_STATE_ACTIVE:
40 word = "active"; break;
41 default:
42 rc = -EIO;
43 }
44 else
45 switch (card->options.sbp.role) {
46 case QETH_SBP_ROLE_NONE:
47 word = "none"; break;
48 case QETH_SBP_ROLE_PRIMARY:
49 word = "primary"; break;
50 case QETH_SBP_ROLE_SECONDARY:
51 word = "secondary"; break;
52 default:
53 rc = -EIO;
54 }
55 if (rc)
56 QETH_CARD_TEXT_(card, 2, "SBP%02x:%02x",
57 card->options.sbp.role, state);
58 else
59 rc = sprintf(buf, "%s\n", word);
60 }
61
62 mutex_unlock(&card->conf_mutex);
63
64 return rc;
65}
66
67static ssize_t qeth_bridge_port_role_show(struct device *dev,
68 struct device_attribute *attr, char *buf)
69{
70 return qeth_bridge_port_role_state_show(dev, attr, buf, 0);
71}
72
73static ssize_t qeth_bridge_port_role_store(struct device *dev,
74 struct device_attribute *attr, const char *buf, size_t count)
75{
76 struct qeth_card *card = dev_get_drvdata(dev);
77 int rc = 0;
78 enum qeth_sbp_roles role;
79
80 if (!card)
81 return -EINVAL;
82 if (sysfs_streq(buf, "primary"))
83 role = QETH_SBP_ROLE_PRIMARY;
84 else if (sysfs_streq(buf, "secondary"))
85 role = QETH_SBP_ROLE_SECONDARY;
86 else if (sysfs_streq(buf, "none"))
87 role = QETH_SBP_ROLE_NONE;
88 else
89 return -EINVAL;
90
91 mutex_lock(&card->conf_mutex);
92
93 if (qeth_card_hw_is_reachable(card)) {
94 rc = qeth_bridgeport_setrole(card, role);
95 if (!rc)
96 card->options.sbp.role = role;
97 } else
98 card->options.sbp.role = role;
99
100 mutex_unlock(&card->conf_mutex);
101
102 return rc ? rc : count;
103}
104
105static DEVICE_ATTR(bridge_role, 0644, qeth_bridge_port_role_show,
106 qeth_bridge_port_role_store);
107
108static ssize_t qeth_bridge_port_state_show(struct device *dev,
109 struct device_attribute *attr, char *buf)
110{
111 return qeth_bridge_port_role_state_show(dev, attr, buf, 1);
112}
113
114static DEVICE_ATTR(bridge_state, 0644, qeth_bridge_port_state_show,
115 NULL);
116
Eugene Crosser9f48b9d2014-01-14 15:54:13 +0100117static ssize_t qeth_bridgeport_hostnotification_show(struct device *dev,
118 struct device_attribute *attr, char *buf)
119{
120 struct qeth_card *card = dev_get_drvdata(dev);
121 int enabled;
122
123 if (!card)
124 return -EINVAL;
125
126 mutex_lock(&card->conf_mutex);
127
128 enabled = card->options.sbp.hostnotification;
129
130 mutex_unlock(&card->conf_mutex);
131
132 return sprintf(buf, "%d\n", enabled);
133}
134
135static ssize_t qeth_bridgeport_hostnotification_store(struct device *dev,
136 struct device_attribute *attr, const char *buf, size_t count)
137{
138 struct qeth_card *card = dev_get_drvdata(dev);
139 int rc = 0;
140 int enable;
141
142 if (!card)
143 return -EINVAL;
144
145 if (sysfs_streq(buf, "0"))
146 enable = 0;
147 else if (sysfs_streq(buf, "1"))
148 enable = 1;
149 else
150 return -EINVAL;
151
152 mutex_lock(&card->conf_mutex);
153
154 if (qeth_card_hw_is_reachable(card)) {
155 rc = qeth_bridgeport_an_set(card, enable);
156 if (!rc)
157 card->options.sbp.hostnotification = enable;
158 } else
159 card->options.sbp.hostnotification = enable;
160
161 mutex_unlock(&card->conf_mutex);
162
163 return rc ? rc : count;
164}
165
166static DEVICE_ATTR(bridge_hostnotify, 0644,
167 qeth_bridgeport_hostnotification_show,
168 qeth_bridgeport_hostnotification_store);
169
Eugene Crosserb4d72c02014-01-14 15:54:11 +0100170static struct attribute *qeth_l2_bridgeport_attrs[] = {
171 &dev_attr_bridge_role.attr,
172 &dev_attr_bridge_state.attr,
Eugene Crosser9f48b9d2014-01-14 15:54:13 +0100173 &dev_attr_bridge_hostnotify.attr,
Eugene Crosserb4d72c02014-01-14 15:54:11 +0100174 NULL,
175};
176
177static struct attribute_group qeth_l2_bridgeport_attr_group = {
178 .attrs = qeth_l2_bridgeport_attrs,
179};
180
181int qeth_l2_create_device_attributes(struct device *dev)
182{
183 return sysfs_create_group(&dev->kobj, &qeth_l2_bridgeport_attr_group);
184}
185
186void qeth_l2_remove_device_attributes(struct device *dev)
187{
188 sysfs_remove_group(&dev->kobj, &qeth_l2_bridgeport_attr_group);
189}
190
191/**
192 * qeth_l2_setup_bridgeport_attrs() - set/restore attrs when turning online.
193 * @card: qeth_card structure pointer
194 *
195 * Note: this function is called with conf_mutex held by the caller
196 */
197void qeth_l2_setup_bridgeport_attrs(struct qeth_card *card)
198{
Eugene Crosser9f48b9d2014-01-14 15:54:13 +0100199 int rc;
200
Eugene Crosserb4d72c02014-01-14 15:54:11 +0100201 if (!card)
202 return;
203 if (!card->options.sbp.supported_funcs)
204 return;
205 if (card->options.sbp.role != QETH_SBP_ROLE_NONE) {
206 /* Conditional to avoid spurious error messages */
207 qeth_bridgeport_setrole(card, card->options.sbp.role);
208 /* Let the callback function refresh the stored role value. */
209 qeth_bridgeport_query_ports(card,
210 &card->options.sbp.role, NULL);
211 }
Eugene Crosser9f48b9d2014-01-14 15:54:13 +0100212 if (card->options.sbp.hostnotification) {
213 rc = qeth_bridgeport_an_set(card, 1);
214 if (rc)
215 card->options.sbp.hostnotification = 0;
216 } else
217 qeth_bridgeport_an_set(card, 0);
Eugene Crosserb4d72c02014-01-14 15:54:11 +0100218}