rtl8366-core.c 10.2 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13
// SPDX-License-Identifier: GPL-2.0
/* Realtek SMI library helpers for the RTL8366x variants
 * RTL8366RB and RTL8366S
 *
 * Copyright (C) 2017 Linus Walleij <linus.walleij@linaro.org>
 * Copyright (C) 2009-2010 Gabor Juhos <juhosg@openwrt.org>
 * Copyright (C) 2010 Antti Seppälä <a.seppala@gmail.com>
 * Copyright (C) 2010 Roman Yeryomin <roman@advem.lv>
 * Copyright (C) 2011 Colin Leitner <colin.leitner@googlemail.com>
 */
#include <linux/if_bridge.h>
#include <net/dsa.h>

14
#include "realtek.h"
15

16
int rtl8366_mc_is_used(struct realtek_priv *priv, int mc_index, int *used)
17 18 19 20 21
{
	int ret;
	int i;

	*used = 0;
22
	for (i = 0; i < priv->num_ports; i++) {
23 24
		int index = 0;

25
		ret = priv->ops->get_mc_index(priv, i, &index);
26 27 28 29 30 31 32 33 34 35 36 37 38
		if (ret)
			return ret;

		if (mc_index == index) {
			*used = 1;
			break;
		}
	}

	return 0;
}
EXPORT_SYMBOL_GPL(rtl8366_mc_is_used);

39 40
/**
 * rtl8366_obtain_mc() - retrieve or allocate a VLAN member configuration
41
 * @priv: the Realtek SMI device instance
42 43 44 45 46
 * @vid: the VLAN ID to look up or allocate
 * @vlanmc: the pointer will be assigned to a pointer to a valid member config
 * if successful
 * @return: index of a new member config or negative error number
 */
47
static int rtl8366_obtain_mc(struct realtek_priv *priv, int vid,
48 49 50 51 52 53 54
			     struct rtl8366_vlan_mc *vlanmc)
{
	struct rtl8366_vlan_4k vlan4k;
	int ret;
	int i;

	/* Try to find an existing member config entry for this VID */
55 56
	for (i = 0; i < priv->num_vlan_mc; i++) {
		ret = priv->ops->get_vlan_mc(priv, i, vlanmc);
57
		if (ret) {
58
			dev_err(priv->dev, "error searching for VLAN MC %d for VID %d\n",
59 60 61 62 63 64 65 66 67
				i, vid);
			return ret;
		}

		if (vid == vlanmc->vid)
			return i;
	}

	/* We have no MC entry for this VID, try to find an empty one */
68 69
	for (i = 0; i < priv->num_vlan_mc; i++) {
		ret = priv->ops->get_vlan_mc(priv, i, vlanmc);
70
		if (ret) {
71
			dev_err(priv->dev, "error searching for VLAN MC %d for VID %d\n",
72 73 74 75 76 77
				i, vid);
			return ret;
		}

		if (vlanmc->vid == 0 && vlanmc->member == 0) {
			/* Update the entry from the 4K table */
78
			ret = priv->ops->get_vlan_4k(priv, vid, &vlan4k);
79
			if (ret) {
80
				dev_err(priv->dev, "error looking for 4K VLAN MC %d for VID %d\n",
81 82 83 84 85 86 87 88
					i, vid);
				return ret;
			}

			vlanmc->vid = vid;
			vlanmc->member = vlan4k.member;
			vlanmc->untag = vlan4k.untag;
			vlanmc->fid = vlan4k.fid;
89
			ret = priv->ops->set_vlan_mc(priv, i, vlanmc);
90
			if (ret) {
91
				dev_err(priv->dev, "unable to set/update VLAN MC %d for VID %d\n",
92 93 94 95
					i, vid);
				return ret;
			}

96
			dev_dbg(priv->dev, "created new MC at index %d for VID %d\n",
97 98 99 100 101 102
				i, vid);
			return i;
		}
	}

