2005-10-30 04:17:58 +07:00
|
|
|
/*
|
|
|
|
* platform_device.h - generic, centralized driver model
|
|
|
|
*
|
|
|
|
* Copyright (c) 2001-2003 Patrick Mochel <mochel@osdl.org>
|
|
|
|
*
|
|
|
|
* This file is released under the GPLv2
|
|
|
|
*
|
|
|
|
* See Documentation/driver-model/ for more information.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _PLATFORM_DEVICE_H_
|
|
|
|
#define _PLATFORM_DEVICE_H_
|
|
|
|
|
|
|
|
#include <linux/device.h>
|
2009-02-04 10:52:40 +07:00
|
|
|
#include <linux/mod_devicetable.h>
|
2005-10-30 04:17:58 +07:00
|
|
|
|
2012-07-28 03:14:59 +07:00
|
|
|
#define PLATFORM_DEVID_NONE (-1)
|
|
|
|
#define PLATFORM_DEVID_AUTO (-2)
|
|
|
|
|
2011-04-08 05:43:01 +07:00
|
|
|
struct mfd_cell;
|
|
|
|
|
2005-10-30 04:17:58 +07:00
|
|
|
struct platform_device {
|
|
|
|
const char * name;
|
2007-09-09 17:54:16 +07:00
|
|
|
int id;
|
2012-07-28 03:14:59 +07:00
|
|
|
bool id_auto;
|
2005-10-30 04:17:58 +07:00
|
|
|
struct device dev;
|
|
|
|
u32 num_resources;
|
|
|
|
struct resource * resource;
|
2009-02-04 10:52:40 +07:00
|
|
|
|
2010-01-01 14:43:28 +07:00
|
|
|
const struct platform_device_id *id_entry;
|
2009-07-08 18:21:31 +07:00
|
|
|
|
2011-04-08 05:43:01 +07:00
|
|
|
/* MFD cell pointer */
|
|
|
|
struct mfd_cell *mfd_cell;
|
|
|
|
|
2009-07-08 18:21:31 +07:00
|
|
|
/* arch specific additions */
|
|
|
|
struct pdev_archdata archdata;
|
2005-10-30 04:17:58 +07:00
|
|
|
};
|
|
|
|
|
2009-02-04 10:52:40 +07:00
|
|
|
#define platform_get_device_id(pdev) ((pdev)->id_entry)
|
|
|
|
|
2005-10-30 04:17:58 +07:00
|
|
|
#define to_platform_device(x) container_of((x), struct platform_device, dev)
|
|
|
|
|
|
|
|
extern int platform_device_register(struct platform_device *);
|
|
|
|
extern void platform_device_unregister(struct platform_device *);
|
|
|
|
|
|
|
|
extern struct bus_type platform_bus_type;
|
|
|
|
extern struct device platform_bus;
|
|
|
|
|
2011-06-10 13:52:57 +07:00
|
|
|
extern void arch_setup_pdev_archdata(struct platform_device *);
|
2005-10-30 04:17:58 +07:00
|
|
|
extern struct resource *platform_get_resource(struct platform_device *, unsigned int, unsigned int);
|
|
|
|
extern int platform_get_irq(struct platform_device *, unsigned int);
|
2009-04-27 07:38:16 +07:00
|
|
|
extern struct resource *platform_get_resource_byname(struct platform_device *, unsigned int, const char *);
|
|
|
|
extern int platform_get_irq_byname(struct platform_device *, const char *);
|
2005-10-30 04:17:58 +07:00
|
|
|
extern int platform_add_devices(struct platform_device **, int);
|
|
|
|
|
2011-08-25 16:16:00 +07:00
|
|
|
struct platform_device_info {
|
|
|
|
struct device *parent;
|
ACPI / platform: Initialize ACPI handles of platform devices in advance
The current platform device creation and registration code in
acpi_create_platform_device() is quite convoluted. This function
takes an ACPI device node as an argument and eventually calls
platform_device_register_resndata() to create and register a
platform device object on the basis of the information contained
in that code. However, it doesn't associate the new platform
device with the ACPI node directly, but instead it relies on
acpi_platform_notify(), called from within device_add(), to find
that ACPI node again with the help of acpi_platform_find_device()
and acpi_platform_match() and then attach the new platform device
to it. This causes an additional ACPI namespace walk to happen and
is clearly suboptimal.
Use the observation that it is now possible to initialize the ACPI
handle of a device before calling device_add() for it to make this
code more straightforward. Namely, add a new field to struct
platform_device_info allowing us to pass the ACPI handle of interest
to platform_device_register_full(), which will then use it to
initialize the new device's ACPI handle before registering it.
This will cause acpi_platform_notify() to use the ACPI handle from
the device structure directly instead of using the .find_device()
routine provided by the device's bus type. In consequence,
acpi_platform_bus, acpi_platform_find_device(), and
acpi_platform_match() are not necessary any more, so remove them.
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Reviewed-by: Mika Westerberg <mika.westerberg@linux.intel.com>
Acked-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-11-21 06:21:59 +07:00
|
|
|
struct acpi_dev_node acpi_node;
|
2011-08-25 16:16:00 +07:00
|
|
|
|
|
|
|
const char *name;
|
|
|
|
int id;
|
|
|
|
|
|
|
|
const struct resource *res;
|
|
|
|
unsigned int num_res;
|
|
|
|
|
|
|
|
const void *data;
|
|
|
|
size_t size_data;
|
|
|
|
u64 dma_mask;
|
|
|
|
};
|
|
|
|
extern struct platform_device *platform_device_register_full(
|
2011-12-09 04:53:29 +07:00
|
|
|
const struct platform_device_info *pdevinfo);
|
2011-08-25 16:16:00 +07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* platform_device_register_resndata - add a platform-level device with
|
|
|
|
* resources and platform-specific data
|
|
|
|
*
|
|
|
|
* @parent: parent device for the device we're adding
|
|
|
|
* @name: base name of the device we're adding
|
|
|
|
* @id: instance id
|
|
|
|
* @res: set of resources that needs to be allocated for the device
|
|
|
|
* @num: number of resources
|
|
|
|
* @data: platform specific data for this platform device
|
|
|
|
* @size: size of platform specific data
|
|
|
|
*
|
|
|
|
* Returns &struct platform_device pointer on success, or ERR_PTR() on error.
|
|
|
|
*/
|
|
|
|
static inline struct platform_device *platform_device_register_resndata(
|
2010-06-21 21:11:44 +07:00
|
|
|
struct device *parent, const char *name, int id,
|
|
|
|
const struct resource *res, unsigned int num,
|
2011-08-25 16:16:00 +07:00
|
|
|
const void *data, size_t size) {
|
|
|
|
|
|
|
|
struct platform_device_info pdevinfo = {
|
|
|
|
.parent = parent,
|
|
|
|
.name = name,
|
|
|
|
.id = id,
|
|
|
|
.res = res,
|
|
|
|
.num_res = num,
|
|
|
|
.data = data,
|
|
|
|
.size_data = size,
|
|
|
|
.dma_mask = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
return platform_device_register_full(&pdevinfo);
|
|
|
|
}
|
2010-06-21 21:11:44 +07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* platform_device_register_simple - add a platform-level device and its resources
|
|
|
|
* @name: base name of the device we're adding
|
|
|
|
* @id: instance id
|
|
|
|
* @res: set of resources that needs to be allocated for the device
|
|
|
|
* @num: number of resources
|
|
|
|
*
|
|
|
|
* This function creates a simple platform device that requires minimal
|
|
|
|
* resource and memory management. Canned release function freeing memory
|
|
|
|
* allocated for the device allows drivers using such devices to be
|
|
|
|
* unloaded without waiting for the last reference to the device to be
|
|
|
|
* dropped.
|
|
|
|
*
|
|
|
|
* This interface is primarily intended for use with legacy drivers which
|
|
|
|
* probe hardware directly. Because such drivers create sysfs device nodes
|
|
|
|
* themselves, rather than letting system infrastructure handle such device
|
|
|
|
* enumeration tasks, they don't fully conform to the Linux driver model.
|
|
|
|
* In particular, when such drivers are built as modules, they can't be
|
|
|
|
* "hotplugged".
|
|
|
|
*
|
|
|
|
* Returns &struct platform_device pointer on success, or ERR_PTR() on error.
|
|
|
|
*/
|
|
|
|
static inline struct platform_device *platform_device_register_simple(
|
|
|
|
const char *name, int id,
|
|
|
|
const struct resource *res, unsigned int num)
|
|
|
|
{
|
|
|
|
return platform_device_register_resndata(NULL, name, id,
|
|
|
|
res, num, NULL, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* platform_device_register_data - add a platform-level device with platform-specific data
|
|
|
|
* @parent: parent device for the device we're adding
|
|
|
|
* @name: base name of the device we're adding
|
|
|
|
* @id: instance id
|
|
|
|
* @data: platform specific data for this platform device
|
|
|
|
* @size: size of platform specific data
|
|
|
|
*
|
|
|
|
* This function creates a simple platform device that requires minimal
|
|
|
|
* resource and memory management. Canned release function freeing memory
|
|
|
|
* allocated for the device allows drivers using such devices to be
|
|
|
|
* unloaded without waiting for the last reference to the device to be
|
|
|
|
* dropped.
|
|
|
|
*
|
|
|
|
* Returns &struct platform_device pointer on success, or ERR_PTR() on error.
|
|
|
|
*/
|
|
|
|
static inline struct platform_device *platform_device_register_data(
|
|
|
|
struct device *parent, const char *name, int id,
|
|
|
|
const void *data, size_t size)
|
|
|
|
{
|
|
|
|
return platform_device_register_resndata(parent, name, id,
|
|
|
|
NULL, 0, data, size);
|
|
|
|
}
|
2005-10-30 04:17:58 +07:00
|
|
|
|
2007-09-09 17:54:16 +07:00
|
|
|
extern struct platform_device *platform_device_alloc(const char *name, int id);
|
2009-01-29 03:01:02 +07:00
|
|
|
extern int platform_device_add_resources(struct platform_device *pdev,
|
|
|
|
const struct resource *res,
|
|
|
|
unsigned int num);
|
2006-12-05 05:57:19 +07:00
|
|
|
extern int platform_device_add_data(struct platform_device *pdev, const void *data, size_t size);
|
2005-11-06 04:19:33 +07:00
|
|
|
extern int platform_device_add(struct platform_device *pdev);
|
2005-12-10 13:36:27 +07:00
|
|
|
extern void platform_device_del(struct platform_device *pdev);
|
2005-11-06 04:19:33 +07:00
|
|
|
extern void platform_device_put(struct platform_device *pdev);
|
|
|
|
|
2005-11-10 00:23:39 +07:00
|
|
|
struct platform_driver {
|
|
|
|
int (*probe)(struct platform_device *);
|
|
|
|
int (*remove)(struct platform_device *);
|
|
|
|
void (*shutdown)(struct platform_device *);
|
|
|
|
int (*suspend)(struct platform_device *, pm_message_t state);
|
|
|
|
int (*resume)(struct platform_device *);
|
|
|
|
struct device_driver driver;
|
2010-01-26 15:35:00 +07:00
|
|
|
const struct platform_device_id *id_table;
|
2005-11-10 00:23:39 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
extern int platform_driver_register(struct platform_driver *);
|
|
|
|
extern void platform_driver_unregister(struct platform_driver *);
|
|
|
|
|
2006-11-17 14:28:47 +07:00
|
|
|
/* non-hotpluggable platform devices may use this so that probe() and
|
|
|
|
* its support may live in __init sections, conserving runtime memory.
|
|
|
|
*/
|
|
|
|
extern int platform_driver_probe(struct platform_driver *driver,
|
|
|
|
int (*probe)(struct platform_device *));
|
|
|
|
|
2011-02-17 05:23:27 +07:00
|
|
|
static inline void *platform_get_drvdata(const struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
return dev_get_drvdata(&pdev->dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void platform_set_drvdata(struct platform_device *pdev, void *data)
|
|
|
|
{
|
|
|
|
dev_set_drvdata(&pdev->dev, data);
|
|
|
|
}
|
2005-11-10 00:23:39 +07:00
|
|
|
|
2011-10-06 00:29:49 +07:00
|
|
|
/* module_platform_driver() - Helper macro for drivers that don't do
|
|
|
|
* anything special in module init/exit. This eliminates a lot of
|
|
|
|
* boilerplate. Each module may only use this macro once, and
|
|
|
|
* calling it replaces module_init() and module_exit()
|
|
|
|
*/
|
|
|
|
#define module_platform_driver(__platform_driver) \
|
2011-11-16 16:13:35 +07:00
|
|
|
module_driver(__platform_driver, platform_driver_register, \
|
|
|
|
platform_driver_unregister)
|
2011-10-06 00:29:49 +07:00
|
|
|
|
2013-01-09 18:15:26 +07:00
|
|
|
/* module_platform_driver_probe() - Helper macro for drivers that don't do
|
|
|
|
* anything special in module init/exit. This eliminates a lot of
|
|
|
|
* boilerplate. Each module may only use this macro once, and
|
|
|
|
* calling it replaces module_init() and module_exit()
|
|
|
|
*/
|
|
|
|
#define module_platform_driver_probe(__platform_driver, __platform_probe) \
|
|
|
|
static int __init __platform_driver##_init(void) \
|
|
|
|
{ \
|
|
|
|
return platform_driver_probe(&(__platform_driver), \
|
|
|
|
__platform_probe); \
|
|
|
|
} \
|
|
|
|
module_init(__platform_driver##_init); \
|
|
|
|
static void __exit __platform_driver##_exit(void) \
|
|
|
|
{ \
|
|
|
|
platform_driver_unregister(&(__platform_driver)); \
|
|
|
|
} \
|
|
|
|
module_exit(__platform_driver##_exit);
|
|
|
|
|
2009-12-30 11:11:20 +07:00
|
|
|
extern struct platform_device *platform_create_bundle(struct platform_driver *driver,
|
|
|
|
int (*probe)(struct platform_device *),
|
|
|
|
struct resource *res, unsigned int n_res,
|
|
|
|
const void *data, size_t size);
|
|
|
|
|
2009-03-31 04:37:25 +07:00
|
|
|
/* early platform driver interface */
|
|
|
|
struct early_platform_driver {
|
|
|
|
const char *class_str;
|
|
|
|
struct platform_driver *pdrv;
|
|
|
|
struct list_head list;
|
|
|
|
int requested_id;
|
2009-11-27 15:38:51 +07:00
|
|
|
char *buffer;
|
|
|
|
int bufsize;
|
2009-03-31 04:37:25 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
#define EARLY_PLATFORM_ID_UNSET -2
|
|
|
|
#define EARLY_PLATFORM_ID_ERROR -3
|
|
|
|
|
|
|
|
extern int early_platform_driver_register(struct early_platform_driver *epdrv,
|
|
|
|
char *buf);
|
|
|
|
extern void early_platform_add_devices(struct platform_device **devs, int num);
|
|
|
|
|
|
|
|
static inline int is_early_platform_device(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
return !pdev->dev.driver;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern void early_platform_driver_register_all(char *class_str);
|
|
|
|
extern int early_platform_driver_probe(char *class_str,
|
|
|
|
int nr_probe, int user_only);
|
|
|
|
extern void early_platform_cleanup(void);
|
|
|
|
|
2009-11-27 15:38:51 +07:00
|
|
|
#define early_platform_init(class_string, platdrv) \
|
|
|
|
early_platform_init_buffer(class_string, platdrv, NULL, 0)
|
2009-03-31 04:37:25 +07:00
|
|
|
|
|
|
|
#ifndef MODULE
|
2009-11-27 15:38:51 +07:00
|
|
|
#define early_platform_init_buffer(class_string, platdrv, buf, bufsiz) \
|
2009-03-31 04:37:25 +07:00
|
|
|
static __initdata struct early_platform_driver early_driver = { \
|
|
|
|
.class_str = class_string, \
|
2009-11-27 15:38:51 +07:00
|
|
|
.buffer = buf, \
|
|
|
|
.bufsize = bufsiz, \
|
|
|
|
.pdrv = platdrv, \
|
2009-03-31 04:37:25 +07:00
|
|
|
.requested_id = EARLY_PLATFORM_ID_UNSET, \
|
|
|
|
}; \
|
2009-11-27 15:38:51 +07:00
|
|
|
static int __init early_platform_driver_setup_func(char *buffer) \
|
2009-03-31 04:37:25 +07:00
|
|
|
{ \
|
2009-11-27 15:38:51 +07:00
|
|
|
return early_platform_driver_register(&early_driver, buffer); \
|
2009-03-31 04:37:25 +07:00
|
|
|
} \
|
|
|
|
early_param(class_string, early_platform_driver_setup_func)
|
|
|
|
#else /* MODULE */
|
2009-11-27 15:38:51 +07:00
|
|
|
#define early_platform_init_buffer(class_string, platdrv, buf, bufsiz) \
|
|
|
|
static inline char *early_platform_driver_setup_func(void) \
|
|
|
|
{ \
|
|
|
|
return bufsiz ? buf : NULL; \
|
|
|
|
}
|
2009-03-31 04:37:25 +07:00
|
|
|
#endif /* MODULE */
|
|
|
|
|
2011-04-29 05:36:05 +07:00
|
|
|
#ifdef CONFIG_SUSPEND
|
|
|
|
extern int platform_pm_suspend(struct device *dev);
|
|
|
|
extern int platform_pm_resume(struct device *dev);
|
|
|
|
#else
|
|
|
|
#define platform_pm_suspend NULL
|
|
|
|
#define platform_pm_resume NULL
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_HIBERNATE_CALLBACKS
|
|
|
|
extern int platform_pm_freeze(struct device *dev);
|
|
|
|
extern int platform_pm_thaw(struct device *dev);
|
|
|
|
extern int platform_pm_poweroff(struct device *dev);
|
|
|
|
extern int platform_pm_restore(struct device *dev);
|
|
|
|
#else
|
|
|
|
#define platform_pm_freeze NULL
|
|
|
|
#define platform_pm_thaw NULL
|
|
|
|
#define platform_pm_poweroff NULL
|
|
|
|
#define platform_pm_restore NULL
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_PM_SLEEP
|
|
|
|
#define USE_PLATFORM_PM_SLEEP_OPS \
|
|
|
|
.suspend = platform_pm_suspend, \
|
|
|
|
.resume = platform_pm_resume, \
|
|
|
|
.freeze = platform_pm_freeze, \
|
|
|
|
.thaw = platform_pm_thaw, \
|
|
|
|
.poweroff = platform_pm_poweroff, \
|
2011-12-18 06:34:24 +07:00
|
|
|
.restore = platform_pm_restore,
|
2011-04-29 05:36:05 +07:00
|
|
|
#else
|
|
|
|
#define USE_PLATFORM_PM_SLEEP_OPS
|
|
|
|
#endif
|
|
|
|
|
2005-10-30 04:17:58 +07:00
|
|
|
#endif /* _PLATFORM_DEVICE_H_ */
|