platform_device.h 10.8 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14
/*
 * 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>
15
#include <linux/mod_devicetable.h>
16

17 18 19
#define PLATFORM_DEVID_NONE	(-1)
#define PLATFORM_DEVID_AUTO	(-2)

20 21
struct mfd_cell;

22
struct platform_device {
23
	const char	*name;
24
	int		id;
25
	bool		id_auto;
26 27
	struct device	dev;
	u32		num_resources;
28
	struct resource	*resource;
29

30
	const struct platform_device_id	*id_entry;
31
	char *driver_override; /* Driver name to force a match */
32

33 34 35
	/* MFD cell pointer */
	struct mfd_cell *mfd_cell;

36 37
	/* arch specific additions */
	struct pdev_archdata	archdata;
38 39
};

40 41
#define platform_get_device_id(pdev)	((pdev)->id_entry)

42 43 44 45 46 47 48 49
#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;

50
extern void arch_setup_pdev_archdata(struct platform_device *);
51 52
extern struct resource *platform_get_resource(struct platform_device *,
					      unsigned int, unsigned int);
53
extern int platform_get_irq(struct platform_device *, unsigned int);
54 55 56
extern struct resource *platform_get_resource_byname(struct platform_device *,
						     unsigned int,
						     const char *);
57
extern int platform_get_irq_byname(struct platform_device *, const char *);
58 59
extern int platform_add_devices(struct platform_device **, int);

60 61
struct platform_device_info {
		struct device *parent;
62
		struct fwnode_handle *fwnode;
63 64 65 66 67 68 69 70 71 72 73 74

		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(
75
		const struct platform_device_info *pdevinfo);
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91

/**
 * 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(
92 93
		struct device *parent, const char *name, int id,
		const struct resource *res, unsigned int num,
94 95 96 97 98 99 100 101 102 103 104 105 106 107 108
		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);
}
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 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

/**
 * 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);
}
163

164
extern struct platform_device *platform_device_alloc(const char *name, int id);
165 166 167
extern int platform_device_add_resources(struct platform_device *pdev,
					 const struct resource *res,
					 unsigned int num);
168 169
extern int platform_device_add_data(struct platform_device *pdev,
				    const void *data, size_t size);
170
extern int platform_device_add(struct platform_device *pdev);
171
extern void platform_device_del(struct platform_device *pdev);
172 173
extern void platform_device_put(struct platform_device *pdev);

R
Russell King 已提交
174 175 176 177 178 179 180
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;
181
	const struct platform_device_id *id_table;
182
	bool prevent_deferred_probe;
R
Russell King 已提交
183 184
};

185 186 187
#define to_platform_driver(drv)	(container_of((drv), struct platform_driver, \
				 driver))

188 189 190 191 192 193 194
/*
 * use a macro to avoid include chaining to get THIS_MODULE
 */
#define platform_driver_register(drv) \
	__platform_driver_register(drv, THIS_MODULE)
extern int __platform_driver_register(struct platform_driver *,
					struct module *);
R
Russell King 已提交
195 196
extern void platform_driver_unregister(struct platform_driver *);

197 198 199
/* non-hotpluggable platform devices may use this so that probe() and
 * its support may live in __init sections, conserving runtime memory.
 */
200 201 202 203
#define platform_driver_probe(drv, probe) \
	__platform_driver_probe(drv, probe, THIS_MODULE)
extern int __platform_driver_probe(struct platform_driver *driver,
		int (*probe)(struct platform_device *), struct module *module);
204

205 206 207 208 209
static inline void *platform_get_drvdata(const struct platform_device *pdev)
{
	return dev_get_drvdata(&pdev->dev);
}

210 211
static inline void platform_set_drvdata(struct platform_device *pdev,
					void *data)
212 213 214
{
	dev_set_drvdata(&pdev->dev, data);
}
R
Russell King 已提交
215

216 217 218 219 220 221
/* 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) \
222 223
	module_driver(__platform_driver, platform_driver_register, \
			platform_driver_unregister)
224

225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242
/* 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);

243 244 245
#define platform_create_bundle(driver, probe, res, n_res, data, size) \
	__platform_create_bundle(driver, probe, res, n_res, data, size, THIS_MODULE)
extern struct platform_device *__platform_create_bundle(
246 247
	struct platform_driver *driver, int (*probe)(struct platform_device *),
	struct resource *res, unsigned int n_res,
248
	const void *data, size_t size, struct module *module);
249

M
Magnus Damm 已提交
250 251 252 253 254 255
/* early platform driver interface */
struct early_platform_driver {
	const char *class_str;
	struct platform_driver *pdrv;
	struct list_head list;
	int requested_id;
256 257
	char *buffer;
	int bufsize;
M
Magnus Damm 已提交
258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276
};

#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);

277 278
#define early_platform_init(class_string, platdrv)		\
	early_platform_init_buffer(class_string, platdrv, NULL, 0)
M
Magnus Damm 已提交
279 280

#ifndef MODULE
281
#define early_platform_init_buffer(class_string, platdrv, buf, bufsiz)	\
M
Magnus Damm 已提交
282 283
static __initdata struct early_platform_driver early_driver = {		\
	.class_str = class_string,					\
284 285 286
	.buffer = buf,							\
	.bufsize = bufsiz,						\
	.pdrv = platdrv,						\
M
Magnus Damm 已提交
287 288
	.requested_id = EARLY_PLATFORM_ID_UNSET,			\
};									\
289
static int __init early_platform_driver_setup_func(char *buffer)	\
M
Magnus Damm 已提交
290
{									\
291
	return early_platform_driver_register(&early_driver, buffer);	\
M
Magnus Damm 已提交
292 293 294
}									\
early_param(class_string, early_platform_driver_setup_func)
#else /* MODULE */
295 296 297 298 299
#define early_platform_init_buffer(class_string, platdrv, buf, bufsiz)	\
static inline char *early_platform_driver_setup_func(void)		\
{									\
	return bufsiz ? buf : NULL;					\
}
M
Magnus Damm 已提交
300 301
#endif /* MODULE */

302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328
#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, \
329
	.restore = platform_pm_restore,
330 331 332 333
#else
#define USE_PLATFORM_PM_SLEEP_OPS
#endif

334
#endif /* _PLATFORM_DEVICE_H_ */