Blame view

drivers/acpi/event.c 4.28 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
  /*
   * event.c - exporting ACPI events via procfs
   *
   *  Copyright (C) 2001, 2002 Andy Grover <andrew.grover@intel.com>
   *  Copyright (C) 2001, 2002 Paul Diefenbaugh <paul.s.diefenbaugh@intel.com>
   *
   */
  
  #include <linux/spinlock.h>
214f2c90b   Paul Gortmaker   acpi: add export....
10
  #include <linux/export.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
11
12
13
  #include <linux/proc_fs.h>
  #include <linux/init.h>
  #include <linux/poll.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
14
  #include <linux/gfp.h>
8b48463f8   Lv Zheng   ACPI: Clean up in...
15
  #include <linux/acpi.h>
864bdfb91   Zhang Rui   ACPI: Export even...
16
17
  #include <net/netlink.h>
  #include <net/genetlink.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
18

a192a9580   Len Brown   ACPI: Move defini...
19
  #include "internal.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
20
  #define _COMPONENT		ACPI_SYSTEM_COMPONENT
f52fd66d2   Len Brown   ACPI: clean up AC...
21
  ACPI_MODULE_NAME("event");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
22

9ee85241f   Zhang Rui   ACPI: create noti...
23
  /* ACPI notifier chain */
c8e773fa4   Adrian Bunk   ACPI: static acpi...
24
  static BLOCKING_NOTIFIER_HEAD(acpi_chain_head);
9ee85241f   Zhang Rui   ACPI: create noti...
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
  
  int acpi_notifier_call_chain(struct acpi_device *dev, u32 type, u32 data)
  {
  	struct acpi_bus_event event;
  
  	strcpy(event.device_class, dev->pnp.device_class);
  	strcpy(event.bus_id, dev->pnp.bus_id);
  	event.type = type;
  	event.data = data;
  	return (blocking_notifier_call_chain(&acpi_chain_head, 0, (void *)&event)
                          == NOTIFY_BAD) ? -EINVAL : 0;
  }
  EXPORT_SYMBOL(acpi_notifier_call_chain);
  
  int register_acpi_notifier(struct notifier_block *nb)
  {
  	return blocking_notifier_chain_register(&acpi_chain_head, nb);
  }
  EXPORT_SYMBOL(register_acpi_notifier);
  
  int unregister_acpi_notifier(struct notifier_block *nb)
  {
  	return blocking_notifier_chain_unregister(&acpi_chain_head, nb);
  }
  EXPORT_SYMBOL(unregister_acpi_notifier);
864bdfb91   Zhang Rui   ACPI: Export even...
50
  #ifdef CONFIG_NET
e13d87473   Adrian Bunk   ACPI: static
51
  static unsigned int acpi_event_seqnum;
864bdfb91   Zhang Rui   ACPI: Export even...
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
  struct acpi_genl_event {
  	acpi_device_class device_class;
  	char bus_id[15];
  	u32 type;
  	u32 data;
  };
  
  /* attributes of acpi_genl_family */
  enum {
  	ACPI_GENL_ATTR_UNSPEC,
  	ACPI_GENL_ATTR_EVENT,	/* ACPI event info needed by user space */
  	__ACPI_GENL_ATTR_MAX,
  };
  #define ACPI_GENL_ATTR_MAX (__ACPI_GENL_ATTR_MAX - 1)
  
  /* commands supported by the acpi_genl_family */
  enum {
  	ACPI_GENL_CMD_UNSPEC,
  	ACPI_GENL_CMD_EVENT,	/* kernel->user notifications for ACPI events */
  	__ACPI_GENL_CMD_MAX,
  };
  #define ACPI_GENL_CMD_MAX (__ACPI_GENL_CMD_MAX - 1)
9c977a453   Zhang Rui   ACPI: export ACPI...
74
75
76
  #define ACPI_GENL_FAMILY_NAME		"acpi_event"
  #define ACPI_GENL_VERSION		0x01
  #define ACPI_GENL_MCAST_GROUP_NAME 	"acpi_mc_group"
864bdfb91   Zhang Rui   ACPI: Export even...
77

2a94fe48f   Johannes Berg   genetlink: make m...
78
79
80
  static const struct genl_multicast_group acpi_event_mcgrps[] = {
  	{ .name = ACPI_GENL_MCAST_GROUP_NAME, },
  };
864bdfb91   Zhang Rui   ACPI: Export even...
81
82
  static struct genl_family acpi_event_genl_family = {
  	.id = GENL_ID_GENERATE,
9c977a453   Zhang Rui   ACPI: export ACPI...
83
  	.name = ACPI_GENL_FAMILY_NAME,
864bdfb91   Zhang Rui   ACPI: Export even...
84
85
  	.version = ACPI_GENL_VERSION,
  	.maxattr = ACPI_GENL_ATTR_MAX,
2a94fe48f   Johannes Berg   genetlink: make m...
86
87
  	.mcgrps = acpi_event_mcgrps,
  	.n_mcgrps = ARRAY_SIZE(acpi_event_mcgrps),
864bdfb91   Zhang Rui   ACPI: Export even...
88
  };
