Skip to content

Commit 38dc3b5

Browse files
author
Jakub Kicinski
committed
Merge branch 'llc-fix-sk_buff-refcounting'
Eric Biggers says: ==================== Patches 1-2 fix the memory leaks that syzbot has reported in net/llc: memory leak in llc_ui_create (2) memory leak in llc_ui_sendmsg memory leak in llc_conn_ac_send_sabme_cmd_p_set_x Patches 3-4 fix related bugs that I noticed while reading this code. Note: I've tested that this fixes the syzbot bugs, but otherwise I don't know of any way to test this code. ==================== Signed-off-by: Jakub Kicinski <[email protected]>
2 parents 503c9ad + 36453c8 commit 38dc3b5

File tree

7 files changed

+69
-89
lines changed

7 files changed

+69
-89
lines changed

include/net/llc_conn.h

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ void llc_sk_reset(struct sock *sk);
104104

105105
/* Access to a connection */
106106
int llc_conn_state_process(struct sock *sk, struct sk_buff *skb);
107-
int llc_conn_send_pdu(struct sock *sk, struct sk_buff *skb);
107+
void llc_conn_send_pdu(struct sock *sk, struct sk_buff *skb);
108108
void llc_conn_rtn_pdu(struct sock *sk, struct sk_buff *skb);
109109
void llc_conn_resend_i_pdu_as_cmd(struct sock *sk, u8 nr, u8 first_p_bit);
110110
void llc_conn_resend_i_pdu_as_rsp(struct sock *sk, u8 nr, u8 first_f_bit);

net/llc/af_llc.c

Lines changed: 20 additions & 14 deletions
Original file line numberDiff line numberDiff line change
@@ -113,22 +113,26 @@ static inline u8 llc_ui_header_len(struct sock *sk, struct sockaddr_llc *addr)
113113
*
114114
* Send data via reliable llc2 connection.
115115
* Returns 0 upon success, non-zero if action did not succeed.
116+
*
117+
* This function always consumes a reference to the skb.
116118
*/
117119
static int llc_ui_send_data(struct sock* sk, struct sk_buff *skb, int noblock)
118120
{
119121
struct llc_sock* llc = llc_sk(sk);
120-
int rc = 0;
121122

122123
if (unlikely(llc_data_accept_state(llc->state) ||
123124
llc->remote_busy_flag ||
124125
llc->p_flag)) {
125126
long timeout = sock_sndtimeo(sk, noblock);
127+
int rc;
126128

127129
rc = llc_ui_wait_for_busy_core(sk, timeout);
130+
if (rc) {
131+
kfree_skb(skb);
132+
return rc;
133+
}
128134
}
129-
if (unlikely(!rc))
130-
rc = llc_build_and_send_pkt(sk, skb);
131-
return rc;
135+
return llc_build_and_send_pkt(sk, skb);
132136
}
133137

134138
static void llc_ui_sk_init(struct socket *sock, struct sock *sk)
@@ -899,7 +903,7 @@ static int llc_ui_sendmsg(struct socket *sock, struct msghdr *msg, size_t len)
899903
DECLARE_SOCKADDR(struct sockaddr_llc *, addr, msg->msg_name);
900904
int flags = msg->msg_flags;
901905
int noblock = flags & MSG_DONTWAIT;
902-
struct sk_buff *skb;
906+
struct sk_buff *skb = NULL;
903907
size_t size = 0;
904908
int rc = -EINVAL, copied = 0, hdrlen;
905909