	/* MC table is full, try to find an unused entry and replace it */
103
	for (i = 0; i < priv->num_vlan_mc; i++) {
104 105
		int used;

106
		ret = rtl8366_mc_is_used(priv, i, &used);
107 108 109 110 111
		if (ret)
			return ret;

		if (!used) {
			/* Update the entry from the 4K table */
112
			ret = priv->ops->get_vlan_4k(priv, vid, &vlan4k);
113 114 115 116 117 118 119
			if (ret)
				return ret;

			vlanmc->vid = vid;
			vlanmc->member = vlan4k.member;
			vlanmc->untag = vlan4k.untag;
			vlanmc->fid = vlan4k.fid;
120
			ret = priv->ops->set_vlan_mc(priv, i, vlanmc);
121
			if (ret) {
122
				dev_err(priv->dev, "unable to set/update VLAN MC %d for VID %d\n",
123 124 125
					i, vid);
				return ret;
			}
126
			dev_dbg(priv->dev, "recycled MC at index %i for VID %d\n",
127 128 129 130 131
				i, vid);
			return i;
		}
	}

132
	dev_err(priv->dev, "all VLAN member configurations are in use\n");
133 134 135
	return -ENOSPC;
}

136
int rtl8366_set_vlan(struct realtek_priv *priv, int vid, u32 member,
137 138
		     u32 untag, u32 fid)
{
139
	struct rtl8366_vlan_mc vlanmc;
140
	struct rtl8366_vlan_4k vlan4k;
141
	int mc;
142 143
	int ret;

144
	if (!priv->ops->is_vlan_valid(priv, vid))
145 146
		return -EINVAL;

147
	dev_dbg(priv->dev,
148 149 150
		"setting VLAN%d 4k members: 0x%02x, untagged: 0x%02x\n",
		vid, member, untag);

151
	/* Update the 4K table */
152
	ret = priv->ops->get_vlan_4k(priv, vid, &vlan4k);
153 154 155
	if (ret)
		return ret;

156 157
	vlan4k.member |= member;
	vlan4k.untag |= untag;
158
	vlan4k.fid = fid;
159
	ret = priv->ops->set_vlan_4k(priv, &vlan4k);
160 161 162
	if (ret)
		return ret;

163
	dev_dbg(priv->dev,
164 165 166
		"resulting VLAN%d 4k members: 0x%02x, untagged: 0x%02x\n",
		vid, vlan4k.member, vlan4k.untag);

167
	/* Find or allocate a member config for this VID */
168
	ret = rtl8366_obtain_mc(priv, vid, &vlanmc);
169 170 171
	if (ret < 0)
		return ret;
	mc = ret;
172

173 174 175 176
	/* Update the MC entry */
	vlanmc.member |= member;
	vlanmc.untag |= untag;
	vlanmc.fid = fid;
177

178
	/* Commit updates to the MC entry */
179
	ret = priv->ops->set_vlan_mc(priv, mc, &vlanmc);
180
	if (ret)
181
		dev_err(priv->dev, "failed to commit changes to VLAN MC index %d for VID %d\n",
182 183
			mc, vid);
	else
184
		dev_dbg(priv->dev,
185 186
			"resulting VLAN%d MC members: 0x%02x, untagged: 0x%02x\n",
			vid, vlanmc.member, vlanmc.untag);
187 188 189 190 191

	return ret;
}
EXPORT_SYMBOL_GPL(rtl8366_set_vlan);

192
int rtl8366_set_pvid(struct realtek_priv *priv, unsigned int port,
193 194 195
		     unsigned int vid)
{
	struct rtl8366_vlan_mc vlanmc;
196
	int mc;
197 198
	int ret;

199
	if (!priv->ops->is_vlan_valid(priv, vid))
200 201
		return -EINVAL;

202
	/* Find or allocate a member config for this VID */
203
	ret = rtl8366_obtain_mc(priv, vid, &vlanmc);
204 205 206
	if (ret < 0)
		return ret;
	mc = ret;
207

208
	ret = priv->ops->set_mc_index(priv, port, mc);
209
	if (ret) {
210
		dev_err(priv->dev, "set PVID: failed to set MC index %d for port %d\n",
211 212
			mc, port);
		return ret;
213 214
	}

215
	dev_dbg(priv->dev, "set PVID: the PVID for port %d set to %d using existing MC index %d\n",
216
		port, vid, mc);
217

218
	return 0;
219 220 221
}
EXPORT_SYMBOL_GPL(rtl8366_set_pvid);

