2009-05-22 01: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-27 03:09:06 +04:00
# include <linux/atomic.h>
2009-05-22 01:01:20 +04:00
/*
* Final freeing of a group
*/
2014-10-10 02:24:35 +04:00
static void fsnotify_final_destroy_group ( struct fsnotify_group * group )
2009-05-22 01:01:20 +04:00
{
if ( group - > ops - > free_group_priv )
group - > ops - > free_group_priv ( group ) ;
kfree ( group ) ;
}
2016-09-20 00:44:27 +03:00
/*
* Stop queueing new events for this group . Once this function returns
* fsnotify_add_event ( ) will not add any new events to the group ' s queue .
*/
void fsnotify_group_stop_queueing ( struct fsnotify_group * group )
{
2016-10-08 02:56:52 +03:00
spin_lock ( & group - > notification_lock ) ;
2016-09-20 00:44:27 +03:00
group - > shutdown = true ;
2016-10-08 02:56:52 +03:00
spin_unlock ( & group - > notification_lock ) ;
2016-09-20 00:44:27 +03:00
}
2009-05-22 01:01:26 +04:00
/*
2011-06-14 19:29:47 +04: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-22 01:01:26 +04:00
*/
2011-06-14 19:29:45 +04:00
void fsnotify_destroy_group ( struct fsnotify_group * group )
2009-05-22 01:01:26 +04:00
{
2016-09-20 00:44:27 +03:00
/*
* Stop queueing new events . The code below is careful enough to not
* require this but fanotify needs to stop queuing events even before
* fsnotify_destroy_group ( ) is called and this makes the other callers
* of fsnotify_destroy_group ( ) to see the same behavior .
*/
fsnotify_group_stop_queueing ( group ) ;
2016-12-21 16:48:18 +03:00
/* Clear all marks for this group and queue them for destruction */
2017-01-04 12:51:58 +03:00
fsnotify_clear_marks_by_group ( group , FSNOTIFY_OBJ_ALL_TYPES ) ;
/*
* Some marks can still be pinned when waiting for response from
* userspace . Wait for those now . fsnotify_prepare_user_wait ( ) will
* not succeed now so this wait is race - free .
*/
wait_event ( group - > notification_waitq , ! atomic_read ( & group - > user_waits ) ) ;
2009-05-22 01:01:26 +04:00
2016-05-20 03:08:59 +03:00
/*
2016-12-21 16:48:18 +03:00
* Wait until all marks get really destroyed . We could actually destroy
* them ourselves instead of waiting for worker to do it , however that
* would be racy as worker can already be processing some marks before
* we even entered fsnotify_destroy_group ( ) .
2016-05-20 03:08:59 +03:00
*/
2016-12-21 16:48:18 +03:00
fsnotify_wait_marks_destroyed ( ) ;
2010-07-28 18:18:38 +04:00
2016-05-20 03:08:59 +03:00
/*
* Since we have waited for fsnotify_mark_srcu in
* fsnotify_mark_destroy_list ( ) there can be no outstanding event
* notification against this group . So clearing the notification queue
* of all events is reliable now .
*/
2011-06-14 19:29:47 +04:00
fsnotify_flush_notify ( group ) ;
2014-02-21 22:14:11 +04: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 19:29:47 +04:00
fsnotify_put_group ( group ) ;
2009-05-22 01:01:26 +04:00
}
2011-06-14 19:29:46 +04:00
/*
* Get reference to a group .
*/
void fsnotify_get_group ( struct fsnotify_group * group )
{
atomic_inc ( & group - > refcnt ) ;
}
2009-05-22 01: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 18:18:39 +04:00
if ( atomic_dec_and_test ( & group - > refcnt ) )
2011-06-14 19:29:47 +04:00
fsnotify_final_destroy_group ( group ) ;
2009-05-22 01:01:20 +04:00
}
/*
2009-12-18 05:24:22 +03:00
* Create a new fsnotify_group and hold a reference for the group returned .
2009-05-22 01:01:20 +04:00
*/
2009-12-18 05:24:22 +03:00
struct fsnotify_group * fsnotify_alloc_group ( const struct fsnotify_ops * ops )
2009-05-22 01:01:20 +04:00
{
2009-12-18 05:24:22 +03:00
struct fsnotify_group * group ;
2009-05-22 01:01:20 +04:00
2009-12-18 04:12:06 +03:00
group = kzalloc ( sizeof ( struct fsnotify_group ) , GFP_KERNEL ) ;
2009-05-22 01:01:20 +04:00
if ( ! group )
return ERR_PTR ( - ENOMEM ) ;
2009-12-18 05:24:23 +03:00
/* set to 0 when there a no external references to this group */
2009-05-22 01:01:20 +04:00
atomic_set ( & group - > refcnt , 1 ) ;
2011-06-14 19:29:47 +04:00
atomic_set ( & group - > num_marks , 0 ) ;
2016-11-10 18:02:11 +03:00
atomic_set ( & group - > user_waits , 0 ) ;
2009-12-18 05:24:23 +03:00
2016-10-08 02:56:52 +03:00
spin_lock_init ( & group - > notification_lock ) ;
2009-05-22 01:01:37 +04:00
INIT_LIST_HEAD ( & group - > notification_list ) ;
init_waitqueue_head ( & group - > notification_waitq ) ;
group - > max_events = UINT_MAX ;
2011-06-14 19:29:50 +04:00
mutex_init ( & group - > mark_mutex ) ;
2009-12-18 05:24:24 +03:00
INIT_LIST_HEAD ( & group - > marks_list ) ;
2009-05-22 01:01:26 +04:00
2009-05-22 01:01:20 +04:00
group - > ops = ops ;
return group ;
}
2011-10-15 01: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 ;
}