@@ -908,18 +912,18 @@ static int llc_ui_sendmsg(struct socket *sock, struct msghdr *msg, size_t len)
908912
lock_sock(sk);
909913
if (addr) {
910914
if (msg->msg_namelen < sizeof(*addr))
911-
goto release;
915+
goto out;
912916
} else {
913917
if (llc_ui_addr_null(&llc->addr))
914-
goto release;
918+
goto out;
915919
addr = &llc->addr;
916920
}
917921
/* must bind connection to sap if user hasn't done it. */
918922
if (sock_flag(sk, SOCK_ZAPPED)) {
919923
/* bind to sap with null dev, exclusive. */
920924
rc = llc_ui_autobind(sock, addr);
921925
if (rc)
922-
goto release;
926+
goto out;
923927
}
924928
hdrlen = llc->dev->hard_header_len + llc_ui_header_len(sk, addr);
925929
size = hdrlen + len;
@@ -928,12 +932,12 @@ static int llc_ui_sendmsg(struct socket *sock, struct msghdr *msg, size_t len)
928932
copied = size - hdrlen;
929933
rc = -EINVAL;
930934
if (copied < 0)
931-
goto release;
935+
goto out;
932936
release_sock(sk);
933937
skb = sock_alloc_send_skb(sk, size, noblock, &rc);
934938
lock_sock(sk);
935939
if (!skb)
936-
goto release;
940+
goto out;
937941
skb->dev = llc->dev;
938942
skb->protocol = llc_proto_type(addr->sllc_arphrd);
939943
skb_reserve(skb, hdrlen);
@@ -943,29 +947,31 @@ static int llc_ui_sendmsg(struct socket *sock, struct msghdr *msg, size_t len)
943947
if (sk->sk_type == SOCK_DGRAM || addr->sllc_ua) {
944948
llc_build_and_send_ui_pkt(llc->sap, skb, addr->sllc_mac,
945949
addr->sllc_sap);
950+
skb = NULL;
946951
goto out;
947952
}
948953
if (addr->sllc_test) {
949954
llc_build_and_send_test_pkt(llc->sap, skb, addr->sllc_mac,
950955
addr->sllc_sap);
956+
skb = NULL;
951957
goto out;
952958
}
953959
if (addr->sllc_xid) {
954960
llc_build_and_send_xid_pkt(llc->sap, skb, addr->sllc_mac,
955961
addr->sllc_sap);
962+
skb = NULL;
956963
goto out;
957964
}
958965
rc = -ENOPROTOOPT;
959966
if (!(sk->sk_type == SOCK_STREAM && !addr->sllc_ua))
960967
goto out;
961968
rc = llc_ui_send_data(sk, skb, noblock);
969+
skb = NULL;
962970
out:
963-
if (rc) {
964-
kfree_skb(skb);
965-
release:
971+
kfree_skb(skb);
972+
if (rc)
966973
dprintk("%s: failed sending from %02X to %02X: %d\n",
967974
__func__, llc->laddr.lsap, llc->daddr.lsap, rc);
968-
}
969975
release_sock(sk);
970976
return rc ? : copied;
971977
}

net/llc/llc_c_ac.c