222
int rtl8366_enable_vlan4k(struct realtek_priv *priv, bool enable)
223 224 225 226 227 228 229 230 231
{
	int ret;

	/* To enable 4k VLAN, ordinary VLAN must be enabled first,
	 * but if we disable 4k VLAN it is fine to leave ordinary
	 * VLAN enabled.
	 */
	if (enable) {
		/* Make sure VLAN is ON */
232
		ret = priv->ops->enable_vlan(priv, true);
233 234 235
		if (ret)
			return ret;

236
		priv->vlan_enabled = true;
237 238
	}

239
	ret = priv->ops->enable_vlan4k(priv, enable);
240 241 242
	if (ret)
		return ret;

243
	priv->vlan4k_enabled = enable;
244 245 246 247
	return 0;
}
EXPORT_SYMBOL_GPL(rtl8366_enable_vlan4k);

248
int rtl8366_enable_vlan(struct realtek_priv *priv, bool enable)
249 250 251
{
	int ret;

252
	ret = priv->ops->enable_vlan(priv, enable);
253 254 255
	if (ret)
		return ret;

256
	priv->vlan_enabled = enable;
257 258 259 260 261

	/* If we turn VLAN off, make sure that we turn off
	 * 4k VLAN as well, if that happened to be on.
	 */
	if (!enable) {
262 263
		priv->vlan4k_enabled = false;
		ret = priv->ops->enable_vlan4k(priv, false);
264 265 266 267 268 269
	}

	return ret;
}
EXPORT_SYMBOL_GPL(rtl8366_enable_vlan);

270
int rtl8366_reset_vlan(struct realtek_priv *priv)
271 272 273 274 275
{
	struct rtl8366_vlan_mc vlanmc;
	int ret;
	int i;

276 277
	rtl8366_enable_vlan(priv, false);
	rtl8366_enable_vlan4k(priv, false);
278 279 280 281 282 283 284

	/* Clear the 16 VLAN member configurations */
	vlanmc.vid = 0;
	vlanmc.priority = 0;
	vlanmc.member = 0;
	vlanmc.untag = 0;
	vlanmc.fid = 0;
285 286
	for (i = 0; i < priv->num_vlan_mc; i++) {
		ret = priv->ops->set_vlan_mc(priv, i, &vlanmc);
287 288 289 290 291 292 293 294
		if (ret)
			return ret;
	}

	return 0;
}
EXPORT_SYMBOL_GPL(rtl8366_reset_vlan);

295
int rtl8366_vlan_add(struct dsa_switch *ds, int port,
296 297
		     const struct switchdev_obj_port_vlan *vlan,
		     struct netlink_ext_ack *extack)
298
{
299 300
	bool untagged = !!(vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED);
	bool pvid = !!(vlan->flags & BRIDGE_VLAN_INFO_PVID);
301
	struct realtek_priv *priv = ds->priv;
302 303 304
	u32 member = 0;
	u32 untag = 0;
	int ret;
305

306
	if (!priv->ops->is_vlan_valid(priv, vlan->vid)) {
307
		NL_SET_ERR_MSG_MOD(extack, "VLAN ID not valid");
308
		return -EINVAL;
309
	}
310 311 312 313 314

	/* Enable VLAN in the hardware
	 * FIXME: what's with this 4k business?
	 * Just rtl8366_enable_vlan() seems inconclusive.
	 */
315
	ret = rtl8366_enable_vlan4k(priv, true);
316 317
	if (ret) {
		NL_SET_ERR_MSG_MOD(extack, "Failed to enable VLAN 4K");
318
		return ret;
319
	}
320

321
	dev_dbg(priv->dev, "add VLAN %d on port %d, %s, %s\n",
322 323
		vlan->vid, port, untagged ? "untagged" : "tagged",
		pvid ? "PVID" : "no PVID");
324

325
	member |= BIT(port);
326

327 328
	if (untagged)
		untag |= BIT(port);
329

330
	ret = rtl8366_set_vlan(priv, vlan->vid, member, untag, 0);
331
	if (ret) {
332
		dev_err(priv->dev, "failed to set up VLAN %04x", vlan->vid);
333 334
		return ret;
	}
335

336
	if (!pvid)
337
		return 0;
338

339
	ret = rtl8366_set_pvid(priv, port, vlan->vid);
340
	if (ret) {
341
		dev_err(priv->dev, "failed to set PVID on port %d to VLAN %04x",
342
			port, vlan->vid);
343 344
		return ret;
	}
345

346
	return 0;
347 348 349 350 351 352
}
EXPORT_SYMBOL_GPL(rtl8366_vlan_add);

