mdio_bus.c 17.3 KB
Newer Older
S
Sergei Shtylyov 已提交
1
/* MDIO Bus interface
2 3 4 5 6 7 8 9 10 11 12
 *
 * Author: Andy Fleming
 *
 * Copyright (c) 2004 Freescale Semiconductor, Inc.
 *
 * This program is free software; you can redistribute  it and/or modify it
 * under  the terms of  the GNU General  Public License as published by the
 * Free Software Foundation;  either version 2 of the  License, or (at your
 * option) any later version.
 *
 */
J
Joe Perches 已提交
13 14 15

#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt

16 17 18 19 20 21 22 23
#include <linux/kernel.h>
#include <linux/string.h>
#include <linux/errno.h>
#include <linux/unistd.h>
#include <linux/slab.h>
#include <linux/interrupt.h>
#include <linux/init.h>
#include <linux/delay.h>
24
#include <linux/device.h>
25 26
#include <linux/gpio.h>
#include <linux/gpio/consumer.h>
27
#include <linux/of_device.h>
28
#include <linux/of_mdio.h>
29
#include <linux/of_gpio.h>
30 31 32 33 34 35 36 37 38
#include <linux/netdevice.h>
#include <linux/etherdevice.h>
#include <linux/skbuff.h>
#include <linux/spinlock.h>
#include <linux/mm.h>
#include <linux/module.h>
#include <linux/mii.h>
#include <linux/ethtool.h>
#include <linux/phy.h>
S
Sergei Shtylyov 已提交
39 40
#include <linux/io.h>
#include <linux/uaccess.h>
41 42 43

#include <asm/irq.h>

44 45 46
#define CREATE_TRACE_POINTS
#include <trace/events/mdio.h>

47 48
#include "mdio-boardinfo.h"

49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90
int mdiobus_register_device(struct mdio_device *mdiodev)
{
	if (mdiodev->bus->mdio_map[mdiodev->addr])
		return -EBUSY;

	mdiodev->bus->mdio_map[mdiodev->addr] = mdiodev;

	return 0;
}
EXPORT_SYMBOL(mdiobus_register_device);

int mdiobus_unregister_device(struct mdio_device *mdiodev)
{
	if (mdiodev->bus->mdio_map[mdiodev->addr] != mdiodev)
		return -EINVAL;

	mdiodev->bus->mdio_map[mdiodev->addr] = NULL;

	return 0;
}
EXPORT_SYMBOL(mdiobus_unregister_device);

struct phy_device *mdiobus_get_phy(struct mii_bus *bus, int addr)
{
	struct mdio_device *mdiodev = bus->mdio_map[addr];

	if (!mdiodev)
		return NULL;

	if (!(mdiodev->flags & MDIO_DEVICE_FLAG_PHY))
		return NULL;

	return container_of(mdiodev, struct phy_device, mdio);
}
EXPORT_SYMBOL(mdiobus_get_phy);

bool mdiobus_is_registered_device(struct mii_bus *bus, int addr)
{
	return bus->mdio_map[addr];
}
EXPORT_SYMBOL(mdiobus_is_registered_device);

91
/**
92
 * mdiobus_alloc_size - allocate a mii_bus structure
93 94
 * @size: extra amount of memory to allocate for private storage.
 * If non-zero, then bus->priv is points to that memory.
95 96 97 98
 *
 * Description: called by a bus driver to allocate an mii_bus
 * structure to fill in.
 */
99
struct mii_bus *mdiobus_alloc_size(size_t size)
100
{
101
	struct mii_bus *bus;
102 103
	size_t aligned_size = ALIGN(sizeof(*bus), NETDEV_ALIGN);
	size_t alloc_size;
104
	int i;
105 106 107 108 109 110

	/* If we alloc extra space, it should be aligned */
	if (size)
		alloc_size = aligned_size + size;
	else
		alloc_size = sizeof(*bus);
111

112
	bus = kzalloc(alloc_size, GFP_KERNEL);
113 114 115 116 117 118
	if (!bus)
		return NULL;

	bus->state = MDIOBUS_ALLOCATED;
	if (size)
		bus->priv = (void *)bus + aligned_size;
119

120 121 122 123
	/* Initialise the interrupts to polling */
	for (i = 0; i < PHY_MAX_ADDR; i++)
		bus->irq[i] = PHY_POLL;

124
	return bus;
125
}
126
EXPORT_SYMBOL(mdiobus_alloc_size);
127

