diff options
author | Elena Reshetova <elena.reshetova@intel.com> | 2017-12-20 14:10:55 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2017-12-20 15:23:44 -0500 |
commit | c380cd5a00f32a96b6c63148a939eb9f8836e184 (patch) | |
tree | e4cfa93ba6e47340d122c234027fa934776ac43a | |
parent | a943e8bc058977663ba1e42389d4f5643cf2ae9e (diff) |
net: convert lcs_reply.refcnt from atomic_t to refcount_t
atomic_t variables are currently used to implement reference
counters with the following properties:
- counter is initialized to 1 using atomic_set()
- a resource is freed upon counter reaching zero
- once counter reaches zero, its further
increments aren't allowed
- counter schema uses basic atomic operations
(set, inc, inc_not_zero, dec_and_test, etc.)
Such atomic variables should be converted to a newly provided
refcount_t type and API that prevents accidental counter overflows
and underflows. This is important since overflows and underflows
can lead to use-after-free situation and be exploitable.
The variable lcs_reply.refcnt is used as pure reference counter.
Convert it to refcount_t and fix up the operations.
Suggested-by: Kees Cook <keescook@chromium.org>
Reviewed-by: David Windsor <dwindsor@gmail.com>
Reviewed-by: Hans Liljestrand <ishkamiel@gmail.com>
Signed-off-by: Elena Reshetova <elena.reshetova@intel.com>
[jwi: removed the WARN_ONs. Use CONFIG_REFCOUNT_FULL if you care.]
Signed-off-by: Julian Wiedmann <jwi@linux.vnet.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | drivers/s390/net/lcs.c | 10 | ||||
-rw-r--r-- | drivers/s390/net/lcs.h | 3 |
2 files changed, 5 insertions, 8 deletions
diff --git a/drivers/s390/net/lcs.c b/drivers/s390/net/lcs.c index 92ae84a927fc..0ee8f33efb54 100644 --- a/drivers/s390/net/lcs.c +++ b/drivers/s390/net/lcs.c | |||
@@ -756,18 +756,14 @@ lcs_get_lancmd(struct lcs_card *card, int count) | |||
756 | static void | 756 | static void |
757 | lcs_get_reply(struct lcs_reply *reply) | 757 | lcs_get_reply(struct lcs_reply *reply) |
758 | { | 758 | { |
759 | WARN_ON(atomic_read(&reply->refcnt) <= 0); | 759 | refcount_inc(&reply->refcnt); |
760 | atomic_inc(&reply->refcnt); | ||
761 | } | 760 | } |
762 | 761 | ||
763 | static void | 762 | static void |
764 | lcs_put_reply(struct lcs_reply *reply) | 763 | lcs_put_reply(struct lcs_reply *reply) |
765 | { | 764 | { |
766 | WARN_ON(atomic_read(&reply->refcnt) <= 0); | 765 | if (refcount_dec_and_test(&reply->refcnt)) |
767 | if (atomic_dec_and_test(&reply->refcnt)) { | ||
768 | kfree(reply); | 766 | kfree(reply); |
769 | } | ||
770 | |||
771 | } | 767 | } |
772 | 768 | ||
773 | static struct lcs_reply * | 769 | static struct lcs_reply * |
@@ -780,7 +776,7 @@ lcs_alloc_reply(struct lcs_cmd *cmd) | |||
780 | reply = kzalloc(sizeof(struct lcs_reply), GFP_ATOMIC); | 776 | reply = kzalloc(sizeof(struct lcs_reply), GFP_ATOMIC); |
781 | if (!reply) | 777 | if (!reply) |
782 | return NULL; | 778 | return NULL; |
783 | atomic_set(&reply->refcnt,1); | 779 | refcount_set(&reply->refcnt, 1); |
784 | reply->sequence_no = cmd->sequence_no; | 780 | reply->sequence_no = cmd->sequence_no; |
785 | reply->received = 0; | 781 | reply->received = 0; |
786 | reply->rc = 0; | 782 | reply->rc = 0; |
diff --git a/drivers/s390/net/lcs.h b/drivers/s390/net/lcs.h index fbc8b90b1f85..bd52caa3b11b 100644 --- a/drivers/s390/net/lcs.h +++ b/drivers/s390/net/lcs.h | |||
@@ -5,6 +5,7 @@ | |||
5 | #include <linux/netdevice.h> | 5 | #include <linux/netdevice.h> |
6 | #include <linux/skbuff.h> | 6 | #include <linux/skbuff.h> |
7 | #include <linux/workqueue.h> | 7 | #include <linux/workqueue.h> |
8 | #include <linux/refcount.h> | ||
8 | #include <asm/ccwdev.h> | 9 | #include <asm/ccwdev.h> |
9 | 10 | ||
10 | #define LCS_DBF_TEXT(level, name, text) \ | 11 | #define LCS_DBF_TEXT(level, name, text) \ |
@@ -271,7 +272,7 @@ struct lcs_buffer { | |||
271 | struct lcs_reply { | 272 | struct lcs_reply { |
272 | struct list_head list; | 273 | struct list_head list; |
273 | __u16 sequence_no; | 274 | __u16 sequence_no; |
274 | atomic_t refcnt; | 275 | refcount_t refcnt; |
275 | /* Callback for completion notification. */ | 276 | /* Callback for completion notification. */ |
276 | void (*callback)(struct lcs_card *, struct lcs_cmd *); | 277 | void (*callback)(struct lcs_card *, struct lcs_cmd *); |
277 | wait_queue_head_t wait_q; | 278 | wait_queue_head_t wait_q; |