Lines changed: 6 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -372,6 +372,7 @@ int llc_conn_ac_send_i_cmd_p_set_1(struct sock *sk, struct sk_buff *skb)
372372
llc_pdu_init_as_i_cmd(skb, 1, llc->vS, llc->vR);
373373
rc = llc_mac_hdr_init(skb, llc->dev->dev_addr, llc->daddr.mac);
374374
if (likely(!rc)) {
375+
skb_get(skb);
375376
llc_conn_send_pdu(sk, skb);
376377
llc_conn_ac_inc_vs_by_1(sk, skb);
377378
}
@@ -389,7 +390,8 @@ static int llc_conn_ac_send_i_cmd_p_set_0(struct sock *sk, struct sk_buff *skb)
389390
llc_pdu_init_as_i_cmd(skb, 0, llc->vS, llc->vR);
390391
rc = llc_mac_hdr_init(skb, llc->dev->dev_addr, llc->daddr.mac);
391392
if (likely(!rc)) {
392-
rc = llc_conn_send_pdu(sk, skb);
393+
skb_get(skb);
394+
llc_conn_send_pdu(sk, skb);
393395
llc_conn_ac_inc_vs_by_1(sk, skb);
394396
}
395397
return rc;
@@ -406,6 +408,7 @@ int llc_conn_ac_send_i_xxx_x_set_0(struct sock *sk, struct sk_buff *skb)
406408
llc_pdu_init_as_i_cmd(skb, 0, llc->vS, llc->vR);
407409
rc = llc_mac_hdr_init(skb, llc->dev->dev_addr, llc->daddr.mac);
408410
if (likely(!rc)) {
411+
skb_get(skb);
409412
llc_conn_send_pdu(sk, skb);
410413
llc_conn_ac_inc_vs_by_1(sk, skb);
411414
}
@@ -916,7 +919,8 @@ static int llc_conn_ac_send_i_rsp_f_set_ackpf(struct sock *sk,
916919
llc_pdu_init_as_i_cmd(skb, llc->ack_pf, llc->vS, llc->vR);
917920
rc = llc_mac_hdr_init(skb, llc->dev->dev_addr, llc->daddr.mac);
918921
if (likely(!rc)) {
919-
rc = llc_conn_send_pdu(sk, skb);
922+
skb_get(skb);
923+
llc_conn_send_pdu(sk, skb);
920924
llc_conn_ac_inc_vs_by_1(sk, skb);
921925
}
922926
return rc;

net/llc/llc_conn.c

Lines changed: 17 additions & 50 deletions
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@
3030
#endif
3131

3232
static int llc_find_offset(int state, int ev_type);
33-
static int llc_conn_send_pdus(struct sock *sk, struct sk_buff *skb);
33+
static void llc_conn_send_pdus(struct sock *sk);
3434
static int llc_conn_service(struct sock *sk, struct sk_buff *skb);
3535
static int llc_exec_conn_trans_actions(struct sock *sk,
3636
struct llc_conn_state_trans *trans,
@@ -55,41 +55,28 @@ int sysctl_llc2_busy_timeout = LLC2_BUSY_TIME * HZ;
5555
* (executing it's actions and changing state), upper layer will be
5656
* indicated or confirmed, if needed. Returns 0 for success, 1 for
5757
* failure. The socket lock has to be held before calling this function.
58+
*
59+
* This function always consumes a reference to the skb.
5860
*/
5961
int llc_conn_state_process(struct sock *sk, struct sk_buff *skb)
6062
{
6163
int rc;
6264
struct llc_sock *llc = llc_sk(skb->sk);
6365
struct llc_conn_state_ev *ev = llc_conn_ev(skb);
6466

65-
/*
66-
* We have to hold the skb, because llc_conn_service will kfree it in
67-
* the sending path and we need to look at the skb->cb, where we encode
68-
* llc_conn_state_ev.
69-
*/
70-
skb_get(skb);
7167
ev->ind_prim = ev->cfm_prim = 0;
7268
/*
7369
* Send event to state machine
7470
*/
7571
rc = llc_conn_service(skb->sk, skb);
7672
if (unlikely(rc != 0)) {
7773
printk(KERN_ERR "%s: llc_conn_service failed\n", __func__);
78-
goto out_kfree_skb;
79-
}
80-
81-
if (unlikely(!ev->ind_prim && !ev->cfm_prim)) {
82-
/* indicate or confirm not required */
83-
if (!skb->next)
84-
goto out_kfree_skb;
8574
goto out_skb_put;
8675
}
8776

88-
if (unlikely(ev->ind_prim && ev->cfm_prim)) /* Paranoia */
89-
skb_get(skb);
90-
9177
switch (ev->ind_prim) {
9278
case LLC_DATA_PRIM:
79+
skb_get(skb);
9380
llc_save_primitive(sk, skb, LLC_DATA_PRIM);
9481
if (unlikely(sock_queue_rcv_skb(sk, skb))) {
9582
/*
@@ -106,6 +93,7 @@ int llc_conn_state_process(struct sock *sk, struct sk_buff *skb)
10693
* skb->sk pointing to the newly created struct sock in
10794
* llc_conn_handler. -acme
10895
*/
96+
skb_get(skb);
10997
skb_queue_tail(&sk->sk_receive_queue, skb);
11098
sk->sk_state_change(sk);
11199
break;
@@ -121,7 +109,6 @@ int llc_conn_state_process(struct sock *sk, struct sk_buff *skb)
121109
sk->sk_state_change(sk);
122110
}
123111
}
124-
kfree_skb(skb);
125112
sock_put(sk);
126113
break;
127114
case LLC_RESET_PRIM:
@@ -130,14 +117,11 @@ int llc_conn_state_process(struct sock *sk, struct sk_buff *skb)
130117
* RESET is not being notified to upper layers for now
131118
*/
132119
printk(KERN_INFO "%s: received a reset ind!\n", __func__);
133-
kfree_skb(skb);
134120
break;
135121
default:
136-
if (ev->ind_prim) {
122+
if (ev->ind_prim)
137123
printk(KERN_INFO "%s: received unknown %d prim!\n",
138124
__func__, ev->ind_prim);
139-
kfree_skb(skb);
140-
}
141125
/* No indication */
142126
break;
143127
}
@@ -179,25 +163,22 @@ int llc_conn_state_process(struct sock *sk, struct sk_buff *skb)
179163
printk(KERN_INFO "%s: received a reset conf!\n", __func__);
180164
break;
181165
default:
182-
if (ev->cfm_prim) {
166+
if (ev->cfm_prim)
183167
printk(KERN_INFO "%s: received unknown %d prim!\n",
184168
__func__, ev->cfm_prim);
185-
break;
186-
}
187-
goto out_skb_put; /* No confirmation */
169+
/* No confirmation */
170+
break;
188171
}
189-
out_kfree_skb:
190-
kfree_skb(skb);
191172
out_skb_put:
192173
kfree_skb(skb);
193174
return rc;
194175
}
195176

196-
int llc_conn_send_pdu(struct sock *sk, struct sk_buff *skb)
177+
void llc_conn_send_pdu(struct sock *sk, struct sk_buff *skb)
197178
{
198179
/* queue PDU to send to MAC layer */
199180
skb_queue_tail(&sk->sk_write_queue, skb);
200-
return llc_conn_send_pdus(sk, skb);
181+
llc_conn_send_pdus(sk);
201182
}
202183

203184
/**
@@ -255,7 +236,7 @@ void llc_conn_resend_i_pdu_as_cmd(struct sock *sk, u8 nr, u8 first_p_bit)
255236
if (howmany_resend > 0)
256237
llc->vS = (llc->vS + 1) % LLC_2_SEQ_NBR_MODULO;
257238
/* any PDUs to re-send are queued up; start sending to MAC */
258-
llc_conn_send_pdus(sk, NULL);
239+
llc_conn_send_pdus(sk);
259240
out:;
260241
}
261242