128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175
static void _devm_mdiobus_free(struct device *dev, void *res)
{
	mdiobus_free(*(struct mii_bus **)res);
}

static int devm_mdiobus_match(struct device *dev, void *res, void *data)
{
	struct mii_bus **r = res;

	if (WARN_ON(!r || !*r))
		return 0;

	return *r == data;
}

/**
 * devm_mdiobus_alloc_size - Resource-managed mdiobus_alloc_size()
 * @dev:		Device to allocate mii_bus for
 * @sizeof_priv:	Space to allocate for private structure.
 *
 * Managed mdiobus_alloc_size. mii_bus allocated with this function is
 * automatically freed on driver detach.
 *
 * If an mii_bus allocated with this function needs to be freed separately,
 * devm_mdiobus_free() must be used.
 *
 * RETURNS:
 * Pointer to allocated mii_bus on success, NULL on failure.
 */
struct mii_bus *devm_mdiobus_alloc_size(struct device *dev, int sizeof_priv)
{
	struct mii_bus **ptr, *bus;

	ptr = devres_alloc(_devm_mdiobus_free, sizeof(*ptr), GFP_KERNEL);
	if (!ptr)
		return NULL;

	/* use raw alloc_dr for kmalloc caller tracing */
	bus = mdiobus_alloc_size(sizeof_priv);
	if (bus) {
		*ptr = bus;
		devres_add(dev, ptr);
	} else {
		devres_free(ptr);
	}

	return bus;
}
176
EXPORT_SYMBOL_GPL(devm_mdiobus_alloc_size);
177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194

/**
 * devm_mdiobus_free - Resource-managed mdiobus_free()
 * @dev:		Device this mii_bus belongs to
 * @bus:		the mii_bus associated with the device
 *
 * Free mii_bus allocated with devm_mdiobus_alloc_size().
 */
void devm_mdiobus_free(struct device *dev, struct mii_bus *bus)
{
	int rc;

	rc = devres_release(dev, _devm_mdiobus_free,
			    devm_mdiobus_match, bus);
	WARN_ON(rc);
}
EXPORT_SYMBOL_GPL(devm_mdiobus_free);

195 196
/**
 * mdiobus_release - mii_bus device release callback
R
Randy Dunlap 已提交
197
 * @d: the target struct device that contains the mii_bus
198 199 200 201 202 203 204
 *
 * Description: called when the last reference to an mii_bus is
 * dropped, to free the underlying memory.
 */
static void mdiobus_release(struct device *d)
{
	struct mii_bus *bus = to_mii_bus(d);
K
Krzysztof Halasa 已提交
205 206 207
	BUG_ON(bus->state != MDIOBUS_RELEASED &&
	       /* for compatibility with error handling in drivers */
	       bus->state != MDIOBUS_ALLOCATED);
208 209 210 211 212 213 214 215
	kfree(bus);
}

static struct class mdio_bus_class = {
	.name		= "mdio_bus",
	.dev_release	= mdiobus_release,
};

216
#if IS_ENABLED(CONFIG_OF_MDIO)
217
/* Helper function for of_mdio_find_bus */
218
static int of_mdio_bus_match(struct device *dev, const void *mdio_bus_np)
219 220 221 222 223
{
	return dev->of_node == mdio_bus_np;
}
/**
 * of_mdio_find_bus - Given an mii_bus node, find the mii_bus.
224
 * @mdio_bus_np: Pointer to the mii_bus.
225
 *
226 227 228
 * Returns a reference to the mii_bus, or NULL if none found.  The
 * embedded struct device will have its reference count incremented,
 * and this must be put once the bus is finished with.
229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247
 *
 * Because the association of a device_node and mii_bus is made via
 * of_mdiobus_register(), the mii_bus cannot be found before it is
 * registered with of_mdiobus_register().
 *
 */
struct mii_bus *of_mdio_find_bus(struct device_node *mdio_bus_np)
{
	struct device *d;

	if (!mdio_bus_np)
		return NULL;

	d = class_find_device(&mdio_bus_class, NULL,  mdio_bus_np,
			      of_mdio_bus_match);

	return d ? to_mii_bus(d) : NULL;
}
EXPORT_SYMBOL(of_mdio_find_bus);
248

249 250 251 252 253
/* Walk the list of subnodes of a mdio bus and look for a node that
 * matches the mdio device's address with its 'reg' property. If
 * found, set the of_node pointer for the mdio device. This allows
 * auto-probed phy devices to be supplied with information passed in
 * via DT.
254
 */
