Blame view

fs/nfs_common/grace.c 3 KB
af558e33b   J. Bruce Fields   nfsd: common grac...
1
2
  /*
   * Common code for control of lockd and nfsv4 grace periods.
f77900296   Jeff Layton   lockd: move lockd...
3
4
   *
   * Transplanted from lockd code
af558e33b   J. Bruce Fields   nfsd: common grac...
5
6
7
   */
  
  #include <linux/module.h>
db9c45534   Stanislav Kinsbursky   LockD: manage gra...
8
  #include <net/net_namespace.h>
f77900296   Jeff Layton   lockd: move lockd...
9
10
  #include <net/netns/generic.h>
  #include <linux/fs.h>
db9c45534   Stanislav Kinsbursky   LockD: manage gra...
11

f77900296   Jeff Layton   lockd: move lockd...
12
  static int grace_net_id;
af558e33b   J. Bruce Fields   nfsd: common grac...
13
14
15
16
  static DEFINE_SPINLOCK(grace_lock);
  
  /**
   * locks_start_grace
f77900296   Jeff Layton   lockd: move lockd...
17
   * @net: net namespace that this lock manager belongs to
af558e33b   J. Bruce Fields   nfsd: common grac...
18
19
20
21
22
23
24
25
26
   * @lm: who this grace period is for
   *
   * A grace period is a period during which locks should not be given
   * out.  Currently grace periods are only enforced by the two lock
   * managers (lockd and nfsd), using the locks_in_grace() function to
   * check when they are in a grace period.
   *
   * This function is called to start a grace period.
   */
f77900296   Jeff Layton   lockd: move lockd...
27
28
  void
  locks_start_grace(struct net *net, struct lock_manager *lm)
af558e33b   J. Bruce Fields   nfsd: common grac...
29
  {
f77900296   Jeff Layton   lockd: move lockd...
30
  	struct list_head *grace_list = net_generic(net, grace_net_id);
db9c45534   Stanislav Kinsbursky   LockD: manage gra...
31

af558e33b   J. Bruce Fields   nfsd: common grac...
32
  	spin_lock(&grace_lock);
f77900296   Jeff Layton   lockd: move lockd...
33
  	list_add(&lm->list, grace_list);
af558e33b   J. Bruce Fields   nfsd: common grac...
34
35
36
37
38
39
  	spin_unlock(&grace_lock);
  }
  EXPORT_SYMBOL_GPL(locks_start_grace);
  
  /**
   * locks_end_grace
f77900296   Jeff Layton   lockd: move lockd...
40
   * @net: net namespace that this lock manager belongs to
af558e33b   J. Bruce Fields   nfsd: common grac...
41
42
43
44
45
46
47
48
   * @lm: who this grace period is for
   *
   * Call this function to state that the given lock manager is ready to
   * resume regular locking.  The grace period will not end until all lock
   * managers that called locks_start_grace() also call locks_end_grace().
   * Note that callers count on it being safe to call this more than once,
   * and the second call should be a no-op.
   */
f77900296   Jeff Layton   lockd: move lockd...
49
50
  void
  locks_end_grace(struct lock_manager *lm)
af558e33b   J. Bruce Fields   nfsd: common grac...
51
52
53
54
55
56
57
58
59
60
61
62
63
64
  {
  	spin_lock(&grace_lock);
  	list_del_init(&lm->list);
  	spin_unlock(&grace_lock);
  }
  EXPORT_SYMBOL_GPL(locks_end_grace);
  
  /**
   * locks_in_grace
   *
   * Lock managers call this function to determine when it is OK for them
   * to answer ordinary lock requests, and when they should accept only
   * lock reclaims.
   */
f77900296   Jeff Layton   lockd: move lockd...
65
  int
c87fb4a37   J. Bruce Fields   lockd: NLM grace ...
66
  __state_in_grace(struct net *net, bool open)
af558e33b   J. Bruce Fields   nfsd: common grac...
67
  {
f77900296   Jeff Layton   lockd: move lockd...
68
  	struct list_head *grace_list = net_generic(net, grace_net_id);
c87fb4a37   J. Bruce Fields   lockd: NLM grace ...
69
  	struct lock_manager *lm;
db9c45534   Stanislav Kinsbursky   LockD: manage gra...
70

c87fb4a37   J. Bruce Fields   lockd: NLM grace ...
71
72
73
74
75
76
77
78
79
80
81
82
83
  	if (!open)
  		return !list_empty(grace_list);
  
  	list_for_each_entry(lm, grace_list, list) {
  		if (lm->block_opens)
  			return true;
  	}
  	return false;
  }
  
  int locks_in_grace(struct net *net)
  {
  	return __state_in_grace(net, 0);
af558e33b   J. Bruce Fields   nfsd: common grac...
84
85
  }
  EXPORT_SYMBOL_GPL(locks_in_grace);
f77900296   Jeff Layton   lockd: move lockd...
86

c87fb4a37   J. Bruce Fields   lockd: NLM grace ...
87
88
89
90
91
  int opens_in_grace(struct net *net)
  {
  	return __state_in_grace(net, 1);
  }
  EXPORT_SYMBOL_GPL(opens_in_grace);
f77900296   Jeff Layton   lockd: move lockd...
92
93
94
95
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
124
125
126
127
128
129
130
131
  static int __net_init
  grace_init_net(struct net *net)
  {
  	struct list_head *grace_list = net_generic(net, grace_net_id);
  
  	INIT_LIST_HEAD(grace_list);
  	return 0;
  }
  
  static void __net_exit
  grace_exit_net(struct net *net)
  {
  	struct list_head *grace_list = net_generic(net, grace_net_id);
  
  	BUG_ON(!list_empty(grace_list));
  }
  
  static struct pernet_operations grace_net_ops = {
  	.init = grace_init_net,
  	.exit = grace_exit_net,
  	.id   = &grace_net_id,
  	.size = sizeof(struct list_head),
  };
  
  static int __init
  init_grace(void)
  {
  	return register_pernet_subsys(&grace_net_ops);
  }
  
  static void __exit
  exit_grace(void)
  {
  	unregister_pernet_subsys(&grace_net_ops);
  }
  
  MODULE_AUTHOR("Jeff Layton <jlayton@primarydata.com>");
  MODULE_LICENSE("GPL");
  module_init(init_grace)
  module_exit(exit_grace)