2 * Copyright (C) 2002 Jeff Dike (jdike@karaya.com)
3 * Licensed under the GPL
7 #include <linux/file.h>
8 #include <linux/module.h>
9 #include <linux/init.h>
10 #include <linux/slab.h>
11 #include <linux/list.h>
12 #include <linux/kernel.h>
13 #include <linux/ctype.h>
14 #include <linux/dcache.h>
15 #include <linux/statfs.h>
16 #include <asm/uaccess.h>
17 #include <asm/fcntl.h>
20 static int init_inode(struct inode
*inode
, struct dentry
*dentry
);
23 struct list_head list
;
24 char contents
[PAGE_SIZE
- sizeof(struct list_head
)];
27 struct hppfs_private
{
28 struct file
*proc_file
;
31 struct hppfs_data
*contents
;
34 struct hppfs_inode_info
{
35 struct dentry
*proc_dentry
;
36 struct inode vfs_inode
;
39 static inline struct hppfs_inode_info
*HPPFS_I(struct inode
*inode
)
41 return container_of(inode
, struct hppfs_inode_info
, vfs_inode
);
44 #define HPPFS_SUPER_MAGIC 0xb00000ee
46 static const struct super_operations hppfs_sbops
;
48 static int is_pid(struct dentry
*dentry
)
50 struct super_block
*sb
;
54 if((sb
->s_op
!= &hppfs_sbops
) || (dentry
->d_parent
!= sb
->s_root
))
57 for(i
= 0; i
< dentry
->d_name
.len
; i
++){
58 if(!isdigit(dentry
->d_name
.name
[i
]))
64 static char *dentry_name(struct dentry
*dentry
, int extra
)
66 struct dentry
*parent
;
73 while(parent
->d_parent
!= parent
){
75 len
+= strlen("pid") + 1;
76 else len
+= parent
->d_name
.len
+ 1;
77 parent
= parent
->d_parent
;
82 name
= kmalloc(len
+ extra
+ 1, GFP_KERNEL
);
83 if(name
== NULL
) return(NULL
);
87 while(parent
->d_parent
!= parent
){
90 seg_len
= strlen("pid");
93 seg_name
= parent
->d_name
.name
;
94 seg_len
= parent
->d_name
.len
;
99 strncpy(&name
[len
+ 1], seg_name
, seg_len
);
100 parent
= parent
->d_parent
;
102 strncpy(name
, root
, strlen(root
));
106 struct dentry_operations hppfs_dentry_ops
= {
109 static int file_removed(struct dentry
*dentry
, const char *file
)
115 if(file
!= NULL
) extra
+= strlen(file
) + 1;
117 host_file
= dentry_name(dentry
, extra
+ strlen("/remove"));
118 if(host_file
== NULL
){
119 printk("file_removed : allocation failed\n");
124 strcat(host_file
, "/");
125 strcat(host_file
, file
);
127 strcat(host_file
, "/remove");
129 fd
= os_open_file(host_file
, of_read(OPENFLAGS()), 0);
138 static void hppfs_read_inode(struct inode
*ino
)
140 struct inode
*proc_ino
;
142 if(HPPFS_I(ino
)->proc_dentry
== NULL
)
145 proc_ino
= HPPFS_I(ino
)->proc_dentry
->d_inode
;
146 ino
->i_uid
= proc_ino
->i_uid
;
147 ino
->i_gid
= proc_ino
->i_gid
;
148 ino
->i_atime
= proc_ino
->i_atime
;
149 ino
->i_mtime
= proc_ino
->i_mtime
;
150 ino
->i_ctime
= proc_ino
->i_ctime
;
151 ino
->i_ino
= proc_ino
->i_ino
;
152 ino
->i_mode
= proc_ino
->i_mode
;
153 ino
->i_nlink
= proc_ino
->i_nlink
;
154 ino
->i_size
= proc_ino
->i_size
;
155 ino
->i_blocks
= proc_ino
->i_blocks
;
158 static struct inode
*hppfs_iget(struct super_block
*sb
)
162 inode
= iget_locked(sb
, 0);
164 return ERR_PTR(-ENOMEM
);
165 if (inode
->i_state
& I_NEW
) {
166 hppfs_read_inode(inode
);
167 unlock_new_inode(inode
);
172 static struct dentry
*hppfs_lookup(struct inode
*ino
, struct dentry
*dentry
,
173 struct nameidata
*nd
)
175 struct dentry
*proc_dentry
, *new, *parent
;
179 deleted
= file_removed(dentry
, NULL
);
181 return(ERR_PTR(deleted
));
183 return(ERR_PTR(-ENOENT
));
186 parent
= HPPFS_I(ino
)->proc_dentry
;
187 mutex_lock(&parent
->d_inode
->i_mutex
);
188 proc_dentry
= d_lookup(parent
, &dentry
->d_name
);
189 if(proc_dentry
== NULL
){
190 proc_dentry
= d_alloc(parent
, &dentry
->d_name
);
191 if(proc_dentry
== NULL
){
192 mutex_unlock(&parent
->d_inode
->i_mutex
);
195 new = (*parent
->d_inode
->i_op
->lookup
)(parent
->d_inode
,
202 mutex_unlock(&parent
->d_inode
->i_mutex
);
204 if(IS_ERR(proc_dentry
))
207 inode
= hppfs_iget(ino
->i_sb
);
209 err
= PTR_ERR(inode
);
213 err
= init_inode(inode
, proc_dentry
);
217 hppfs_read_inode(inode
);
219 d_add(dentry
, inode
);
220 dentry
->d_op
= &hppfs_dentry_ops
;
228 return(ERR_PTR(err
));
231 static const struct inode_operations hppfs_file_iops
= {
234 static ssize_t
read_proc(struct file
*file
, char __user
*buf
, ssize_t count
,
235 loff_t
*ppos
, int is_user
)
237 ssize_t (*read
)(struct file
*, char __user
*, size_t, loff_t
*);
240 read
= file
->f_path
.dentry
->d_inode
->i_fop
->read
;
245 n
= (*read
)(file
, buf
, count
, &file
->f_pos
);
250 if(ppos
) *ppos
= file
->f_pos
;
254 static ssize_t
hppfs_read_file(int fd
, char __user
*buf
, ssize_t count
)
261 new_buf
= kmalloc(PAGE_SIZE
, GFP_KERNEL
);
263 printk("hppfs_read_file : kmalloc failed\n");
268 cur
= min_t(ssize_t
, count
, PAGE_SIZE
);
269 err
= os_read_file(fd
, new_buf
, cur
);
271 printk("hppfs_read : read failed, errno = %d\n",
279 if(copy_to_user(buf
, new_buf
, err
)){
292 static ssize_t
hppfs_read(struct file
*file
, char __user
*buf
, size_t count
,
295 struct hppfs_private
*hppfs
= file
->private_data
;
296 struct hppfs_data
*data
;
300 if(hppfs
->contents
!= NULL
){
301 if(*ppos
>= hppfs
->len
) return(0);
303 data
= hppfs
->contents
;
305 while(off
>= sizeof(data
->contents
)){
306 data
= list_entry(data
->list
.next
, struct hppfs_data
,
308 off
-= sizeof(data
->contents
);
311 if(off
+ count
> hppfs
->len
)
312 count
= hppfs
->len
- off
;
313 copy_to_user(buf
, &data
->contents
[off
], count
);
316 else if(hppfs
->host_fd
!= -1){
317 err
= os_seek_file(hppfs
->host_fd
, *ppos
);
319 printk("hppfs_read : seek failed, errno = %d\n", err
);
322 count
= hppfs_read_file(hppfs
->host_fd
, buf
, count
);
326 else count
= read_proc(hppfs
->proc_file
, buf
, count
, ppos
, 1);
331 static ssize_t
hppfs_write(struct file
*file
, const char __user
*buf
, size_t len
,
334 struct hppfs_private
*data
= file
->private_data
;
335 struct file
*proc_file
= data
->proc_file
;
336 ssize_t (*write
)(struct file
*, const char __user
*, size_t, loff_t
*);
339 write
= proc_file
->f_path
.dentry
->d_inode
->i_fop
->write
;
341 proc_file
->f_pos
= file
->f_pos
;
342 err
= (*write
)(proc_file
, buf
, len
, &proc_file
->f_pos
);
343 file
->f_pos
= proc_file
->f_pos
;
348 static int open_host_sock(char *host_file
, int *filter_out
)
353 end
= &host_file
[strlen(host_file
)];
356 fd
= os_connect_socket(host_file
);
362 fd
= os_connect_socket(host_file
);
366 static void free_contents(struct hppfs_data
*head
)
368 struct hppfs_data
*data
;
369 struct list_head
*ele
, *next
;
371 if(head
== NULL
) return;
373 list_for_each_safe(ele
, next
, &head
->list
){
374 data
= list_entry(ele
, struct hppfs_data
, list
);
380 static struct hppfs_data
*hppfs_get_data(int fd
, int filter
,
381 struct file
*proc_file
,
382 struct file
*hppfs_file
,
385 struct hppfs_data
*data
, *new, *head
;
389 data
= kmalloc(sizeof(*data
), GFP_KERNEL
);
391 printk("hppfs_get_data : head allocation failed\n");
395 INIT_LIST_HEAD(&data
->list
);
401 while((n
= read_proc(proc_file
, data
->contents
,
402 sizeof(data
->contents
), NULL
, 0)) > 0)
403 os_write_file(fd
, data
->contents
, n
);
404 err
= os_shutdown_socket(fd
, 0, 1);
406 printk("hppfs_get_data : failed to shut down "
412 n
= os_read_file(fd
, data
->contents
, sizeof(data
->contents
));
415 printk("hppfs_get_data : read failed, errno = %d\n",
424 if(n
< sizeof(data
->contents
))
427 new = kmalloc(sizeof(*data
), GFP_KERNEL
);
429 printk("hppfs_get_data : data allocation failed\n");
434 INIT_LIST_HEAD(&new->list
);
435 list_add(&new->list
, &data
->list
);
443 return(ERR_PTR(err
));
446 static struct hppfs_private
*hppfs_data(void)
448 struct hppfs_private
*data
;
450 data
= kmalloc(sizeof(*data
), GFP_KERNEL
);
454 *data
= ((struct hppfs_private
) { .host_fd
= -1,
456 .contents
= NULL
} );
460 static int file_mode(int fmode
)
462 if(fmode
== (FMODE_READ
| FMODE_WRITE
))
464 if(fmode
== FMODE_READ
)
466 if(fmode
== FMODE_WRITE
)
471 static int hppfs_open(struct inode
*inode
, struct file
*file
)
473 struct hppfs_private
*data
;
474 struct dentry
*proc_dentry
;
476 int err
, fd
, type
, filter
;
483 host_file
= dentry_name(file
->f_path
.dentry
, strlen("/rw"));
484 if(host_file
== NULL
)
487 proc_dentry
= HPPFS_I(inode
)->proc_dentry
;
489 /* XXX This isn't closed anywhere */
490 data
->proc_file
= dentry_open(dget(proc_dentry
), NULL
,
491 file_mode(file
->f_mode
));
492 err
= PTR_ERR(data
->proc_file
);
493 if(IS_ERR(data
->proc_file
))
496 type
= os_file_type(host_file
);
497 if(type
== OS_TYPE_FILE
){
498 fd
= os_open_file(host_file
, of_read(OPENFLAGS()), 0);
501 else printk("hppfs_open : failed to open '%s', errno = %d\n",
504 data
->contents
= NULL
;
506 else if(type
== OS_TYPE_DIR
){
507 fd
= open_host_sock(host_file
, &filter
);
509 data
->contents
= hppfs_get_data(fd
, filter
,
512 if(!IS_ERR(data
->contents
))
515 else printk("hppfs_open : failed to open a socket in "
516 "'%s', errno = %d\n", host_file
, -fd
);
520 file
->private_data
= data
;
526 free_contents(data
->contents
);
532 static int hppfs_dir_open(struct inode
*inode
, struct file
*file
)
534 struct hppfs_private
*data
;
535 struct dentry
*proc_dentry
;
543 proc_dentry
= HPPFS_I(inode
)->proc_dentry
;
544 data
->proc_file
= dentry_open(dget(proc_dentry
), NULL
,
545 file_mode(file
->f_mode
));
546 err
= PTR_ERR(data
->proc_file
);
547 if(IS_ERR(data
->proc_file
))
550 file
->private_data
= data
;
559 static loff_t
hppfs_llseek(struct file
*file
, loff_t off
, int where
)
561 struct hppfs_private
*data
= file
->private_data
;
562 struct file
*proc_file
= data
->proc_file
;
563 loff_t (*llseek
)(struct file
*, loff_t
, int);
566 llseek
= proc_file
->f_path
.dentry
->d_inode
->i_fop
->llseek
;
568 ret
= (*llseek
)(proc_file
, off
, where
);
573 return(default_llseek(file
, off
, where
));
576 static const struct file_operations hppfs_file_fops
= {
578 .llseek
= hppfs_llseek
,
580 .write
= hppfs_write
,
584 struct hppfs_dirent
{
587 struct dentry
*dentry
;
590 static int hppfs_filldir(void *d
, const char *name
, int size
,
591 loff_t offset
, u64 inode
, unsigned int type
)
593 struct hppfs_dirent
*dirent
= d
;
595 if(file_removed(dirent
->dentry
, name
))
598 return((*dirent
->filldir
)(dirent
->vfs_dirent
, name
, size
, offset
,
602 static int hppfs_readdir(struct file
*file
, void *ent
, filldir_t filldir
)
604 struct hppfs_private
*data
= file
->private_data
;
605 struct file
*proc_file
= data
->proc_file
;
606 int (*readdir
)(struct file
*, void *, filldir_t
);
607 struct hppfs_dirent dirent
= ((struct hppfs_dirent
)
610 .dentry
= file
->f_path
.dentry
} );
613 readdir
= proc_file
->f_path
.dentry
->d_inode
->i_fop
->readdir
;
615 proc_file
->f_pos
= file
->f_pos
;
616 err
= (*readdir
)(proc_file
, &dirent
, hppfs_filldir
);
617 file
->f_pos
= proc_file
->f_pos
;
622 static int hppfs_fsync(struct file
*file
, struct dentry
*dentry
, int datasync
)
627 static const struct file_operations hppfs_dir_fops
= {
629 .readdir
= hppfs_readdir
,
630 .open
= hppfs_dir_open
,
631 .fsync
= hppfs_fsync
,
634 static int hppfs_statfs(struct dentry
*dentry
, struct kstatfs
*sf
)
641 sf
->f_type
= HPPFS_SUPER_MAGIC
;
645 static struct inode
*hppfs_alloc_inode(struct super_block
*sb
)
647 struct hppfs_inode_info
*hi
;
649 hi
= kmalloc(sizeof(*hi
), GFP_KERNEL
);
653 *hi
= ((struct hppfs_inode_info
) { .proc_dentry
= NULL
});
654 inode_init_once(&hi
->vfs_inode
);
655 return(&hi
->vfs_inode
);
658 void hppfs_delete_inode(struct inode
*ino
)
663 static void hppfs_destroy_inode(struct inode
*inode
)
665 kfree(HPPFS_I(inode
));
668 static const struct super_operations hppfs_sbops
= {
669 .alloc_inode
= hppfs_alloc_inode
,
670 .destroy_inode
= hppfs_destroy_inode
,
671 .delete_inode
= hppfs_delete_inode
,
672 .statfs
= hppfs_statfs
,
675 static int hppfs_readlink(struct dentry
*dentry
, char __user
*buffer
, int buflen
)
677 struct file
*proc_file
;
678 struct dentry
*proc_dentry
;
681 proc_dentry
= HPPFS_I(dentry
->d_inode
)->proc_dentry
;
682 proc_file
= dentry_open(dget(proc_dentry
), NULL
, O_RDONLY
);
683 if (IS_ERR(proc_file
))
684 return PTR_ERR(proc_file
);
686 ret
= proc_dentry
->d_inode
->i_op
->readlink(proc_dentry
, buffer
, buflen
);
693 static void* hppfs_follow_link(struct dentry
*dentry
, struct nameidata
*nd
)
695 struct file
*proc_file
;
696 struct dentry
*proc_dentry
;
699 proc_dentry
= HPPFS_I(dentry
->d_inode
)->proc_dentry
;
700 proc_file
= dentry_open(dget(proc_dentry
), NULL
, O_RDONLY
);
701 if (IS_ERR(proc_file
))
704 ret
= proc_dentry
->d_inode
->i_op
->follow_link(proc_dentry
, nd
);
711 static const struct inode_operations hppfs_dir_iops
= {
712 .lookup
= hppfs_lookup
,
715 static const struct inode_operations hppfs_link_iops
= {
716 .readlink
= hppfs_readlink
,
717 .follow_link
= hppfs_follow_link
,
720 static int init_inode(struct inode
*inode
, struct dentry
*dentry
)
722 if(S_ISDIR(dentry
->d_inode
->i_mode
)){
723 inode
->i_op
= &hppfs_dir_iops
;
724 inode
->i_fop
= &hppfs_dir_fops
;
726 else if(S_ISLNK(dentry
->d_inode
->i_mode
)){
727 inode
->i_op
= &hppfs_link_iops
;
728 inode
->i_fop
= &hppfs_file_fops
;
731 inode
->i_op
= &hppfs_file_iops
;
732 inode
->i_fop
= &hppfs_file_fops
;
735 HPPFS_I(inode
)->proc_dentry
= dentry
;
740 static int hppfs_fill_super(struct super_block
*sb
, void *d
, int silent
)
742 struct inode
*root_inode
;
743 struct file_system_type
*procfs
;
744 struct super_block
*proc_sb
;
748 procfs
= get_fs_type("proc");
752 if(list_empty(&procfs
->fs_supers
))
755 proc_sb
= list_entry(procfs
->fs_supers
.next
, struct super_block
,
758 sb
->s_blocksize
= 1024;
759 sb
->s_blocksize_bits
= 10;
760 sb
->s_magic
= HPPFS_SUPER_MAGIC
;
761 sb
->s_op
= &hppfs_sbops
;
763 root_inode
= hppfs_iget(sb
);
764 if (IS_ERR(root_inode
)) {
765 err
= PTR_ERR(root_inode
);
769 err
= init_inode(root_inode
, proc_sb
->s_root
);
774 sb
->s_root
= d_alloc_root(root_inode
);
775 if(sb
->s_root
== NULL
)
778 hppfs_read_inode(root_inode
);
788 static int hppfs_read_super(struct file_system_type
*type
,
789 int flags
, const char *dev_name
,
790 void *data
, struct vfsmount
*mnt
)
792 return get_sb_nodev(type
, flags
, data
, hppfs_fill_super
, mnt
);
795 static struct file_system_type hppfs_type
= {
796 .owner
= THIS_MODULE
,
798 .get_sb
= hppfs_read_super
,
799 .kill_sb
= kill_anon_super
,
803 static int __init
init_hppfs(void)
805 return(register_filesystem(&hppfs_type
));
808 static void __exit
exit_hppfs(void)
810 unregister_filesystem(&hppfs_type
);
813 module_init(init_hppfs
)
814 module_exit(exit_hppfs
)
815 MODULE_LICENSE("GPL");
818 * Overrides for Emacs so that we follow Linus's tabbing style.
819 * Emacs will notice this stuff at the end of the file and automatically
820 * adjust the settings for this buffer only. This must remain at the end
822 * ---------------------------------------------------------------------------
824 * c-file-style: "linux"