d634bd01b3
commit b74555de21acd791f12c4a1aeaf653dd7ac21133 upstream. syzbot reported: BUG: memory leak unreferenced object 0xffff88811eb3de00 (size 224): comm "syz-executor559", pid 7315, jiffies 4294943019 (age 10.300s) hex dump (first 32 bytes): 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................ 00 a0 38 24 81 88 ff ff 00 c0 f2 15 81 88 ff ff ..8$............ backtrace: [<000000008d1c66a1>] kmemleak_alloc_recursive include/linux/kmemleak.h:55 [inline] [<000000008d1c66a1>] slab_post_alloc_hook mm/slab.h:439 [inline] [<000000008d1c66a1>] slab_alloc_node mm/slab.c:3269 [inline] [<000000008d1c66a1>] kmem_cache_alloc_node+0x153/0x2a0 mm/slab.c:3579 [<00000000447d9496>] __alloc_skb+0x6e/0x210 net/core/skbuff.c:198 [<000000000cdbf82f>] alloc_skb include/linux/skbuff.h:1058 [inline] [<000000000cdbf82f>] llc_alloc_frame+0x66/0x110 net/llc/llc_sap.c:54 [<000000002418b52e>] llc_conn_ac_send_sabme_cmd_p_set_x+0x2f/0x140 net/llc/llc_c_ac.c:777 [<000000001372ae17>] llc_exec_conn_trans_actions net/llc/llc_conn.c:475 [inline] [<000000001372ae17>] llc_conn_service net/llc/llc_conn.c:400 [inline] [<000000001372ae17>] llc_conn_state_process+0x1ac/0x640 net/llc/llc_conn.c:75 [<00000000f27e53c1>] llc_establish_connection+0x110/0x170 net/llc/llc_if.c:109 [<00000000291b2ca0>] llc_ui_connect+0x10e/0x370 net/llc/af_llc.c:477 [<000000000f9c740b>] __sys_connect+0x11d/0x170 net/socket.c:1840 [...] The bug is that most callers of llc_conn_send_pdu() assume it consumes a reference to the skb, when actually due to commitb85ab56c3f
("llc: properly handle dev_queue_xmit() return value") it doesn't. Revert most of that commit, and instead make the few places that need llc_conn_send_pdu() to *not* consume a reference call skb_get() before. Fixes:b85ab56c3f
("llc: properly handle dev_queue_xmit() return value") Reported-by: syzbot+6b825a6494a04cc0e3f7@syzkaller.appspotmail.com Signed-off-by: Eric Biggers <ebiggers@google.com> Signed-off-by: Jakub Kicinski <jakub.kicinski@netronome.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
119 lines
4.1 KiB
C
119 lines
4.1 KiB
C
#ifndef LLC_CONN_H
|
|
#define LLC_CONN_H
|
|
/*
|
|
* Copyright (c) 1997 by Procom Technology, Inc.
|
|
* 2001, 2002 by Arnaldo Carvalho de Melo <acme@conectiva.com.br>
|
|
*
|
|
* This program can be redistributed or modified under the terms of the
|
|
* GNU General Public License as published by the Free Software Foundation.
|
|
* This program is distributed without any warranty or implied warranty
|
|
* of merchantability or fitness for a particular purpose.
|
|
*
|
|
* See the GNU General Public License for more details.
|
|
*/
|
|
#include <linux/timer.h>
|
|
#include <net/llc_if.h>
|
|
#include <net/sock.h>
|
|
#include <linux/llc.h>
|
|
|
|
#define LLC_EVENT 1
|
|
#define LLC_PACKET 2
|
|
|
|
#define LLC2_P_TIME 2
|
|
#define LLC2_ACK_TIME 1
|
|
#define LLC2_REJ_TIME 3
|
|
#define LLC2_BUSY_TIME 3
|
|
|
|
struct llc_timer {
|
|
struct timer_list timer;
|
|
unsigned long expire; /* timer expire time */
|
|
};
|
|
|
|
struct llc_sock {
|
|
/* struct sock must be the first member of llc_sock */
|
|
struct sock sk;
|
|
struct sockaddr_llc addr; /* address sock is bound to */
|
|
u8 state; /* state of connection */
|
|
struct llc_sap *sap; /* pointer to parent SAP */
|
|
struct llc_addr laddr; /* lsap/mac pair */
|
|
struct llc_addr daddr; /* dsap/mac pair */
|
|
struct net_device *dev; /* device to send to remote */
|
|
u32 copied_seq; /* head of yet unread data */
|
|
u8 retry_count; /* number of retries */
|
|
u8 ack_must_be_send;
|
|
u8 first_pdu_Ns;
|
|
u8 npta;
|
|
struct llc_timer ack_timer;
|
|
struct llc_timer pf_cycle_timer;
|
|
struct llc_timer rej_sent_timer;
|
|
struct llc_timer busy_state_timer; /* ind busy clr at remote LLC */
|
|
u8 vS; /* seq# next in-seq I-PDU tx'd*/
|
|
u8 vR; /* seq# next in-seq I-PDU rx'd*/
|
|
u32 n2; /* max nbr re-tx's for timeout*/
|
|
u32 n1; /* max nbr octets in I PDU */
|
|
u8 k; /* tx window size; max = 127 */
|
|
u8 rw; /* rx window size; max = 127 */
|
|
u8 p_flag; /* state flags */
|
|
u8 f_flag;
|
|
u8 s_flag;
|
|
u8 data_flag;
|
|
u8 remote_busy_flag;
|
|
u8 cause_flag;
|
|
struct sk_buff_head pdu_unack_q; /* PUDs sent/waiting ack */
|
|
u16 link; /* network layer link number */
|
|
u8 X; /* a temporary variable */
|
|
u8 ack_pf; /* this flag indicates what is
|
|
the P-bit of acknowledge */
|
|
u8 failed_data_req; /* recognize that already exist a
|
|
failed llc_data_req_handler
|
|
(tx_buffer_full or unacceptable
|
|
state */
|
|
u8 dec_step;
|
|
u8 inc_cntr;
|
|
u8 dec_cntr;
|
|
u8 connect_step;
|
|
u8 last_nr; /* NR of last pdu received */
|
|
u32 rx_pdu_hdr; /* used for saving header of last pdu
|
|
received and caused sending FRMR.
|
|
Used for resending FRMR */
|
|
u32 cmsg_flags;
|
|
struct hlist_node dev_hash_node;
|
|
};
|
|
|
|
static inline struct llc_sock *llc_sk(const struct sock *sk)
|
|
{
|
|
return (struct llc_sock *)sk;
|
|
}
|
|
|
|
static __inline__ void llc_set_backlog_type(struct sk_buff *skb, char type)
|
|
{
|
|
skb->cb[sizeof(skb->cb) - 1] = type;
|
|
}
|
|
|
|
static __inline__ char llc_backlog_type(struct sk_buff *skb)
|
|
{
|
|
return skb->cb[sizeof(skb->cb) - 1];
|
|
}
|
|
|
|
struct sock *llc_sk_alloc(struct net *net, int family, gfp_t priority,
|
|
struct proto *prot, int kern);
|
|
void llc_sk_stop_all_timers(struct sock *sk, bool sync);
|
|
void llc_sk_free(struct sock *sk);
|
|
|
|
void llc_sk_reset(struct sock *sk);
|
|
|
|
/* Access to a connection */
|
|
int llc_conn_state_process(struct sock *sk, struct sk_buff *skb);
|
|
void llc_conn_send_pdu(struct sock *sk, struct sk_buff *skb);
|
|
void llc_conn_rtn_pdu(struct sock *sk, struct sk_buff *skb);
|
|
void llc_conn_resend_i_pdu_as_cmd(struct sock *sk, u8 nr, u8 first_p_bit);
|
|
void llc_conn_resend_i_pdu_as_rsp(struct sock *sk, u8 nr, u8 first_f_bit);
|
|
int llc_conn_remove_acked_pdus(struct sock *conn, u8 nr, u16 *how_many_unacked);
|
|
struct sock *llc_lookup_established(struct llc_sap *sap, struct llc_addr *daddr,
|
|
struct llc_addr *laddr);
|
|
void llc_sap_add_socket(struct llc_sap *sap, struct sock *sk);
|
|
void llc_sap_remove_socket(struct llc_sap *sap, struct sock *sk);
|
|
|
|
u8 llc_data_accept_state(u8 state);
|
|
void llc_build_offset_table(void);
|
|
#endif /* LLC_CONN_H */
|