int rtl8366_vlan_del(struct dsa_switch *ds, int port,
		     const struct switchdev_obj_port_vlan *vlan)
{
353
	struct realtek_priv *priv = ds->priv;
354
	int ret, i;
355

356
	dev_dbg(priv->dev, "del VLAN %d on port %d\n", vlan->vid, port);
357

358
	for (i = 0; i < priv->num_vlan_mc; i++) {
359
		struct rtl8366_vlan_mc vlanmc;
360

361
		ret = priv->ops->get_vlan_mc(priv, i, &vlanmc);
362 363
		if (ret)
			return ret;
364

365 366 367 368 369 370 371 372 373
		if (vlan->vid == vlanmc.vid) {
			/* Remove this port from the VLAN */
			vlanmc.member &= ~BIT(port);
			vlanmc.untag &= ~BIT(port);
			/*
			 * If no ports are members of this VLAN
			 * anymore then clear the whole member
			 * config so it can be reused.
			 */
374
			if (!vlanmc.member) {
375 376 377 378
				vlanmc.vid = 0;
				vlanmc.priority = 0;
				vlanmc.fid = 0;
			}
379
			ret = priv->ops->set_vlan_mc(priv, i, &vlanmc);
380
			if (ret) {
381
				dev_err(priv->dev,
382 383
					"failed to remove VLAN %04x\n",
					vlan->vid);
384 385
				return ret;
			}
386
			break;
387 388 389 390 391 392 393 394 395 396
		}
	}

	return 0;
}
EXPORT_SYMBOL_GPL(rtl8366_vlan_del);

void rtl8366_get_strings(struct dsa_switch *ds, int port, u32 stringset,
			 uint8_t *data)
{
397
	struct realtek_priv *priv = ds->priv;
398 399 400
	struct rtl8366_mib_counter *mib;
	int i;

401
	if (port >= priv->num_ports)
402 403
		return;

404 405
	for (i = 0; i < priv->num_mib_counters; i++) {
		mib = &priv->mib_counters[i];
406 407 408 409 410 411 412 413
		strncpy(data + i * ETH_GSTRING_LEN,
			mib->name, ETH_GSTRING_LEN);
	}
}
EXPORT_SYMBOL_GPL(rtl8366_get_strings);

int rtl8366_get_sset_count(struct dsa_switch *ds, int port, int sset)
{
414
	struct realtek_priv *priv = ds->priv;
415 416 417 418

	/* We only support SS_STATS */
	if (sset != ETH_SS_STATS)
		return 0;
419
	if (port >= priv->num_ports)
420 421
		return -EINVAL;

422
	return priv->num_mib_counters;
423 424 425 426 427
}
EXPORT_SYMBOL_GPL(rtl8366_get_sset_count);

void rtl8366_get_ethtool_stats(struct dsa_switch *ds, int port, uint64_t *data)
{
428
	struct realtek_priv *priv = ds->priv;
429 430 431
	int i;
	int ret;

432
	if (port >= priv->num_ports)
433 434
		return;

435
	for (i = 0; i < priv->num_mib_counters; i++) {
436 437 438
		struct rtl8366_mib_counter *mib;
		u64 mibvalue = 0;

439 440
		mib = &priv->mib_counters[i];
		ret = priv->ops->get_mib_counter(priv, port, mib, &mibvalue);
441
		if (ret) {
442
			dev_err(priv->dev, "error reading MIB counter %s\n",
443 444 445 446 447 448
				mib->name);
		}
		data[i] = mibvalue;
	}
}
EXPORT_SYMBOL_GPL(rtl8366_get_ethtool_stats);