diff options
author | Peter Zijlstra <a.p.zijlstra@chello.nl> | 2011-12-13 04:36:20 -0500 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@suse.de> | 2011-12-13 11:18:25 -0500 |
commit | 4af679cd7cbb0a0d8774b5cdb34bffcaa4e86e52 (patch) | |
tree | b97bd32f0af95d22133e98a774966ced2eee0664 /include/linux/kref.h | |
parent | 175cad266774d09c57a62c65baa6b4e1edafdf3a (diff) |
kref: Inline all functions
These are tiny functions, there's no point in having them out-of-line.
Cc: Alexey Dobriyan <adobriyan@gmail.com>
Cc: Eric Dumazet <eric.dumazet@gmail.com>
Cc: Ingo Molnar <mingo@elte.hu>
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Link: http://lkml.kernel.org/n/tip-8eccvi2ur2fzgi00xdjlbf5z@git.kernel.org
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
Diffstat (limited to 'include/linux/kref.h')
-rw-r--r-- | include/linux/kref.h | 80 |
1 files changed, 75 insertions, 5 deletions
diff --git a/include/linux/kref.h b/include/linux/kref.h index d4a62ab2ee5e..1cbae9f2ef77 100644 --- a/include/linux/kref.h +++ b/include/linux/kref.h | |||
@@ -16,15 +16,85 @@ | |||
16 | #define _KREF_H_ | 16 | #define _KREF_H_ |
17 | 17 | ||
18 | #include <linux/types.h> | 18 | #include <linux/types.h> |
19 | #include <linux/slab.h> | ||
19 | 20 | ||
20 | struct kref { | 21 | struct kref { |
21 | atomic_t refcount; | 22 | atomic_t refcount; |
22 | }; | 23 | }; |
23 | 24 | ||
24 | void kref_init(struct kref *kref); | 25 | /** |
25 | void kref_get(struct kref *kref); | 26 | * kref_init - initialize object. |
26 | int kref_put(struct kref *kref, void (*release) (struct kref *kref)); | 27 | * @kref: object in question. |
27 | int kref_sub(struct kref *kref, unsigned int count, | 28 | */ |
28 | void (*release) (struct kref *kref)); | 29 | static inline void kref_init(struct kref *kref) |
30 | { | ||
31 | atomic_set(&kref->refcount, 1); | ||
32 | smp_mb(); | ||
33 | } | ||
34 | |||
35 | /** | ||
36 | * kref_get - increment refcount for object. | ||
37 | * @kref: object. | ||
38 | */ | ||
39 | static inline void kref_get(struct kref *kref) | ||
40 | { | ||
41 | WARN_ON(!atomic_read(&kref->refcount)); | ||
42 | atomic_inc(&kref->refcount); | ||
43 | smp_mb__after_atomic_inc(); | ||
44 | } | ||
45 | |||
46 | /** | ||
47 | * kref_put - decrement refcount for object. | ||
48 | * @kref: object. | ||
49 | * @release: pointer to the function that will clean up the object when the | ||
50 | * last reference to the object is released. | ||
51 | * This pointer is required, and it is not acceptable to pass kfree | ||
52 | * in as this function. | ||
53 | * | ||
54 | * Decrement the refcount, and if 0, call release(). | ||
55 | * Return 1 if the object was removed, otherwise return 0. Beware, if this | ||
56 | * function returns 0, you still can not count on the kref from remaining in | ||
57 | * memory. Only use the return value if you want to see if the kref is now | ||
58 | * gone, not present. | ||
59 | */ | ||
60 | static inline int kref_put(struct kref *kref, void (*release)(struct kref *kref)) | ||
61 | { | ||
62 | WARN_ON(release == NULL); | ||
63 | WARN_ON(release == (void (*)(struct kref *))kfree); | ||
64 | |||
65 | if (atomic_dec_and_test(&kref->refcount)) { | ||
66 | release(kref); | ||
67 | return 1; | ||
68 | } | ||
69 | return 0; | ||
70 | } | ||
71 | |||
72 | |||
73 | /** | ||
74 | * kref_sub - subtract a number of refcounts for object. | ||
75 | * @kref: object. | ||
76 | * @count: Number of recounts to subtract. | ||
77 | * @release: pointer to the function that will clean up the object when the | ||
78 | * last reference to the object is released. | ||
79 | * This pointer is required, and it is not acceptable to pass kfree | ||
80 | * in as this function. | ||
81 | * | ||
82 | * Subtract @count from the refcount, and if 0, call release(). | ||
83 | * Return 1 if the object was removed, otherwise return 0. Beware, if this | ||
84 | * function returns 0, you still can not count on the kref from remaining in | ||
85 | * memory. Only use the return value if you want to see if the kref is now | ||
86 | * gone, not present. | ||
87 | */ | ||
88 | static inline int kref_sub(struct kref *kref, unsigned int count, | ||
89 | void (*release)(struct kref *kref)) | ||
90 | { | ||
91 | WARN_ON(release == NULL); | ||
92 | WARN_ON(release == (void (*)(struct kref *))kfree); | ||
29 | 93 | ||
94 | if (atomic_sub_and_test((int) count, &kref->refcount)) { | ||
95 | release(kref); | ||
96 | return 1; | ||
97 | } | ||
98 | return 0; | ||
99 | } | ||
30 | #endif /* _KREF_H_ */ | 100 | #endif /* _KREF_H_ */ |