255 256
static void of_mdiobus_link_mdiodev(struct mii_bus *bus,
				    struct mdio_device *mdiodev)
257
{
258
	struct device *dev = &mdiodev->dev;
259 260
	struct device_node *child;

A
Andrew Lunn 已提交
261
	if (dev->of_node || !bus->dev.of_node)
262 263
		return;

A
Andrew Lunn 已提交
264
	for_each_available_child_of_node(bus->dev.of_node, child) {
265 266
		int addr;

J
Jon Mason 已提交
267 268
		addr = of_mdio_parse_addr(dev, child);
		if (addr < 0)
269 270
			continue;

271
		if (addr == mdiodev->addr) {
272 273 274 275 276 277
			dev->of_node = child;
			return;
		}
	}
}
#else /* !IS_ENABLED(CONFIG_OF_MDIO) */
278 279
static inline void of_mdiobus_link_mdiodev(struct mii_bus *mdio,
					   struct mdio_device *mdiodev)
280 281
{
}
282 283
#endif

284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313
/**
 * mdiobus_create_device_from_board_info - create a full MDIO device given
 * a mdio_board_info structure
 * @bus: MDIO bus to create the devices on
 * @bi: mdio_board_info structure describing the devices
 *
 * Returns 0 on success or < 0 on error.
 */
static int mdiobus_create_device(struct mii_bus *bus,
				 struct mdio_board_info *bi)
{
	struct mdio_device *mdiodev;
	int ret = 0;

	mdiodev = mdio_device_create(bus, bi->mdio_addr);
	if (IS_ERR(mdiodev))
		return -ENODEV;

	strncpy(mdiodev->modalias, bi->modalias,
		sizeof(mdiodev->modalias));
	mdiodev->bus_match = mdio_device_bus_match;
	mdiodev->dev.platform_data = (void *)bi->platform_data;

	ret = mdio_device_register(mdiodev);
	if (ret)
		mdio_device_free(mdiodev);

	return ret;
}

314
/**
315
 * __mdiobus_register - bring up all the PHYs on a given bus and attach them to bus
316
 * @bus: target mii_bus
317
 * @owner: module containing bus accessor functions
A
Andy Fleming 已提交
318
 *
319
 * Description: Called by a bus driver to bring up all the PHYs
320 321
 *   on a given bus, and attach them to the bus. Drivers should use
 *   mdiobus_register() rather than __mdiobus_register() unless they
322 323 324
 *   need to pass a specific owner module. MDIO devices which are not
 *   PHYs will not be brought up by this function. They are expected to
 *   to be explicitly listed in DT and instantiated by of_mdiobus_register().
325 326
 *
 * Returns 0 on success or < 0 on error.
A
Andy Fleming 已提交
327
 */
