blob: a47bb74da72b930cfe9a69ad11105c6b978c3b6d [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * shutdown.c - power management functions for the device tree.
3 *
4 * Copyright (c) 2002-3 Patrick Mochel
5 * 2002-3 Open Source Development Lab
6 *
7 * This file is released under the GPLv2
8 *
9 */
10
11#include <linux/config.h>
12#include <linux/device.h>
13#include <asm/semaphore.h>
14
15#include "power.h"
16
17#define to_dev(node) container_of(node, struct device, kobj.entry)
18
19extern struct subsystem devices_subsys;
20
21
Linus Torvalds1da177e2005-04-16 15:20:36 -070022/**
23 * We handle system devices differently - we suspend and shut them
24 * down last and resume them first. That way, we don't do anything stupid like
25 * shutting down the interrupt controller before any devices..
26 *
27 * Note that there are not different stages for power management calls -
28 * they only get one called once when interrupts are disabled.
29 */
30
31extern int sysdev_shutdown(void);
32
33/**
34 * device_shutdown - call ->shutdown() on each device to shutdown.
35 */
36void device_shutdown(void)
37{
38 struct device * dev;
39
40 down_write(&devices_subsys.rwsem);
David Brownell82428b62005-05-09 08:07:00 -070041 list_for_each_entry_reverse(dev, &devices_subsys.kset.list,
42 kobj.entry) {
Russell King594c8282006-01-05 14:29:51 +000043 if (dev->bus && dev->bus->shutdown) {
44 dev_dbg(dev, "shutdown\n");
45 dev->bus->shutdown(dev);
46 } else if (dev->driver && dev->driver->shutdown) {
David Brownell82428b62005-05-09 08:07:00 -070047 dev_dbg(dev, "shutdown\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -070048 dev->driver->shutdown(dev);
David Brownell82428b62005-05-09 08:07:00 -070049 }
Linus Torvalds1da177e2005-04-16 15:20:36 -070050 }
51 up_write(&devices_subsys.rwsem);
52
53 sysdev_shutdown();
54}
55