container.c 3.0 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2
 * container.c  - ACPI Generic Container Driver
L
Linus Torvalds 已提交
3 4 5 6 7
 *
 * Copyright (C) 2004 Anil S Keshavamurthy (anil.s.keshavamurthy@intel.com)
 * Copyright (C) 2004 Keiichiro Tokunaga (tokunaga.keiich@jp.fujitsu.com)
 * Copyright (C) 2004 Motoyuki Ito (motoyuki@soft.fujitsu.com)
 * Copyright (C) 2004 FUJITSU LIMITED
8 9
 * Copyright (C) 2004, 2013 Intel Corp.
 * Author: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
L
Linus Torvalds 已提交
10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 *
 *  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.
 *
 *  This program is distributed in the hope that it will be useful, but
 *  WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 *  General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License along
 *  with this program; if not, write to the Free Software Foundation, Inc.,
 *  59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 */
#include <linux/acpi.h>
30
#include <linux/container.h>
L
Linus Torvalds 已提交
31

32 33
#include "internal.h"

34 35
#define PREFIX "ACPI: "

L
Linus Torvalds 已提交
36
#define _COMPONENT			ACPI_CONTAINER_COMPONENT
37
ACPI_MODULE_NAME("container");
L
Linus Torvalds 已提交
38

39 40 41 42 43 44 45
static const struct acpi_device_id container_device_ids[] = {
	{"ACPI0004", 0},
	{"PNP0A05", 0},
	{"PNP0A06", 0},
	{"", 0},
};

46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
static int acpi_container_offline(struct container_dev *cdev)
{
	struct acpi_device *adev = ACPI_COMPANION(&cdev->dev);
	struct acpi_device *child;

	/* Check all of the dependent devices' physical companions. */
	list_for_each_entry(child, &adev->children, node)
		if (!acpi_scan_is_offline(child, false))
			return -EBUSY;

	return 0;
}

static void acpi_container_release(struct device *dev)
{
	kfree(to_container_dev(dev));
}

64
static int container_device_attach(struct acpi_device *adev,
65 66
				   const struct acpi_device_id *not_used)
{
67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
	struct container_dev *cdev;
	struct device *dev;
	int ret;

	cdev = kzalloc(sizeof(*cdev), GFP_KERNEL);
	if (!cdev)
		return -ENOMEM;

	cdev->offline = acpi_container_offline;
	dev = &cdev->dev;
	dev->bus = &container_subsys;
	dev_set_name(dev, "%s", dev_name(&adev->dev));
	ACPI_COMPANION_SET(dev, adev);
	dev->release = acpi_container_release;
	ret = device_register(dev);
	if (ret)
		return ret;

	adev->driver_data = dev;
86 87 88
	return 1;
}

89 90
static void container_device_detach(struct acpi_device *adev)
{
91 92 93 94 95
	struct device *dev = acpi_driver_data(adev);

	adev->driver_data = NULL;
	if (dev)
		device_unregister(dev);
96 97
}

98
static struct acpi_scan_handler container_handler = {
99
	.ids = container_device_ids,
100
	.attach = container_device_attach,
101
	.detach = container_device_detach,
102 103
	.hotplug = {
		.enabled = true,
104
		.demand_offline = true,
105
	},
L
Linus Torvalds 已提交
106 107
};

108
void __init acpi_container_init(void)
L
Linus Torvalds 已提交
109
{
110
	acpi_scan_add_handler_with_hotplug(&container_handler, "container");
L
Linus Torvalds 已提交
111
}