R
Russell King 已提交
328
int __mdiobus_register(struct mii_bus *bus, struct module *owner)
A
Andy Fleming 已提交
329
{
330
	struct mdio_device *mdiodev;
K
Krzysztof Halasa 已提交
331
	int i, err;
332
	struct gpio_desc *gpiod;
A
Andy Fleming 已提交
333 334

	if (NULL == bus || NULL == bus->name ||
S
Sergei Shtylyov 已提交
335
	    NULL == bus->read || NULL == bus->write)
A
Andy Fleming 已提交
336 337
		return -EINVAL;

338 339 340
	BUG_ON(bus->state != MDIOBUS_ALLOCATED &&
	       bus->state != MDIOBUS_UNREGISTERED);

R
Russell King 已提交
341
	bus->owner = owner;
342 343 344
	bus->dev.parent = bus->parent;
	bus->dev.class = &mdio_bus_class;
	bus->dev.groups = NULL;
345
	dev_set_name(&bus->dev, "%s", bus->id);
346 347 348

	err = device_register(&bus->dev);
	if (err) {
J
Joe Perches 已提交
349
		pr_err("mii_bus %s failed to register\n", bus->id);
350
		put_device(&bus->dev);
351 352 353
		return -EINVAL;
	}

354 355
	mutex_init(&bus->mdio_lock);

356 357 358 359 360 361 362 363 364
	/* de-assert bus level PHY GPIO reset */
	gpiod = devm_gpiod_get(&bus->dev, "reset", GPIOD_OUT_LOW);
	if (IS_ERR(gpiod)) {
		err = PTR_ERR(gpiod);
		if (err != -ENOENT) {
			dev_err(&bus->dev,
				"mii_bus %s couldn't get reset GPIO\n",
				bus->id);
			return err;
365
		}
366 367 368 369 370 371
	} else	{
		bus->reset_gpiod = gpiod;

		gpiod_set_value_cansleep(gpiod, 1);
		udelay(bus->reset_delay_us);
		gpiod_set_value_cansleep(gpiod, 0);
372 373
	}

374 375 376
	if (bus->reset)
		bus->reset(bus);

A
Andy Fleming 已提交
377
	for (i = 0; i < PHY_MAX_ADDR; i++) {
378 379
		if ((bus->phy_mask & (1 << i)) == 0) {
			struct phy_device *phydev;
A
Andy Fleming 已提交
380

381
			phydev = mdiobus_scan(bus, i);
382
			if (IS_ERR(phydev) && (PTR_ERR(phydev) != -ENODEV)) {
383
				err = PTR_ERR(phydev);
K
Krzysztof Halasa 已提交
384 385
				goto error;
			}
386
		}
A
Andy Fleming 已提交
387 388
	}

389
	mdiobus_setup_mdiodev_from_board_info(bus, mdiobus_create_device);
390

K
Krzysztof Halasa 已提交
391
	bus->state = MDIOBUS_REGISTERED;
A
Andy Fleming 已提交
392
	pr_info("%s: probed\n", bus->name);
K
Krzysztof Halasa 已提交
393
	return 0;
A
Andy Fleming 已提交
394

K
Krzysztof Halasa 已提交
395 396
error:
	while (--i >= 0) {
397 398 399 400 401 402
		mdiodev = bus->mdio_map[i];
		if (!mdiodev)
			continue;

		mdiodev->device_remove(mdiodev);
		mdiodev->device_free(mdiodev);
K
Krzysztof Halasa 已提交
403
	}
404 405

	/* Put PHYs in RESET to save power */
406 407
	if (bus->reset_gpiod)
		gpiod_set_value_cansleep(bus->reset_gpiod, 1);
408

K
Krzysztof Halasa 已提交
409
	device_del(&bus->dev);
A
Andy Fleming 已提交
410 411
	return err;
}
R
Russell King 已提交
412
EXPORT_SYMBOL(__mdiobus_register);
A
Andy Fleming 已提交
413 414 415

void mdiobus_unregister(struct mii_bus *bus)
{
416
	struct mdio_device *mdiodev;
A
Andy Fleming 已提交
417 418
	int i;

419 420 421
	BUG_ON(bus->state != MDIOBUS_REGISTERED);
	bus->state = MDIOBUS_UNREGISTERED;

A
Andy Fleming 已提交
422
	for (i = 0; i < PHY_MAX_ADDR; i++) {
423 424 425 426
		mdiodev = bus->mdio_map[i];
		if (!mdiodev)
			continue;

427 428
		mdiodev->device_remove(mdiodev);
		mdiodev->device_free(mdiodev);
A
Andy Fleming 已提交
429
	}
430 431

	/* Put PHYs in RESET to save power */
432 433
	if (bus->reset_gpiod)
		gpiod_set_value_cansleep(bus->reset_gpiod, 1);
434

435
	device_del(&bus->dev);
A
Andy Fleming 已提交
436 437 438
}
EXPORT_SYMBOL(mdiobus_unregister);

439 440 441 442
/**
 * mdiobus_free - free a struct mii_bus
 * @bus: mii_bus to free
 *
443 444 445
 * This function releases the reference to the underlying device
 * object in the mii_bus.  If this is the last reference, the mii_bus
 * will be freed.
446 447 448
 */
void mdiobus_free(struct mii_bus *bus)
{
S
Sergei Shtylyov 已提交
449
	/* For compatibility with error handling in drivers. */
450 451 452 453 454 455 456 457 458
	if (bus->state == MDIOBUS_ALLOCATED) {
		kfree(bus);
		return;
	}

	BUG_ON(bus->state != MDIOBUS_UNREGISTERED);
	bus->state = MDIOBUS_RELEASED;

	put_device(&bus->dev);
459 460 461
}
EXPORT_SYMBOL(mdiobus_free);

462 463 464 465 466 467 468 469 470 471 472 473
/**
 * mdiobus_scan - scan a bus for MDIO devices.
 * @bus: mii_bus to scan
 * @addr: address on bus to scan
 *
 * This function scans the MDIO bus, looking for devices which can be
 * identified using a vendor/product ID in registers 2 and 3. Not all
 * MDIO devices have such registers, but PHY devices typically
 * do. Hence this function assumes anything found is a PHY, or can be
 * treated as a PHY. Other MDIO devices, such as switches, will
 * probably not be found during the scan.
 */
