Blame view

drivers/acpi/event.c 4.22 KB
b24413180   Greg Kroah-Hartman   License cleanup: ...
1
  // SPDX-License-Identifier: GPL-2.0
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2
3
4
5
6
7
8
9
10
  /*
   * 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....
11
  #include <linux/export.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
12
13
14
  #include <linux/proc_fs.h>
  #include <linux/init.h>
  #include <linux/poll.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
15
  #include <linux/gfp.h>
8b48463f8   Lv Zheng   ACPI: Clean up in...
16
  #include <linux/acpi.h>
864bdfb91   Zhang Rui   ACPI: Export even...
17
18
  #include <net/netlink.h>
  #include <net/genetlink.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
19

a192a9580   Len Brown   ACPI: Move defini...
20
  #include "internal.h"
9ee85241f   Zhang Rui   ACPI: create noti...
21
  /* ACPI notifier chain */
c8e773fa4   Adrian Bunk   ACPI: static acpi...
22
  static BLOCKING_NOTIFIER_HEAD(acpi_chain_head);
9ee85241f   Zhang Rui   ACPI: create noti...
23
24
25
26
27
28
29
30
31
32
  
  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)
c6237b210   Maximilian Luz   ACPI: Fix whitesp...
33
  			== NOTIFY_BAD) ? -EINVAL : 0;
9ee85241f   Zhang Rui   ACPI: create noti...
34
35
36
37
38
39
40
41
42
43
44
45
46
47
  }
  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...
48
  #ifdef CONFIG_NET
e13d87473   Adrian Bunk   ACPI: static
49
  static unsigned int acpi_event_seqnum;
864bdfb91   Zhang Rui   ACPI: Export even...
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
  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...
72
73
74
  #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...
75

2a94fe48f   Johannes Berg   genetlink: make m...
76
77
78
  static const struct genl_multicast_group acpi_event_mcgrps[] = {
  	{ .name = ACPI_GENL_MCAST_GROUP_NAME, },
  };
56989f6d8   Johannes Berg   genetlink: mark f...
79
  static struct genl_family acpi_event_genl_family __ro_after_init = {
489111e5c   Johannes Berg   genetlink: static...
80
  	.module = THIS_MODULE,
9c977a453   Zhang Rui   ACPI: export ACPI...
81
  	.name = ACPI_GENL_FAMILY_NAME,
864bdfb91   Zhang Rui   ACPI: Export even...
82
83
  	.version = ACPI_GENL_VERSION,
  	.maxattr = ACPI_GENL_ATTR_MAX,
2a94fe48f   Johannes Berg   genetlink: make m...
84
85
  	.mcgrps = acpi_event_mcgrps,
  	.n_mcgrps = ARRAY_SIZE(acpi_event_mcgrps),
864bdfb91   Zhang Rui   ACPI: Export even...
86
  };
962ce8ca0   Zhang Rui   ACPI: don't dupli...
87
88
  int acpi_bus_generate_netlink_event(const char *device_class,
  				      const char *bus_id,
864bdfb91   Zhang Rui   ACPI: Export even...
89
90
91
92
93
94
95
  				      u8 type, int data)
  {
  	struct sk_buff *skb;
  	struct nlattr *attr;
  	struct acpi_genl_event *event;
  	void *msg_header;
  	int size;
864bdfb91   Zhang Rui   ACPI: Export even...
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
  
  	/* 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...
124
  	memset(event, 0, sizeof(struct acpi_genl_event));
c7d5f21e8   Gustavo A. R. Silva   ACPI: event: repl...
125
126
  	strscpy(event->device_class, device_class, sizeof(event->device_class));
  	strscpy(event->bus_id, bus_id, sizeof(event->bus_id));
864bdfb91   Zhang Rui   ACPI: Export even...
127
128
129
130
  	event->type = type;
  	event->data = data;
  
  	/* send multicast genetlink message */
053c095a8   Johannes Berg   netlink: make nlm...
131
  	genlmsg_end(skb, msg_header);
864bdfb91   Zhang Rui   ACPI: Export even...
132

2a94fe48f   Johannes Berg   genetlink: make m...
133
  	genlmsg_multicast(&acpi_event_genl_family, skb, 0, 0, GFP_ATOMIC);
864bdfb91   Zhang Rui   ACPI: Export even...
134
135
  	return 0;
  }
864bdfb91   Zhang Rui   ACPI: Export even...
136

962ce8ca0   Zhang Rui   ACPI: don't dupli...
137
  EXPORT_SYMBOL(acpi_bus_generate_netlink_event);
56989f6d8   Johannes Berg   genetlink: mark f...
138
  static int __init acpi_event_genetlink_init(void)
864bdfb91   Zhang Rui   ACPI: Export even...
139
  {
2a94fe48f   Johannes Berg   genetlink: make m...
140
  	return genl_register_family(&acpi_event_genl_family);
864bdfb91   Zhang Rui   ACPI: Export even...
141
142
143
  }
  
  #else
3e069ee0c   Len Brown   ACPI: fix ia64 al...
144
145
146
  int acpi_bus_generate_netlink_event(const char *device_class,
  				      const char *bus_id,
  				      u8 type, int data)
864bdfb91   Zhang Rui   ACPI: Export even...
147
148
149
  {
  	return 0;
  }
864bdfb91   Zhang Rui   ACPI: Export even...
150

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

864bdfb91   Zhang Rui   ACPI: Export even...
153
154
155
156
157
  static int acpi_event_genetlink_init(void)
  {
  	return -ENODEV;
  }
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
158
159
  static int __init acpi_event_init(void)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
160
  	int error = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
161
  	if (acpi_disabled)
d550d98d3   Patrick Mochel   ACPI: delete trac...
162
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
163

864bdfb91   Zhang Rui   ACPI: Export even...
164
165
166
167
168
169
  	/* 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...
170
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
171
  }
864bdfb91   Zhang Rui   ACPI: Export even...
172
  fs_initcall(acpi_event_init);