962ce8ca0   Zhang Rui   ACPI: don't dupli...
89
90
  int acpi_bus_generate_netlink_event(const char *device_class,
  				      const char *bus_id,
864bdfb91   Zhang Rui   ACPI: Export even...
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
  				      u8 type, int data)
  {
  	struct sk_buff *skb;
  	struct nlattr *attr;
  	struct acpi_genl_event *event;
  	void *msg_header;
  	int size;
  	int result;
  
  	/* allocate memory */
  	size = nla_total_size(sizeof(struct acpi_genl_event)) +
  	    nla_total_size(0);
  
  	skb = genlmsg_new(size, GFP_ATOMIC);
  	if (!skb)
  		return -ENOMEM;
  
  	/* add the genetlink message header */
  	msg_header = genlmsg_put(skb, 0, acpi_event_seqnum++,
  				 &acpi_event_genl_family, 0,
  				 ACPI_GENL_CMD_EVENT);
  	if (!msg_header) {
  		nlmsg_free(skb);
  		return -ENOMEM;
  	}
  
  	/* fill the data */
  	attr =
  	    nla_reserve(skb, ACPI_GENL_ATTR_EVENT,
  			sizeof(struct acpi_genl_event));
  	if (!attr) {
  		nlmsg_free(skb);
  		return -EINVAL;
  	}
  
  	event = nla_data(attr);
864bdfb91   Zhang Rui   ACPI: Export even...
127
  	memset(event, 0, sizeof(struct acpi_genl_event));
962ce8ca0   Zhang Rui   ACPI: don't dupli...
128
129
  	strcpy(event->device_class, device_class);
  	strcpy(event->bus_id, bus_id);
864bdfb91   Zhang Rui   ACPI: Export even...
130
131
132
133
134
135
136
137
138
  	event->type = type;
  	event->data = data;
  
  	/* send multicast genetlink message */
  	result = genlmsg_end(skb, msg_header);
  	if (result < 0) {
  		nlmsg_free(skb);
  		return result;
  	}
2a94fe48f   Johannes Berg   genetlink: make m...
139
  	genlmsg_multicast(&acpi_event_genl_family, skb, 0, 0, GFP_ATOMIC);
864bdfb91   Zhang Rui   ACPI: Export even...
140
141
  	return 0;
  }
864bdfb91   Zhang Rui   ACPI: Export even...
142

962ce8ca0   Zhang Rui   ACPI: don't dupli...
143
  EXPORT_SYMBOL(acpi_bus_generate_netlink_event);
864bdfb91   Zhang Rui   ACPI: Export even...
144
145
  static int acpi_event_genetlink_init(void)
  {
2a94fe48f   Johannes Berg   genetlink: make m...
146
  	return genl_register_family(&acpi_event_genl_family);
864bdfb91   Zhang Rui   ACPI: Export even...
147
148
149
  }
  
  #else
3e069ee0c   Len Brown   ACPI: fix ia64 al...
150
151
152
  int acpi_bus_generate_netlink_event(const char *device_class,
  				      const char *bus_id,
  				      u8 type, int data)
864bdfb91   Zhang Rui   ACPI: Export even...
153
154
155
  {
  	return 0;
  }
864bdfb91   Zhang Rui   ACPI: Export even...
156

66baf327a   Henrique de Moraes Holschuh   ACPI: fix CONFIG_...
157
  EXPORT_SYMBOL(acpi_bus_generate_netlink_event);
962ce8ca0   Zhang Rui   ACPI: don't dupli...
158

864bdfb91   Zhang Rui   ACPI: Export even...
159
160
161
162
163
  static int acpi_event_genetlink_init(void)
  {
  	return -ENODEV;
  }
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
164
165
  static int __init acpi_event_init(void)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
166
  	int error = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
167
  	if (acpi_disabled)
d550d98d3   Patrick Mochel   ACPI: delete trac...
168
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
169

864bdfb91   Zhang Rui   ACPI: Export even...
170
171
172
173
174
175
  	/* create genetlink for acpi event */
  	error = acpi_event_genetlink_init();
  	if (error)
  		printk(KERN_WARNING PREFIX
  		       "Failed to create genetlink family for ACPI event
  ");
864bdfb91   Zhang Rui   ACPI: Export even...
176
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
177
  }
864bdfb91   Zhang Rui   ACPI: Export even...
178
  fs_initcall(acpi_event_init);