474 475 476 477 478
struct phy_device *mdiobus_scan(struct mii_bus *bus, int addr)
{
	struct phy_device *phydev;
	int err;

479
	phydev = get_phy_device(bus, addr, false);
480
	if (IS_ERR(phydev))
481 482
		return phydev;

483 484 485 486
	/*
	 * For DT, see if the auto-probed phy has a correspoding child
	 * in the bus node, and set the of_node pointer in this case.
	 */
487
	of_mdiobus_link_mdiodev(bus, &phydev->mdio);
488

489
	err = phy_device_register(phydev);
490 491
	if (err) {
		phy_device_free(phydev);
492
		return ERR_PTR(-ENODEV);
493 494 495 496 497 498
	}

	return phydev;
}
EXPORT_SYMBOL(mdiobus_scan);

499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517
/**
 * mdiobus_read_nested - Nested version of the mdiobus_read function
 * @bus: the mii_bus struct
 * @addr: the phy address
 * @regnum: register number to read
 *
 * In case of nested MDIO bus access avoid lockdep false positives by
 * using mutex_lock_nested().
 *
 * NOTE: MUST NOT be called from interrupt context,
 * because the bus read/write functions may wait for an interrupt
 * to conclude the operation.
 */
int mdiobus_read_nested(struct mii_bus *bus, int addr, u32 regnum)
{
	int retval;

	BUG_ON(in_interrupt());

518
	mutex_lock_nested(&bus->mdio_lock, MDIO_MUTEX_NESTED);
519 520 521
	retval = bus->read(bus, addr, regnum);
	mutex_unlock(&bus->mdio_lock);

522 523
	trace_mdio_access(bus, 1, addr, regnum, retval, retval);

524 525 526 527
	return retval;
}
EXPORT_SYMBOL(mdiobus_read_nested);

528 529 530 531 532 533 534 535 536 537
/**
 * mdiobus_read - Convenience function for reading a given MII mgmt register
 * @bus: the mii_bus struct
 * @addr: the phy address
 * @regnum: register number to read
 *
 * NOTE: MUST NOT be called from interrupt context,
 * because the bus read/write functions may wait for an interrupt
 * to conclude the operation.
 */
538
int mdiobus_read(struct mii_bus *bus, int addr, u32 regnum)
539 540 541 542 543 544 545 546 547
{
	int retval;

	BUG_ON(in_interrupt());

	mutex_lock(&bus->mdio_lock);
	retval = bus->read(bus, addr, regnum);
	mutex_unlock(&bus->mdio_lock);

548 549
	trace_mdio_access(bus, 1, addr, regnum, retval, retval);

550 551 552 553
	return retval;
}
EXPORT_SYMBOL(mdiobus_read);

554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573
/**
 * mdiobus_write_nested - Nested version of the mdiobus_write function
 * @bus: the mii_bus struct
 * @addr: the phy address
 * @regnum: register number to write
 * @val: value to write to @regnum
 *
 * In case of nested MDIO bus access avoid lockdep false positives by
 * using mutex_lock_nested().
 *
 * NOTE: MUST NOT be called from interrupt context,
 * because the bus read/write functions may wait for an interrupt
 * to conclude the operation.
 */
int mdiobus_write_nested(struct mii_bus *bus, int addr, u32 regnum, u16 val)
{
	int err;

	BUG_ON(in_interrupt());

574
	mutex_lock_nested(&bus->mdio_lock, MDIO_MUTEX_NESTED);
575 576 577
	err = bus->write(bus, addr, regnum, val);
	mutex_unlock(&bus->mdio_lock);

578 579
	trace_mdio_access(bus, 0, addr, regnum, val, err);

580 581 582 583
	return err;
}
EXPORT_SYMBOL(mdiobus_write_nested);

584 585 586 587 588 589 590 591 592 593 594
/**
 * mdiobus_write - Convenience function for writing a given MII mgmt register
 * @bus: the mii_bus struct
 * @addr: the phy address
 * @regnum: register number to write
 * @val: value to write to @regnum
 *
 * NOTE: MUST NOT be called from interrupt context,
 * because the bus read/write functions may wait for an interrupt
 * to conclude the operation.
 */
