xref: /openbmc/linux/drivers/misc/ibmasm/ibmasmfs.c (revision 1da177e4c3f41524e886b7f1b8a0c1fc7321cac2)
1*1da177e4SLinus Torvalds /*
2*1da177e4SLinus Torvalds  * IBM ASM Service Processor Device Driver
3*1da177e4SLinus Torvalds  *
4*1da177e4SLinus Torvalds  * This program is free software; you can redistribute it and/or modify
5*1da177e4SLinus Torvalds  * it under the terms of the GNU General Public License as published by
6*1da177e4SLinus Torvalds  * the Free Software Foundation; either version 2 of the License, or
7*1da177e4SLinus Torvalds  * (at your option) any later version.
8*1da177e4SLinus Torvalds  *
9*1da177e4SLinus Torvalds  * This program is distributed in the hope that it will be useful,
10*1da177e4SLinus Torvalds  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11*1da177e4SLinus Torvalds  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12*1da177e4SLinus Torvalds  * GNU General Public License for more details.
13*1da177e4SLinus Torvalds  *
14*1da177e4SLinus Torvalds  * You should have received a copy of the GNU General Public License
15*1da177e4SLinus Torvalds  * along with this program; if not, write to the Free Software
16*1da177e4SLinus Torvalds  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17*1da177e4SLinus Torvalds  *
18*1da177e4SLinus Torvalds  * Copyright (C) IBM Corporation, 2004
19*1da177e4SLinus Torvalds  *
20*1da177e4SLinus Torvalds  * Author: Max Asb�ck <amax@us.ibm.com>
21*1da177e4SLinus Torvalds  *
22*1da177e4SLinus Torvalds  */
23*1da177e4SLinus Torvalds 
24*1da177e4SLinus Torvalds /*
25*1da177e4SLinus Torvalds  * Parts of this code are based on an article by Jonathan Corbet
26*1da177e4SLinus Torvalds  * that appeared in Linux Weekly News.
27*1da177e4SLinus Torvalds  */
28*1da177e4SLinus Torvalds 
29*1da177e4SLinus Torvalds 
30*1da177e4SLinus Torvalds /*
31*1da177e4SLinus Torvalds  * The IBMASM file virtual filesystem. It creates the following hierarchy
32*1da177e4SLinus Torvalds  * dymamically when mounted from user space:
33*1da177e4SLinus Torvalds  *
34*1da177e4SLinus Torvalds  *    /ibmasm
35*1da177e4SLinus Torvalds  *    |-- 0
36*1da177e4SLinus Torvalds  *    |   |-- command
37*1da177e4SLinus Torvalds  *    |   |-- event
38*1da177e4SLinus Torvalds  *    |   |-- reverse_heartbeat
39*1da177e4SLinus Torvalds  *    |   `-- remote_video
40*1da177e4SLinus Torvalds  *    |       |-- connected
41*1da177e4SLinus Torvalds  *    |       |-- depth
42*1da177e4SLinus Torvalds  *    |       |-- events
43*1da177e4SLinus Torvalds  *    |       |-- height
44*1da177e4SLinus Torvalds  *    |       `-- width
45*1da177e4SLinus Torvalds  *    .
46*1da177e4SLinus Torvalds  *    .
47*1da177e4SLinus Torvalds  *    .
48*1da177e4SLinus Torvalds  *    `-- n
49*1da177e4SLinus Torvalds  *        |-- command
50*1da177e4SLinus Torvalds  *        |-- event
51*1da177e4SLinus Torvalds  *        |-- reverse_heartbeat
52*1da177e4SLinus Torvalds  *        `-- remote_video
53*1da177e4SLinus Torvalds  *            |-- connected
54*1da177e4SLinus Torvalds  *            |-- depth
55*1da177e4SLinus Torvalds  *            |-- events
56*1da177e4SLinus Torvalds  *            |-- height
57*1da177e4SLinus Torvalds  *            `-- width
58*1da177e4SLinus Torvalds  *
59*1da177e4SLinus Torvalds  * For each service processor the following files are created:
60*1da177e4SLinus Torvalds  *
61*1da177e4SLinus Torvalds  * command: execute dot commands
62*1da177e4SLinus Torvalds  * 	write: execute a dot command on the service processor
63*1da177e4SLinus Torvalds  * 	read: return the result of a previously executed dot command
64*1da177e4SLinus Torvalds  *
65*1da177e4SLinus Torvalds  * events: listen for service processor events
66*1da177e4SLinus Torvalds  * 	read: sleep (interruptible) until an event occurs
67*1da177e4SLinus Torvalds  *      write: wakeup sleeping event listener
68*1da177e4SLinus Torvalds  *
69*1da177e4SLinus Torvalds  * reverse_heartbeat: send a heartbeat to the service processor
70*1da177e4SLinus Torvalds  * 	read: sleep (interruptible) until the reverse heartbeat fails
71*1da177e4SLinus Torvalds  *      write: wakeup sleeping heartbeat listener
72*1da177e4SLinus Torvalds  *
73*1da177e4SLinus Torvalds  * remote_video/width
74*1da177e4SLinus Torvalds  * remote_video/height
75*1da177e4SLinus Torvalds  * remote_video/width: control remote display settings
76*1da177e4SLinus Torvalds  * 	write: set value
77*1da177e4SLinus Torvalds  * 	read: read value
78*1da177e4SLinus Torvalds  *
79*1da177e4SLinus Torvalds  * remote_video/connected
80*1da177e4SLinus Torvalds  * 	read: return "1" if web browser VNC java applet is connected,
81*1da177e4SLinus Torvalds  * 		"0" otherwise
82*1da177e4SLinus Torvalds  *
83*1da177e4SLinus Torvalds  * remote_video/events
84*1da177e4SLinus Torvalds  * 	read: sleep until a remote mouse or keyboard event occurs, then return
85*1da177e4SLinus Torvalds  * 		then event.
86*1da177e4SLinus Torvalds  */
87*1da177e4SLinus Torvalds 
88*1da177e4SLinus Torvalds #include <linux/fs.h>
89*1da177e4SLinus Torvalds #include <linux/pagemap.h>
90*1da177e4SLinus Torvalds #include <asm/uaccess.h>
91*1da177e4SLinus Torvalds #include <asm/io.h>
92*1da177e4SLinus Torvalds #include "ibmasm.h"
93*1da177e4SLinus Torvalds #include "remote.h"
94*1da177e4SLinus Torvalds #include "dot_command.h"
95*1da177e4SLinus Torvalds 
96*1da177e4SLinus Torvalds #define IBMASMFS_MAGIC 0x66726f67
97*1da177e4SLinus Torvalds 
98*1da177e4SLinus Torvalds static LIST_HEAD(service_processors);
99*1da177e4SLinus Torvalds 
100*1da177e4SLinus Torvalds static struct inode *ibmasmfs_make_inode(struct super_block *sb, int mode);
101*1da177e4SLinus Torvalds static void ibmasmfs_create_files (struct super_block *sb, struct dentry *root);
102*1da177e4SLinus Torvalds static int ibmasmfs_fill_super (struct super_block *sb, void *data, int silent);
103*1da177e4SLinus Torvalds 
104*1da177e4SLinus Torvalds 
105*1da177e4SLinus Torvalds static struct super_block *ibmasmfs_get_super(struct file_system_type *fst,
106*1da177e4SLinus Torvalds 			int flags, const char *name, void *data)
107*1da177e4SLinus Torvalds {
108*1da177e4SLinus Torvalds 	return get_sb_single(fst, flags, data, ibmasmfs_fill_super);
109*1da177e4SLinus Torvalds }
110*1da177e4SLinus Torvalds 
111*1da177e4SLinus Torvalds static struct super_operations ibmasmfs_s_ops = {
112*1da177e4SLinus Torvalds 	.statfs		= simple_statfs,
113*1da177e4SLinus Torvalds 	.drop_inode	= generic_delete_inode,
114*1da177e4SLinus Torvalds };
115*1da177e4SLinus Torvalds 
116*1da177e4SLinus Torvalds static struct file_operations *ibmasmfs_dir_ops = &simple_dir_operations;
117*1da177e4SLinus Torvalds 
118*1da177e4SLinus Torvalds static struct file_system_type ibmasmfs_type = {
119*1da177e4SLinus Torvalds 	.owner          = THIS_MODULE,
120*1da177e4SLinus Torvalds 	.name           = "ibmasmfs",
121*1da177e4SLinus Torvalds 	.get_sb         = ibmasmfs_get_super,
122*1da177e4SLinus Torvalds 	.kill_sb        = kill_litter_super,
123*1da177e4SLinus Torvalds };
124*1da177e4SLinus Torvalds 
125*1da177e4SLinus Torvalds static int ibmasmfs_fill_super (struct super_block *sb, void *data, int silent)
126*1da177e4SLinus Torvalds {
127*1da177e4SLinus Torvalds 	struct inode *root;
128*1da177e4SLinus Torvalds 	struct dentry *root_dentry;
129*1da177e4SLinus Torvalds 
130*1da177e4SLinus Torvalds 	sb->s_blocksize = PAGE_CACHE_SIZE;
131*1da177e4SLinus Torvalds 	sb->s_blocksize_bits = PAGE_CACHE_SHIFT;
132*1da177e4SLinus Torvalds 	sb->s_magic = IBMASMFS_MAGIC;
133*1da177e4SLinus Torvalds 	sb->s_op = &ibmasmfs_s_ops;
134*1da177e4SLinus Torvalds 	sb->s_time_gran = 1;
135*1da177e4SLinus Torvalds 
136*1da177e4SLinus Torvalds 	root = ibmasmfs_make_inode (sb, S_IFDIR | 0500);
137*1da177e4SLinus Torvalds 	if (!root)
138*1da177e4SLinus Torvalds 		return -ENOMEM;
139*1da177e4SLinus Torvalds 
140*1da177e4SLinus Torvalds 	root->i_op = &simple_dir_inode_operations;
141*1da177e4SLinus Torvalds 	root->i_fop = ibmasmfs_dir_ops;
142*1da177e4SLinus Torvalds 
143*1da177e4SLinus Torvalds 	root_dentry = d_alloc_root(root);
144*1da177e4SLinus Torvalds 	if (!root_dentry) {
145*1da177e4SLinus Torvalds 		iput(root);
146*1da177e4SLinus Torvalds 		return -ENOMEM;
147*1da177e4SLinus Torvalds 	}
148*1da177e4SLinus Torvalds 	sb->s_root = root_dentry;
149*1da177e4SLinus Torvalds 
150*1da177e4SLinus Torvalds 	ibmasmfs_create_files(sb, root_dentry);
151*1da177e4SLinus Torvalds 	return 0;
152*1da177e4SLinus Torvalds }
153*1da177e4SLinus Torvalds 
154*1da177e4SLinus Torvalds static struct inode *ibmasmfs_make_inode(struct super_block *sb, int mode)
155*1da177e4SLinus Torvalds {
156*1da177e4SLinus Torvalds 	struct inode *ret = new_inode(sb);
157*1da177e4SLinus Torvalds 
158*1da177e4SLinus Torvalds 	if (ret) {
159*1da177e4SLinus Torvalds 		ret->i_mode = mode;
160*1da177e4SLinus Torvalds 		ret->i_uid = ret->i_gid = 0;
161*1da177e4SLinus Torvalds 		ret->i_blksize = PAGE_CACHE_SIZE;
162*1da177e4SLinus Torvalds 		ret->i_blocks = 0;
163*1da177e4SLinus Torvalds 		ret->i_atime = ret->i_mtime = ret->i_ctime = CURRENT_TIME;
164*1da177e4SLinus Torvalds 	}
165*1da177e4SLinus Torvalds 	return ret;
166*1da177e4SLinus Torvalds }
167*1da177e4SLinus Torvalds 
168*1da177e4SLinus Torvalds static struct dentry *ibmasmfs_create_file (struct super_block *sb,
169*1da177e4SLinus Torvalds 			struct dentry *parent,
170*1da177e4SLinus Torvalds 		       	const char *name,
171*1da177e4SLinus Torvalds 			struct file_operations *fops,
172*1da177e4SLinus Torvalds 			void *data,
173*1da177e4SLinus Torvalds 			int mode)
174*1da177e4SLinus Torvalds {
175*1da177e4SLinus Torvalds 	struct dentry *dentry;
176*1da177e4SLinus Torvalds 	struct inode *inode;
177*1da177e4SLinus Torvalds 
178*1da177e4SLinus Torvalds 	dentry = d_alloc_name(parent, name);
179*1da177e4SLinus Torvalds 	if (!dentry)
180*1da177e4SLinus Torvalds 		return NULL;
181*1da177e4SLinus Torvalds 
182*1da177e4SLinus Torvalds 	inode = ibmasmfs_make_inode(sb, S_IFREG | mode);
183*1da177e4SLinus Torvalds 	if (!inode) {
184*1da177e4SLinus Torvalds 		dput(dentry);
185*1da177e4SLinus Torvalds 		return NULL;
186*1da177e4SLinus Torvalds 	}
187*1da177e4SLinus Torvalds 
188*1da177e4SLinus Torvalds 	inode->i_fop = fops;
189*1da177e4SLinus Torvalds 	inode->u.generic_ip = data;
190*1da177e4SLinus Torvalds 
191*1da177e4SLinus Torvalds 	d_add(dentry, inode);
192*1da177e4SLinus Torvalds 	return dentry;
193*1da177e4SLinus Torvalds }
194*1da177e4SLinus Torvalds 
195*1da177e4SLinus Torvalds static struct dentry *ibmasmfs_create_dir (struct super_block *sb,
196*1da177e4SLinus Torvalds 				struct dentry *parent,
197*1da177e4SLinus Torvalds 				const char *name)
198*1da177e4SLinus Torvalds {
199*1da177e4SLinus Torvalds 	struct dentry *dentry;
200*1da177e4SLinus Torvalds 	struct inode *inode;
201*1da177e4SLinus Torvalds 
202*1da177e4SLinus Torvalds 	dentry = d_alloc_name(parent, name);
203*1da177e4SLinus Torvalds 	if (!dentry)
204*1da177e4SLinus Torvalds 		return NULL;
205*1da177e4SLinus Torvalds 
206*1da177e4SLinus Torvalds 	inode = ibmasmfs_make_inode(sb, S_IFDIR | 0500);
207*1da177e4SLinus Torvalds 	if (!inode) {
208*1da177e4SLinus Torvalds 		dput(dentry);
209*1da177e4SLinus Torvalds 		return NULL;
210*1da177e4SLinus Torvalds 	}
211*1da177e4SLinus Torvalds 
212*1da177e4SLinus Torvalds 	inode->i_op = &simple_dir_inode_operations;
213*1da177e4SLinus Torvalds 	inode->i_fop = ibmasmfs_dir_ops;
214*1da177e4SLinus Torvalds 
215*1da177e4SLinus Torvalds 	d_add(dentry, inode);
216*1da177e4SLinus Torvalds 	return dentry;
217*1da177e4SLinus Torvalds }
218*1da177e4SLinus Torvalds 
219*1da177e4SLinus Torvalds int ibmasmfs_register(void)
220*1da177e4SLinus Torvalds {
221*1da177e4SLinus Torvalds 	return register_filesystem(&ibmasmfs_type);
222*1da177e4SLinus Torvalds }
223*1da177e4SLinus Torvalds 
224*1da177e4SLinus Torvalds void ibmasmfs_unregister(void)
225*1da177e4SLinus Torvalds {
226*1da177e4SLinus Torvalds 	unregister_filesystem(&ibmasmfs_type);
227*1da177e4SLinus Torvalds }
228*1da177e4SLinus Torvalds 
229*1da177e4SLinus Torvalds void ibmasmfs_add_sp(struct service_processor *sp)
230*1da177e4SLinus Torvalds {
231*1da177e4SLinus Torvalds 	list_add(&sp->node, &service_processors);
232*1da177e4SLinus Torvalds }
233*1da177e4SLinus Torvalds 
234*1da177e4SLinus Torvalds /* struct to save state between command file operations */
235*1da177e4SLinus Torvalds struct ibmasmfs_command_data {
236*1da177e4SLinus Torvalds 	struct service_processor	*sp;
237*1da177e4SLinus Torvalds 	struct command			*command;
238*1da177e4SLinus Torvalds };
239*1da177e4SLinus Torvalds 
240*1da177e4SLinus Torvalds /* struct to save state between event file operations */
241*1da177e4SLinus Torvalds struct ibmasmfs_event_data {
242*1da177e4SLinus Torvalds 	struct service_processor	*sp;
243*1da177e4SLinus Torvalds 	struct event_reader		reader;
244*1da177e4SLinus Torvalds 	int				active;
245*1da177e4SLinus Torvalds };
246*1da177e4SLinus Torvalds 
247*1da177e4SLinus Torvalds /* struct to save state between reverse heartbeat file operations */
248*1da177e4SLinus Torvalds struct ibmasmfs_heartbeat_data {
249*1da177e4SLinus Torvalds 	struct service_processor	*sp;
250*1da177e4SLinus Torvalds 	struct reverse_heartbeat	heartbeat;
251*1da177e4SLinus Torvalds 	int				active;
252*1da177e4SLinus Torvalds };
253*1da177e4SLinus Torvalds 
254*1da177e4SLinus Torvalds static int command_file_open(struct inode *inode, struct file *file)
255*1da177e4SLinus Torvalds {
256*1da177e4SLinus Torvalds 	struct ibmasmfs_command_data *command_data;
257*1da177e4SLinus Torvalds 
258*1da177e4SLinus Torvalds 	if (!inode->u.generic_ip)
259*1da177e4SLinus Torvalds 		return -ENODEV;
260*1da177e4SLinus Torvalds 
261*1da177e4SLinus Torvalds 	command_data = kmalloc(sizeof(struct ibmasmfs_command_data), GFP_KERNEL);
262*1da177e4SLinus Torvalds 	if (!command_data)
263*1da177e4SLinus Torvalds 		return -ENOMEM;
264*1da177e4SLinus Torvalds 
265*1da177e4SLinus Torvalds 	command_data->command = NULL;
266*1da177e4SLinus Torvalds 	command_data->sp = inode->u.generic_ip;
267*1da177e4SLinus Torvalds 	file->private_data = command_data;
268*1da177e4SLinus Torvalds 	return 0;
269*1da177e4SLinus Torvalds }
270*1da177e4SLinus Torvalds 
271*1da177e4SLinus Torvalds static int command_file_close(struct inode *inode, struct file *file)
272*1da177e4SLinus Torvalds {
273*1da177e4SLinus Torvalds 	struct ibmasmfs_command_data *command_data = file->private_data;
274*1da177e4SLinus Torvalds 
275*1da177e4SLinus Torvalds 	if (command_data->command)
276*1da177e4SLinus Torvalds 		command_put(command_data->command);
277*1da177e4SLinus Torvalds 
278*1da177e4SLinus Torvalds 	kfree(command_data);
279*1da177e4SLinus Torvalds 	return 0;
280*1da177e4SLinus Torvalds }
281*1da177e4SLinus Torvalds 
282*1da177e4SLinus Torvalds static ssize_t command_file_read(struct file *file, char __user *buf, size_t count, loff_t *offset)
283*1da177e4SLinus Torvalds {
284*1da177e4SLinus Torvalds 	struct ibmasmfs_command_data *command_data = file->private_data;
285*1da177e4SLinus Torvalds 	struct command *cmd;
286*1da177e4SLinus Torvalds 	int len;
287*1da177e4SLinus Torvalds 	unsigned long flags;
288*1da177e4SLinus Torvalds 
289*1da177e4SLinus Torvalds 	if (*offset < 0)
290*1da177e4SLinus Torvalds 		return -EINVAL;
291*1da177e4SLinus Torvalds 	if (count == 0 || count > IBMASM_CMD_MAX_BUFFER_SIZE)
292*1da177e4SLinus Torvalds 		return 0;
293*1da177e4SLinus Torvalds 	if (*offset != 0)
294*1da177e4SLinus Torvalds 		return 0;
295*1da177e4SLinus Torvalds 
296*1da177e4SLinus Torvalds 	spin_lock_irqsave(&command_data->sp->lock, flags);
297*1da177e4SLinus Torvalds 	cmd = command_data->command;
298*1da177e4SLinus Torvalds 	if (cmd == NULL) {
299*1da177e4SLinus Torvalds 		spin_unlock_irqrestore(&command_data->sp->lock, flags);
300*1da177e4SLinus Torvalds 		return 0;
301*1da177e4SLinus Torvalds 	}
302*1da177e4SLinus Torvalds 	command_data->command = NULL;
303*1da177e4SLinus Torvalds 	spin_unlock_irqrestore(&command_data->sp->lock, flags);
304*1da177e4SLinus Torvalds 
305*1da177e4SLinus Torvalds 	if (cmd->status != IBMASM_CMD_COMPLETE) {
306*1da177e4SLinus Torvalds 		command_put(cmd);
307*1da177e4SLinus Torvalds 		return -EIO;
308*1da177e4SLinus Torvalds 	}
309*1da177e4SLinus Torvalds 	len = min(count, cmd->buffer_size);
310*1da177e4SLinus Torvalds 	if (copy_to_user(buf, cmd->buffer, len)) {
311*1da177e4SLinus Torvalds 		command_put(cmd);
312*1da177e4SLinus Torvalds 		return -EFAULT;
313*1da177e4SLinus Torvalds 	}
314*1da177e4SLinus Torvalds 	command_put(cmd);
315*1da177e4SLinus Torvalds 
316*1da177e4SLinus Torvalds 	return len;
317*1da177e4SLinus Torvalds }
318*1da177e4SLinus Torvalds 
319*1da177e4SLinus Torvalds static ssize_t command_file_write(struct file *file, const char __user *ubuff, size_t count, loff_t *offset)
320*1da177e4SLinus Torvalds {
321*1da177e4SLinus Torvalds 	struct ibmasmfs_command_data *command_data = file->private_data;
322*1da177e4SLinus Torvalds 	struct command *cmd;
323*1da177e4SLinus Torvalds 	unsigned long flags;
324*1da177e4SLinus Torvalds 
325*1da177e4SLinus Torvalds 	if (*offset < 0)
326*1da177e4SLinus Torvalds 		return -EINVAL;
327*1da177e4SLinus Torvalds 	if (count == 0 || count > IBMASM_CMD_MAX_BUFFER_SIZE)
328*1da177e4SLinus Torvalds 		return 0;
329*1da177e4SLinus Torvalds 	if (*offset != 0)
330*1da177e4SLinus Torvalds 		return 0;
331*1da177e4SLinus Torvalds 
332*1da177e4SLinus Torvalds 	/* commands are executed sequentially, only one command at a time */
333*1da177e4SLinus Torvalds 	if (command_data->command)
334*1da177e4SLinus Torvalds 		return -EAGAIN;
335*1da177e4SLinus Torvalds 
336*1da177e4SLinus Torvalds 	cmd = ibmasm_new_command(count);
337*1da177e4SLinus Torvalds 	if (!cmd)
338*1da177e4SLinus Torvalds 		return -ENOMEM;
339*1da177e4SLinus Torvalds 
340*1da177e4SLinus Torvalds 	if (copy_from_user(cmd->buffer, ubuff, count)) {
341*1da177e4SLinus Torvalds 		command_put(cmd);
342*1da177e4SLinus Torvalds 		return -EFAULT;
343*1da177e4SLinus Torvalds 	}
344*1da177e4SLinus Torvalds 
345*1da177e4SLinus Torvalds 	spin_lock_irqsave(&command_data->sp->lock, flags);
346*1da177e4SLinus Torvalds 	if (command_data->command) {
347*1da177e4SLinus Torvalds 		spin_unlock_irqrestore(&command_data->sp->lock, flags);
348*1da177e4SLinus Torvalds 		command_put(cmd);
349*1da177e4SLinus Torvalds 		return -EAGAIN;
350*1da177e4SLinus Torvalds 	}
351*1da177e4SLinus Torvalds 	command_data->command = cmd;
352*1da177e4SLinus Torvalds 	spin_unlock_irqrestore(&command_data->sp->lock, flags);
353*1da177e4SLinus Torvalds 
354*1da177e4SLinus Torvalds 	ibmasm_exec_command(command_data->sp, cmd);
355*1da177e4SLinus Torvalds 	ibmasm_wait_for_response(cmd, get_dot_command_timeout(cmd->buffer));
356*1da177e4SLinus Torvalds 
357*1da177e4SLinus Torvalds 	return count;
358*1da177e4SLinus Torvalds }
359*1da177e4SLinus Torvalds 
360*1da177e4SLinus Torvalds static int event_file_open(struct inode *inode, struct file *file)
361*1da177e4SLinus Torvalds {
362*1da177e4SLinus Torvalds 	struct ibmasmfs_event_data *event_data;
363*1da177e4SLinus Torvalds 	struct service_processor *sp;
364*1da177e4SLinus Torvalds 
365*1da177e4SLinus Torvalds 	if (!inode->u.generic_ip)
366*1da177e4SLinus Torvalds 		return -ENODEV;
367*1da177e4SLinus Torvalds 
368*1da177e4SLinus Torvalds 	sp = inode->u.generic_ip;
369*1da177e4SLinus Torvalds 
370*1da177e4SLinus Torvalds 	event_data = kmalloc(sizeof(struct ibmasmfs_event_data), GFP_KERNEL);
371*1da177e4SLinus Torvalds 	if (!event_data)
372*1da177e4SLinus Torvalds 		return -ENOMEM;
373*1da177e4SLinus Torvalds 
374*1da177e4SLinus Torvalds 	ibmasm_event_reader_register(sp, &event_data->reader);
375*1da177e4SLinus Torvalds 
376*1da177e4SLinus Torvalds 	event_data->sp = sp;
377*1da177e4SLinus Torvalds 	file->private_data = event_data;
378*1da177e4SLinus Torvalds 	return 0;
379*1da177e4SLinus Torvalds }
380*1da177e4SLinus Torvalds 
381*1da177e4SLinus Torvalds static int event_file_close(struct inode *inode, struct file *file)
382*1da177e4SLinus Torvalds {
383*1da177e4SLinus Torvalds 	struct ibmasmfs_event_data *event_data = file->private_data;
384*1da177e4SLinus Torvalds 
385*1da177e4SLinus Torvalds 	ibmasm_event_reader_unregister(event_data->sp, &event_data->reader);
386*1da177e4SLinus Torvalds 	kfree(event_data);
387*1da177e4SLinus Torvalds 	return 0;
388*1da177e4SLinus Torvalds }
389*1da177e4SLinus Torvalds 
390*1da177e4SLinus Torvalds static ssize_t event_file_read(struct file *file, char __user *buf, size_t count, loff_t *offset)
391*1da177e4SLinus Torvalds {
392*1da177e4SLinus Torvalds 	struct ibmasmfs_event_data *event_data = file->private_data;
393*1da177e4SLinus Torvalds 	struct event_reader *reader = &event_data->reader;
394*1da177e4SLinus Torvalds 	int ret;
395*1da177e4SLinus Torvalds 
396*1da177e4SLinus Torvalds 	if (*offset < 0)
397*1da177e4SLinus Torvalds 		return -EINVAL;
398*1da177e4SLinus Torvalds 	if (count == 0 || count > IBMASM_EVENT_MAX_SIZE)
399*1da177e4SLinus Torvalds 		return 0;
400*1da177e4SLinus Torvalds 	if (*offset != 0)
401*1da177e4SLinus Torvalds 		return 0;
402*1da177e4SLinus Torvalds 
403*1da177e4SLinus Torvalds 	ret = ibmasm_get_next_event(event_data->sp, reader);
404*1da177e4SLinus Torvalds 	if (ret <= 0)
405*1da177e4SLinus Torvalds 		return ret;
406*1da177e4SLinus Torvalds 
407*1da177e4SLinus Torvalds 	if (count < reader->data_size)
408*1da177e4SLinus Torvalds 		return -EINVAL;
409*1da177e4SLinus Torvalds 
410*1da177e4SLinus Torvalds         if (copy_to_user(buf, reader->data, reader->data_size))
411*1da177e4SLinus Torvalds 		return -EFAULT;
412*1da177e4SLinus Torvalds 
413*1da177e4SLinus Torvalds 	return reader->data_size;
414*1da177e4SLinus Torvalds }
415*1da177e4SLinus Torvalds 
416*1da177e4SLinus Torvalds static ssize_t event_file_write(struct file *file, const char __user *buf, size_t count, loff_t *offset)
417*1da177e4SLinus Torvalds {
418*1da177e4SLinus Torvalds 	struct ibmasmfs_event_data *event_data = file->private_data;
419*1da177e4SLinus Torvalds 
420*1da177e4SLinus Torvalds 	if (*offset < 0)
421*1da177e4SLinus Torvalds 		return -EINVAL;
422*1da177e4SLinus Torvalds 	if (count != 1)
423*1da177e4SLinus Torvalds 		return 0;
424*1da177e4SLinus Torvalds 	if (*offset != 0)
425*1da177e4SLinus Torvalds 		return 0;
426*1da177e4SLinus Torvalds 
427*1da177e4SLinus Torvalds 	wake_up_interruptible(&event_data->reader.wait);
428*1da177e4SLinus Torvalds 	return 0;
429*1da177e4SLinus Torvalds }
430*1da177e4SLinus Torvalds 
431*1da177e4SLinus Torvalds static int r_heartbeat_file_open(struct inode *inode, struct file *file)
432*1da177e4SLinus Torvalds {
433*1da177e4SLinus Torvalds 	struct ibmasmfs_heartbeat_data *rhbeat;
434*1da177e4SLinus Torvalds 
435*1da177e4SLinus Torvalds 	if (!inode->u.generic_ip)
436*1da177e4SLinus Torvalds 		return -ENODEV;
437*1da177e4SLinus Torvalds 
438*1da177e4SLinus Torvalds 	rhbeat = kmalloc(sizeof(struct ibmasmfs_heartbeat_data), GFP_KERNEL);
439*1da177e4SLinus Torvalds 	if (!rhbeat)
440*1da177e4SLinus Torvalds 		return -ENOMEM;
441*1da177e4SLinus Torvalds 
442*1da177e4SLinus Torvalds 	rhbeat->sp = (struct service_processor *)inode->u.generic_ip;
443*1da177e4SLinus Torvalds 	rhbeat->active = 0;
444*1da177e4SLinus Torvalds 	ibmasm_init_reverse_heartbeat(rhbeat->sp, &rhbeat->heartbeat);
445*1da177e4SLinus Torvalds 	file->private_data = rhbeat;
446*1da177e4SLinus Torvalds 	return 0;
447*1da177e4SLinus Torvalds }
448*1da177e4SLinus Torvalds 
449*1da177e4SLinus Torvalds static int r_heartbeat_file_close(struct inode *inode, struct file *file)
450*1da177e4SLinus Torvalds {
451*1da177e4SLinus Torvalds 	struct ibmasmfs_heartbeat_data *rhbeat = file->private_data;
452*1da177e4SLinus Torvalds 
453*1da177e4SLinus Torvalds 	kfree(rhbeat);
454*1da177e4SLinus Torvalds 	return 0;
455*1da177e4SLinus Torvalds }
456*1da177e4SLinus Torvalds 
457*1da177e4SLinus Torvalds static ssize_t r_heartbeat_file_read(struct file *file, char __user *buf, size_t count, loff_t *offset)
458*1da177e4SLinus Torvalds {
459*1da177e4SLinus Torvalds 	struct ibmasmfs_heartbeat_data *rhbeat = file->private_data;
460*1da177e4SLinus Torvalds 	unsigned long flags;
461*1da177e4SLinus Torvalds 	int result;
462*1da177e4SLinus Torvalds 
463*1da177e4SLinus Torvalds 	if (*offset < 0)
464*1da177e4SLinus Torvalds 		return -EINVAL;
465*1da177e4SLinus Torvalds 	if (count == 0 || count > 1024)
466*1da177e4SLinus Torvalds 		return 0;
467*1da177e4SLinus Torvalds 	if (*offset != 0)
468*1da177e4SLinus Torvalds 		return 0;
469*1da177e4SLinus Torvalds 
470*1da177e4SLinus Torvalds 	/* allow only one reverse heartbeat per process */
471*1da177e4SLinus Torvalds 	spin_lock_irqsave(&rhbeat->sp->lock, flags);
472*1da177e4SLinus Torvalds 	if (rhbeat->active) {
473*1da177e4SLinus Torvalds 		spin_unlock_irqrestore(&rhbeat->sp->lock, flags);
474*1da177e4SLinus Torvalds 		return -EBUSY;
475*1da177e4SLinus Torvalds 	}
476*1da177e4SLinus Torvalds 	rhbeat->active = 1;
477*1da177e4SLinus Torvalds 	spin_unlock_irqrestore(&rhbeat->sp->lock, flags);
478*1da177e4SLinus Torvalds 
479*1da177e4SLinus Torvalds 	result = ibmasm_start_reverse_heartbeat(rhbeat->sp, &rhbeat->heartbeat);
480*1da177e4SLinus Torvalds 	rhbeat->active = 0;
481*1da177e4SLinus Torvalds 
482*1da177e4SLinus Torvalds 	return result;
483*1da177e4SLinus Torvalds }
484*1da177e4SLinus Torvalds 
485*1da177e4SLinus Torvalds static ssize_t r_heartbeat_file_write(struct file *file, const char __user *buf, size_t count, loff_t *offset)
486*1da177e4SLinus Torvalds {
487*1da177e4SLinus Torvalds 	struct ibmasmfs_heartbeat_data *rhbeat = file->private_data;
488*1da177e4SLinus Torvalds 
489*1da177e4SLinus Torvalds 	if (*offset < 0)
490*1da177e4SLinus Torvalds 		return -EINVAL;
491*1da177e4SLinus Torvalds 	if (count != 1)
492*1da177e4SLinus Torvalds 		return 0;
493*1da177e4SLinus Torvalds 	if (*offset != 0)
494*1da177e4SLinus Torvalds 		return 0;
495*1da177e4SLinus Torvalds 
496*1da177e4SLinus Torvalds 	if (rhbeat->active)
497*1da177e4SLinus Torvalds 		ibmasm_stop_reverse_heartbeat(&rhbeat->heartbeat);
498*1da177e4SLinus Torvalds 
499*1da177e4SLinus Torvalds 	return 1;
500*1da177e4SLinus Torvalds }
501*1da177e4SLinus Torvalds 
502*1da177e4SLinus Torvalds static int remote_settings_file_open(struct inode *inode, struct file *file)
503*1da177e4SLinus Torvalds {
504*1da177e4SLinus Torvalds 	file->private_data = inode->u.generic_ip;
505*1da177e4SLinus Torvalds 	return 0;
506*1da177e4SLinus Torvalds }
507*1da177e4SLinus Torvalds 
508*1da177e4SLinus Torvalds static int remote_settings_file_close(struct inode *inode, struct file *file)
509*1da177e4SLinus Torvalds {
510*1da177e4SLinus Torvalds 	return 0;
511*1da177e4SLinus Torvalds }
512*1da177e4SLinus Torvalds 
513*1da177e4SLinus Torvalds static ssize_t remote_settings_file_read(struct file *file, char __user *buf, size_t count, loff_t *offset)
514*1da177e4SLinus Torvalds {
515*1da177e4SLinus Torvalds 	void __iomem *address = (void __iomem *)file->private_data;
516*1da177e4SLinus Torvalds 	unsigned char *page;
517*1da177e4SLinus Torvalds 	int retval;
518*1da177e4SLinus Torvalds 	int len = 0;
519*1da177e4SLinus Torvalds 	unsigned int value;
520*1da177e4SLinus Torvalds 
521*1da177e4SLinus Torvalds 	if (*offset < 0)
522*1da177e4SLinus Torvalds 		return -EINVAL;
523*1da177e4SLinus Torvalds 	if (count == 0 || count > 1024)
524*1da177e4SLinus Torvalds 		return 0;
525*1da177e4SLinus Torvalds 	if (*offset != 0)
526*1da177e4SLinus Torvalds 		return 0;
527*1da177e4SLinus Torvalds 
528*1da177e4SLinus Torvalds 	page = (unsigned char *)__get_free_page(GFP_KERNEL);
529*1da177e4SLinus Torvalds 	if (!page)
530*1da177e4SLinus Torvalds 		return -ENOMEM;
531*1da177e4SLinus Torvalds 
532*1da177e4SLinus Torvalds 	value = readl(address);
533*1da177e4SLinus Torvalds 	len = sprintf(page, "%d\n", value);
534*1da177e4SLinus Torvalds 
535*1da177e4SLinus Torvalds 	if (copy_to_user(buf, page, len)) {
536*1da177e4SLinus Torvalds 		retval = -EFAULT;
537*1da177e4SLinus Torvalds 		goto exit;
538*1da177e4SLinus Torvalds 	}
539*1da177e4SLinus Torvalds 	*offset += len;
540*1da177e4SLinus Torvalds 	retval = len;
541*1da177e4SLinus Torvalds 
542*1da177e4SLinus Torvalds exit:
543*1da177e4SLinus Torvalds 	free_page((unsigned long)page);
544*1da177e4SLinus Torvalds 	return retval;
545*1da177e4SLinus Torvalds }
546*1da177e4SLinus Torvalds 
547*1da177e4SLinus Torvalds static ssize_t remote_settings_file_write(struct file *file, const char __user *ubuff, size_t count, loff_t *offset)
548*1da177e4SLinus Torvalds {
549*1da177e4SLinus Torvalds 	void __iomem *address = (void __iomem *)file->private_data;
550*1da177e4SLinus Torvalds 	char *buff;
551*1da177e4SLinus Torvalds 	unsigned int value;
552*1da177e4SLinus Torvalds 
553*1da177e4SLinus Torvalds 	if (*offset < 0)
554*1da177e4SLinus Torvalds 		return -EINVAL;
555*1da177e4SLinus Torvalds 	if (count == 0 || count > 1024)
556*1da177e4SLinus Torvalds 		return 0;
557*1da177e4SLinus Torvalds 	if (*offset != 0)
558*1da177e4SLinus Torvalds 		return 0;
559*1da177e4SLinus Torvalds 
560*1da177e4SLinus Torvalds 	buff = kmalloc (count + 1, GFP_KERNEL);
561*1da177e4SLinus Torvalds 	if (!buff)
562*1da177e4SLinus Torvalds 		return -ENOMEM;
563*1da177e4SLinus Torvalds 
564*1da177e4SLinus Torvalds 	memset(buff, 0x0, count + 1);
565*1da177e4SLinus Torvalds 
566*1da177e4SLinus Torvalds 	if (copy_from_user(buff, ubuff, count)) {
567*1da177e4SLinus Torvalds 		kfree(buff);
568*1da177e4SLinus Torvalds 		return -EFAULT;
569*1da177e4SLinus Torvalds 	}
570*1da177e4SLinus Torvalds 
571*1da177e4SLinus Torvalds 	value = simple_strtoul(buff, NULL, 10);
572*1da177e4SLinus Torvalds 	writel(value, address);
573*1da177e4SLinus Torvalds 	kfree(buff);
574*1da177e4SLinus Torvalds 
575*1da177e4SLinus Torvalds 	return count;
576*1da177e4SLinus Torvalds }
577*1da177e4SLinus Torvalds 
578*1da177e4SLinus Torvalds static int remote_event_file_open(struct inode *inode, struct file *file)
579*1da177e4SLinus Torvalds {
580*1da177e4SLinus Torvalds 	struct service_processor *sp;
581*1da177e4SLinus Torvalds 	unsigned long flags;
582*1da177e4SLinus Torvalds 	struct remote_queue *q;
583*1da177e4SLinus Torvalds 
584*1da177e4SLinus Torvalds 	file->private_data = inode->u.generic_ip;
585*1da177e4SLinus Torvalds 	sp = file->private_data;
586*1da177e4SLinus Torvalds 	q = &sp->remote_queue;
587*1da177e4SLinus Torvalds 
588*1da177e4SLinus Torvalds 	/* allow only one event reader */
589*1da177e4SLinus Torvalds 	spin_lock_irqsave(&sp->lock, flags);
590*1da177e4SLinus Torvalds 	if (q->open) {
591*1da177e4SLinus Torvalds 		spin_unlock_irqrestore(&sp->lock, flags);
592*1da177e4SLinus Torvalds 		return -EBUSY;
593*1da177e4SLinus Torvalds 	}
594*1da177e4SLinus Torvalds 	q->open = 1;
595*1da177e4SLinus Torvalds 	spin_unlock_irqrestore(&sp->lock, flags);
596*1da177e4SLinus Torvalds 
597*1da177e4SLinus Torvalds 	enable_mouse_interrupts(sp);
598*1da177e4SLinus Torvalds 
599*1da177e4SLinus Torvalds 	return 0;
600*1da177e4SLinus Torvalds }
601*1da177e4SLinus Torvalds 
602*1da177e4SLinus Torvalds static int remote_event_file_close(struct inode *inode, struct file *file)
603*1da177e4SLinus Torvalds {
604*1da177e4SLinus Torvalds 	struct service_processor *sp = file->private_data;
605*1da177e4SLinus Torvalds 
606*1da177e4SLinus Torvalds 	disable_mouse_interrupts(sp);
607*1da177e4SLinus Torvalds 	wake_up_interruptible(&sp->remote_queue.wait);
608*1da177e4SLinus Torvalds 	sp->remote_queue.open = 0;
609*1da177e4SLinus Torvalds 
610*1da177e4SLinus Torvalds 	return 0;
611*1da177e4SLinus Torvalds }
612*1da177e4SLinus Torvalds 
613*1da177e4SLinus Torvalds static ssize_t remote_event_file_read(struct file *file, char __user *buf, size_t count, loff_t *offset)
614*1da177e4SLinus Torvalds {
615*1da177e4SLinus Torvalds 	struct service_processor *sp = file->private_data;
616*1da177e4SLinus Torvalds 	struct remote_queue *q = &sp->remote_queue;
617*1da177e4SLinus Torvalds 	size_t data_size;
618*1da177e4SLinus Torvalds 	struct remote_event *reader = q->reader;
619*1da177e4SLinus Torvalds 	size_t num_events;
620*1da177e4SLinus Torvalds 
621*1da177e4SLinus Torvalds 	if (*offset < 0)
622*1da177e4SLinus Torvalds 		return -EINVAL;
623*1da177e4SLinus Torvalds 	if (count == 0 || count > 1024)
624*1da177e4SLinus Torvalds 		return 0;
625*1da177e4SLinus Torvalds 	if (*offset != 0)
626*1da177e4SLinus Torvalds 		return 0;
627*1da177e4SLinus Torvalds 
628*1da177e4SLinus Torvalds 	if (wait_event_interruptible(q->wait, q->reader != q->writer))
629*1da177e4SLinus Torvalds 		return -ERESTARTSYS;
630*1da177e4SLinus Torvalds 
631*1da177e4SLinus Torvalds 	/* only get multiples of struct remote_event */
632*1da177e4SLinus Torvalds 	num_events = min((count/sizeof(struct remote_event)), ibmasm_events_available(q));
633*1da177e4SLinus Torvalds 	if (!num_events)
634*1da177e4SLinus Torvalds 		return 0;
635*1da177e4SLinus Torvalds 
636*1da177e4SLinus Torvalds 	data_size = num_events * sizeof(struct remote_event);
637*1da177e4SLinus Torvalds 
638*1da177e4SLinus Torvalds 	if (copy_to_user(buf, reader, data_size))
639*1da177e4SLinus Torvalds 		return -EFAULT;
640*1da177e4SLinus Torvalds 
641*1da177e4SLinus Torvalds 	ibmasm_advance_reader(q, num_events);
642*1da177e4SLinus Torvalds 
643*1da177e4SLinus Torvalds 	return data_size;
644*1da177e4SLinus Torvalds }
645*1da177e4SLinus Torvalds 
646*1da177e4SLinus Torvalds 
647*1da177e4SLinus Torvalds static struct file_operations command_fops = {
648*1da177e4SLinus Torvalds 	.open =		command_file_open,
649*1da177e4SLinus Torvalds 	.release =	command_file_close,
650*1da177e4SLinus Torvalds 	.read =		command_file_read,
651*1da177e4SLinus Torvalds 	.write =	command_file_write,
652*1da177e4SLinus Torvalds };
653*1da177e4SLinus Torvalds 
654*1da177e4SLinus Torvalds static struct file_operations event_fops = {
655*1da177e4SLinus Torvalds 	.open =		event_file_open,
656*1da177e4SLinus Torvalds 	.release =	event_file_close,
657*1da177e4SLinus Torvalds 	.read =		event_file_read,
658*1da177e4SLinus Torvalds 	.write =	event_file_write,
659*1da177e4SLinus Torvalds };
660*1da177e4SLinus Torvalds 
661*1da177e4SLinus Torvalds static struct file_operations r_heartbeat_fops = {
662*1da177e4SLinus Torvalds 	.open =		r_heartbeat_file_open,
663*1da177e4SLinus Torvalds 	.release =	r_heartbeat_file_close,
664*1da177e4SLinus Torvalds 	.read =		r_heartbeat_file_read,
665*1da177e4SLinus Torvalds 	.write =	r_heartbeat_file_write,
666*1da177e4SLinus Torvalds };
667*1da177e4SLinus Torvalds 
668*1da177e4SLinus Torvalds static struct file_operations remote_settings_fops = {
669*1da177e4SLinus Torvalds 	.open =		remote_settings_file_open,
670*1da177e4SLinus Torvalds 	.release =	remote_settings_file_close,
671*1da177e4SLinus Torvalds 	.read =		remote_settings_file_read,
672*1da177e4SLinus Torvalds 	.write =	remote_settings_file_write,
673*1da177e4SLinus Torvalds };
674*1da177e4SLinus Torvalds 
675*1da177e4SLinus Torvalds static struct file_operations remote_event_fops = {
676*1da177e4SLinus Torvalds 	.open =		remote_event_file_open,
677*1da177e4SLinus Torvalds 	.release =	remote_event_file_close,
678*1da177e4SLinus Torvalds 	.read =		remote_event_file_read,
679*1da177e4SLinus Torvalds };
680*1da177e4SLinus Torvalds 
681*1da177e4SLinus Torvalds 
682*1da177e4SLinus Torvalds static void ibmasmfs_create_files (struct super_block *sb, struct dentry *root)
683*1da177e4SLinus Torvalds {
684*1da177e4SLinus Torvalds 	struct list_head *entry;
685*1da177e4SLinus Torvalds 	struct service_processor *sp;
686*1da177e4SLinus Torvalds 
687*1da177e4SLinus Torvalds 	list_for_each(entry, &service_processors) {
688*1da177e4SLinus Torvalds 		struct dentry *dir;
689*1da177e4SLinus Torvalds 		struct dentry *remote_dir;
690*1da177e4SLinus Torvalds 		sp = list_entry(entry, struct service_processor, node);
691*1da177e4SLinus Torvalds 		dir = ibmasmfs_create_dir(sb, root, sp->dirname);
692*1da177e4SLinus Torvalds 		if (!dir)
693*1da177e4SLinus Torvalds 			continue;
694*1da177e4SLinus Torvalds 
695*1da177e4SLinus Torvalds 		ibmasmfs_create_file(sb, dir, "command", &command_fops, sp, S_IRUSR|S_IWUSR);
696*1da177e4SLinus Torvalds 		ibmasmfs_create_file(sb, dir, "event", &event_fops, sp, S_IRUSR|S_IWUSR);
697*1da177e4SLinus Torvalds 		ibmasmfs_create_file(sb, dir, "reverse_heartbeat", &r_heartbeat_fops, sp, S_IRUSR|S_IWUSR);
698*1da177e4SLinus Torvalds 
699*1da177e4SLinus Torvalds 		remote_dir = ibmasmfs_create_dir(sb, dir, "remote_video");
700*1da177e4SLinus Torvalds 		if (!remote_dir)
701*1da177e4SLinus Torvalds 			continue;
702*1da177e4SLinus Torvalds 
703*1da177e4SLinus Torvalds 		ibmasmfs_create_file(sb, remote_dir, "width", &remote_settings_fops, (void *)display_width(sp), S_IRUSR|S_IWUSR);
704*1da177e4SLinus Torvalds 		ibmasmfs_create_file(sb, remote_dir, "height", &remote_settings_fops, (void *)display_height(sp), S_IRUSR|S_IWUSR);
705*1da177e4SLinus Torvalds 		ibmasmfs_create_file(sb, remote_dir, "depth", &remote_settings_fops, (void *)display_depth(sp), S_IRUSR|S_IWUSR);
706*1da177e4SLinus Torvalds 		ibmasmfs_create_file(sb, remote_dir, "connected", &remote_settings_fops, (void *)vnc_status(sp), S_IRUSR);
707*1da177e4SLinus Torvalds 		ibmasmfs_create_file(sb, remote_dir, "events", &remote_event_fops, (void *)sp, S_IRUSR);
708*1da177e4SLinus Torvalds 	}
709*1da177e4SLinus Torvalds }
710