2009-05-21 17:01:20 -04:00
/*
* Copyright ( C ) 2008 Red Hat , Inc . , Eric Paris < eparis @ redhat . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 , or ( at your option )
* any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; see the file COPYING . If not , write to
* the Free Software Foundation , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include <linux/list.h>
# include <linux/mutex.h>
# include <linux/slab.h>
# include <linux/srcu.h>
# include <linux/rculist.h>
# include <linux/wait.h>
# include <linux/fsnotify_backend.h>
# include "fsnotify.h"
2011-07-26 16:09:06 -07:00
# include <linux/atomic.h>
2009-05-21 17:01:20 -04:00
/*
* Final freeing of a group
*/
2014-10-09 15:24:35 -07:00
static void fsnotify_final_destroy_group ( struct fsnotify_group * group )
2009-05-21 17:01:20 -04:00
{
if ( group - > ops - > free_group_priv )
group - > ops - > free_group_priv ( group ) ;
kfree ( group ) ;
}
2009-05-21 17:01:26 -04:00
/*
2011-06-14 17:29:47 +02:00
* Trying to get rid of a group . Remove all marks , flush all events and release
* the group reference .
* Note that another thread calling fsnotify_clear_marks_by_group ( ) may still
* hold a ref to the group .
2009-05-21 17:01:26 -04:00
*/
2011-06-14 17:29:45 +02:00
void fsnotify_destroy_group ( struct fsnotify_group * group )
2009-05-21 17:01:26 -04:00
{
2009-12-17 21:24:24 -05:00
/* clear all inode marks for this group */
2009-05-21 17:01:26 -04:00
fsnotify_clear_marks_by_group ( group ) ;
2010-07-28 10:18:38 -04:00
synchronize_srcu ( & fsnotify_mark_srcu ) ;
2011-06-14 17:29:47 +02:00
/* clear the notification queue of all events */
fsnotify_flush_notify ( group ) ;
2014-02-21 19:14:11 +01:00
/*
* Destroy overflow event ( we cannot use fsnotify_destroy_event ( ) as
* that deliberately ignores overflow events .
*/
if ( group - > overflow_event )
group - > ops - > free_event ( group - > overflow_event ) ;
2011-06-14 17:29:47 +02:00
fsnotify_put_group ( group ) ;
2009-05-21 17:01:26 -04:00
}
2011-06-14 17:29:46 +02:00
/*
* Get reference to a group .
*/
void fsnotify_get_group ( struct fsnotify_group * group )
{
atomic_inc ( & group - > refcnt ) ;
}
2009-05-21 17:01:20 -04:00
/*
* Drop a reference to a group . Free it if it ' s through .
*/
void fsnotify_put_group ( struct fsnotify_group * group )
{
2010-07-28 10:18:39 -04:00
if ( atomic_dec_and_test ( & group - > refcnt ) )
2011-06-14 17:29:47 +02:00
fsnotify_final_destroy_group ( group ) ;
2009-05-21 17:01:20 -04:00
}
/*
2009-12-17 21:24:22 -05:00
* Create a new fsnotify_group and hold a reference for the group returned .
2009-05-21 17:01:20 -04:00
*/
2009-12-17 21:24:22 -05:00
struct fsnotify_group * fsnotify_alloc_group ( const struct fsnotify_ops * ops )
2009-05-21 17:01:20 -04:00
{
2009-12-17 21:24:22 -05:00
struct fsnotify_group * group ;
2009-05-21 17:01:20 -04:00
2009-12-17 20:12:06 -05:00
group = kzalloc ( sizeof ( struct fsnotify_group ) , GFP_KERNEL ) ;
2009-05-21 17:01:20 -04:00
if ( ! group )
return ERR_PTR ( - ENOMEM ) ;
2009-12-17 21:24:23 -05:00
/* set to 0 when there a no external references to this group */
2009-05-21 17:01:20 -04:00
atomic_set ( & group - > refcnt , 1 ) ;
2011-06-14 17:29:47 +02:00
atomic_set ( & group - > num_marks , 0 ) ;
2009-12-17 21:24:23 -05:00
2009-05-21 17:01:37 -04:00
mutex_init ( & group - > notification_mutex ) ;
INIT_LIST_HEAD ( & group - > notification_list ) ;
init_waitqueue_head ( & group - > notification_waitq ) ;
group - > max_events = UINT_MAX ;
2011-06-14 17:29:50 +02:00
mutex_init ( & group - > mark_mutex ) ;
2009-12-17 21:24:24 -05:00
INIT_LIST_HEAD ( & group - > marks_list ) ;
2009-05-21 17:01:26 -04:00
2009-05-21 17:01:20 -04:00
group - > ops = ops ;
return group ;
}
2011-10-14 17:43:39 -04:00
int fsnotify_fasync ( int fd , struct file * file , int on )
{
struct fsnotify_group * group = file - > private_data ;
return fasync_helper ( fd , file , on , & group - > fsn_fa ) > = 0 ? 0 : - EIO ;
}