595
int mdiobus_write(struct mii_bus *bus, int addr, u32 regnum, u16 val)
596 597 598 599 600 601 602 603 604
{
	int err;

	BUG_ON(in_interrupt());

	mutex_lock(&bus->mdio_lock);
	err = bus->write(bus, addr, regnum, val);
	mutex_unlock(&bus->mdio_lock);

605 606
	trace_mdio_access(bus, 0, addr, regnum, val, err);

607 608 609 610
	return err;
}
EXPORT_SYMBOL(mdiobus_write);

611
/**
612 613 614 615
 * mdio_bus_match - determine if given MDIO driver supports the given
 *		    MDIO device
 * @dev: target MDIO device
 * @drv: given MDIO driver
616
 *
617 618 619 620
 * Description: Given a MDIO device, and a MDIO driver, return 1 if
 *   the driver supports the device.  Otherwise, return 0. This may
 *   require calling the devices own match function, since different classes
 *   of MDIO devices have different match criteria.
621 622 623
 */
static int mdio_bus_match(struct device *dev, struct device_driver *drv)
{
624
	struct mdio_device *mdio = to_mdio_device(dev);
625

626 627 628
	if (of_driver_match_device(dev, drv))
		return 1;

629 630
	if (mdio->bus_match)
		return mdio->bus_match(dev, drv);
631

632
	return 0;
633 634
}

635 636 637 638 639 640 641 642 643 644 645 646
static int mdio_uevent(struct device *dev, struct kobj_uevent_env *env)
{
	int rc;

	/* Some devices have extra OF data and an OF-style MODALIAS */
	rc = of_device_uevent_modalias(dev, env);
	if (rc != -ENODEV)
		return rc;

	return 0;
}

647 648
#ifdef CONFIG_PM
static int mdio_bus_suspend(struct device *dev)
649
{
650
	struct mdio_device *mdio = to_mdio_device(dev);
651

652 653
	if (mdio->pm_ops && mdio->pm_ops->suspend)
		return mdio->pm_ops->suspend(dev);
A
Anton Vorontsov 已提交
654

655
	return 0;
656 657
}

658
static int mdio_bus_resume(struct device *dev)
659
{
660
	struct mdio_device *mdio = to_mdio_device(dev);
A
Anton Vorontsov 已提交
661

662 663
	if (mdio->pm_ops && mdio->pm_ops->resume)
		return mdio->pm_ops->resume(dev);
A
Anton Vorontsov 已提交
664 665

	return 0;
666 667
}

668 669
static int mdio_bus_restore(struct device *dev)
{
670
	struct mdio_device *mdio = to_mdio_device(dev);
671

672 673
	if (mdio->pm_ops && mdio->pm_ops->restore)
		return mdio->pm_ops->restore(dev);
674 675 676 677

	return 0;
}

S
Sergei Shtylyov 已提交
678
static const struct dev_pm_ops mdio_bus_pm_ops = {
679 680 681 682 683 684 685 686 687 688 689 690 691 692 693
	.suspend = mdio_bus_suspend,
	.resume = mdio_bus_resume,
	.freeze = mdio_bus_suspend,
	.thaw = mdio_bus_resume,
	.restore = mdio_bus_restore,
};

#define MDIO_BUS_PM_OPS (&mdio_bus_pm_ops)

#else

#define MDIO_BUS_PM_OPS NULL

#endif /* CONFIG_PM */

694 695 696
struct bus_type mdio_bus_type = {
	.name		= "mdio_bus",
	.match		= mdio_bus_match,
697
	.uevent		= mdio_uevent,
698
	.pm		= MDIO_BUS_PM_OPS,
699
};
700
EXPORT_SYMBOL(mdio_bus_type);
701

702
int __init mdio_bus_init(void)
703
{
704 705 706 707 708 709 710 711 712 713
	int ret;

	ret = class_register(&mdio_bus_class);
	if (!ret) {
		ret = bus_register(&mdio_bus_type);
		if (ret)
			class_unregister(&mdio_bus_class);
	}

	return ret;
714
}
715
EXPORT_SYMBOL_GPL(mdio_bus_init);
716

717
#if IS_ENABLED(CONFIG_PHYLIB)
718
void mdio_bus_exit(void)
A
Andy Fleming 已提交
719
{
720
	class_unregister(&mdio_bus_class);
A
Andy Fleming 已提交
721 722
	bus_unregister(&mdio_bus_type);
}
723 724 725 726 727 728 729
EXPORT_SYMBOL_GPL(mdio_bus_exit);
#else
module_init(mdio_bus_init);
/* no module_exit, intentional */
MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("MDIO bus/device layer");
#endif