@@ -296,7 +277,7 @@ void llc_conn_resend_i_pdu_as_rsp(struct sock *sk, u8 nr, u8 first_f_bit)
296277
if (howmany_resend > 0)
297278
llc->vS = (llc->vS + 1) % LLC_2_SEQ_NBR_MODULO;
298279
/* any PDUs to re-send are queued up; start sending to MAC */
299-
llc_conn_send_pdus(sk, NULL);
280+
llc_conn_send_pdus(sk);
300281
out:;
301282
}
302283

@@ -340,16 +321,12 @@ int llc_conn_remove_acked_pdus(struct sock *sk, u8 nr, u16 *how_many_unacked)
340321
/**
341322
* llc_conn_send_pdus - Sends queued PDUs
342323
* @sk: active connection
343-
* @hold_skb: the skb held by caller, or NULL if does not care
344324
*
345-
* Sends queued pdus to MAC layer for transmission. When @hold_skb is
346-
* NULL, always return 0. Otherwise, return 0 if @hold_skb is sent
347-
* successfully, or 1 for failure.
325+
* Sends queued pdus to MAC layer for transmission.
348326
*/
349-
static int llc_conn_send_pdus(struct sock *sk, struct sk_buff *hold_skb)
327+
static void llc_conn_send_pdus(struct sock *sk)
350328
{
351329
struct sk_buff *skb;
352-
int ret = 0;
353330

354331
while ((skb = skb_dequeue(&sk->sk_write_queue)) != NULL) {
355332
struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
@@ -361,20 +338,10 @@ static int llc_conn_send_pdus(struct sock *sk, struct sk_buff *hold_skb)
361338
skb_queue_tail(&llc_sk(sk)->pdu_unack_q, skb);
362339
if (!skb2)
363340
break;
364-
dev_queue_xmit(skb2);
365-
} else {
366-
bool is_target = skb == hold_skb;
367-
int rc;
368-
369-
if (is_target)
370-
skb_get(skb);
371-
rc = dev_queue_xmit(skb);
372-
if (is_target)
373-
ret = rc;
341+
skb = skb2;
374342
}
343+
dev_queue_xmit(skb);
375344
}
376-
377-
return ret;
378345
}
379346

380347
/**

net/llc/llc_if.c

Lines changed: 8 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,8 @@
3838
* closed and -EBUSY when sending data is not permitted in this state or
3939
* LLC has send an I pdu with p bit set to 1 and is waiting for it's
4040
* response.
41+
*
42+
* This function always consumes a reference to the skb.
4143
*/
4244
int llc_build_and_send_pkt(struct sock *sk, struct sk_buff *skb)
4345
{
@@ -46,20 +48,22 @@ int llc_build_and_send_pkt(struct sock *sk, struct sk_buff *skb)
4648
struct llc_sock *llc = llc_sk(sk);
4749

4850
if (unlikely(llc->state == LLC_CONN_STATE_ADM))
49-
goto out;
51+
goto out_free;
5052
rc = -EBUSY;
5153
if (unlikely(llc_data_accept_state(llc->state) || /* data_conn_refuse */
5254
llc->p_flag)) {
5355
llc->failed_data_req = 1;
54-
goto out;
56+
goto out_free;
5557
}
5658
ev = llc_conn_ev(skb);
5759
ev->type = LLC_CONN_EV_TYPE_PRIM;
5860
ev->prim = LLC_DATA_PRIM;
5961
ev->prim_type = LLC_PRIM_TYPE_REQ;
6062
skb->dev = llc->dev;
61-
rc = llc_conn_state_process(sk, skb);
62-
out:
63+
return llc_conn_state_process(sk, skb);
64+
65+
out_free:
66+
kfree_skb(skb);
6367
return rc;
6468
}
6569

0 commit comments

Comments
 (0)