summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorElena Reshetova <elena.reshetova@intel.com>2017-12-20 14:10:55 -0500
committerDavid S. Miller <davem@davemloft.net>2017-12-20 15:23:44 -0500
commitc380cd5a00f32a96b6c63148a939eb9f8836e184 (patch)
treee4cfa93ba6e47340d122c234027fa934776ac43a
parenta943e8bc058977663ba1e42389d4f5643cf2ae9e (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.c10
-rw-r--r--drivers/s390/net/lcs.h3
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)
756static void 756static void
757lcs_get_reply(struct lcs_reply *reply) 757lcs_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
763static void 762static void
764lcs_put_reply(struct lcs_reply *reply) 763lcs_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
773static struct lcs_reply * 769static 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 {
271struct lcs_reply { 272struct 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;