mirror of
https://github.com/followmsi/android_kernel_google_msm.git
synced 2024-11-06 23:17:41 +00:00
Merge branch 'lineage-18.1' of https://github.com/LineageOS/android_kernel_google_msm into followmsi-12
This commit is contained in:
commit
3a6cf41324
44 changed files with 397 additions and 290 deletions
|
@ -487,7 +487,7 @@ SYSCALL_DEFINE6(sparc_ipc, unsigned int, call, int, first, unsigned long, second
|
||||||
switch (call) {
|
switch (call) {
|
||||||
case SHMAT: {
|
case SHMAT: {
|
||||||
ulong raddr;
|
ulong raddr;
|
||||||
err = do_shmat(first, ptr, (int)second, &raddr);
|
err = do_shmat(first, ptr, (int)second, &raddr, SHMLBA);
|
||||||
if (!err) {
|
if (!err) {
|
||||||
if (put_user(raddr,
|
if (put_user(raddr,
|
||||||
(ulong __user *) third))
|
(ulong __user *) third))
|
||||||
|
|
|
@ -44,7 +44,7 @@ asmlinkage long xtensa_shmat(int shmid, char __user *shmaddr, int shmflg)
|
||||||
unsigned long ret;
|
unsigned long ret;
|
||||||
long err;
|
long err;
|
||||||
|
|
||||||
err = do_shmat(shmid, shmaddr, shmflg, &ret);
|
err = do_shmat(shmid, shmaddr, shmflg, &ret, SHMLBA);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
return (long)ret;
|
return (long)ret;
|
||||||
|
|
|
@ -107,12 +107,14 @@ struct shmid_kernel /* private to the kernel */
|
||||||
#define SHM_NORESERVE 010000 /* don't check for reservations */
|
#define SHM_NORESERVE 010000 /* don't check for reservations */
|
||||||
|
|
||||||
#ifdef CONFIG_SYSVIPC
|
#ifdef CONFIG_SYSVIPC
|
||||||
long do_shmat(int shmid, char __user *shmaddr, int shmflg, unsigned long *addr);
|
long do_shmat(int shmid, char __user *shmaddr, int shmflg, unsigned long *addr,
|
||||||
|
unsigned long shmlba);
|
||||||
extern int is_file_shm_hugepages(struct file *file);
|
extern int is_file_shm_hugepages(struct file *file);
|
||||||
extern void exit_shm(struct task_struct *task);
|
extern void exit_shm(struct task_struct *task);
|
||||||
#else
|
#else
|
||||||
static inline long do_shmat(int shmid, char __user *shmaddr,
|
static inline long do_shmat(int shmid, char __user *shmaddr,
|
||||||
int shmflg, unsigned long *addr)
|
int shmflg, unsigned long *addr,
|
||||||
|
unsigned long shmlba)
|
||||||
{
|
{
|
||||||
return -ENOSYS;
|
return -ENOSYS;
|
||||||
}
|
}
|
||||||
|
|
|
@ -89,7 +89,8 @@ static inline void flowi4_init_output(struct flowi4 *fl4, int oif,
|
||||||
__u32 mark, __u8 tos, __u8 scope,
|
__u32 mark, __u8 tos, __u8 scope,
|
||||||
__u8 proto, __u8 flags,
|
__u8 proto, __u8 flags,
|
||||||
__be32 daddr, __be32 saddr,
|
__be32 daddr, __be32 saddr,
|
||||||
__be16 dport, __be16 sport)
|
__be16 dport, __be16 sport,
|
||||||
|
kuid_t uid)
|
||||||
{
|
{
|
||||||
fl4->flowi4_oif = oif;
|
fl4->flowi4_oif = oif;
|
||||||
fl4->flowi4_iif = LOOPBACK_IFINDEX;
|
fl4->flowi4_iif = LOOPBACK_IFINDEX;
|
||||||
|
@ -99,6 +100,7 @@ static inline void flowi4_init_output(struct flowi4 *fl4, int oif,
|
||||||
fl4->flowi4_proto = proto;
|
fl4->flowi4_proto = proto;
|
||||||
fl4->flowi4_flags = flags;
|
fl4->flowi4_flags = flags;
|
||||||
fl4->flowi4_secid = 0;
|
fl4->flowi4_secid = 0;
|
||||||
|
fl4->flowi4_uid = uid;
|
||||||
fl4->daddr = daddr;
|
fl4->daddr = daddr;
|
||||||
fl4->saddr = saddr;
|
fl4->saddr = saddr;
|
||||||
fl4->fl4_dport = dport;
|
fl4->fl4_dport = dport;
|
||||||
|
|
|
@ -169,6 +169,7 @@ struct ip_reply_arg {
|
||||||
/* -1 if not needed */
|
/* -1 if not needed */
|
||||||
int bound_dev_if;
|
int bound_dev_if;
|
||||||
u8 tos;
|
u8 tos;
|
||||||
|
kuid_t uid;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define IP_REPLY_ARG_NOSRCCHECK 1
|
#define IP_REPLY_ARG_NOSRCCHECK 1
|
||||||
|
|
|
@ -137,7 +137,7 @@ extern void rt6_redirect(const struct in6_addr *dest,
|
||||||
int on_link);
|
int on_link);
|
||||||
|
|
||||||
extern void ip6_update_pmtu(struct sk_buff *skb, struct net *net, __be32 mtu,
|
extern void ip6_update_pmtu(struct sk_buff *skb, struct net *net, __be32 mtu,
|
||||||
int oif, u32 mark);
|
int oif, u32 mark, kuid_t uid);
|
||||||
extern void ip6_sk_update_pmtu(struct sk_buff *skb, struct sock *sk,
|
extern void ip6_sk_update_pmtu(struct sk_buff *skb, struct sock *sk,
|
||||||
__be32 mtu);
|
__be32 mtu);
|
||||||
|
|
||||||
|
|
|
@ -146,7 +146,7 @@ static inline struct rtable *ip_route_output_ports(struct net *net, struct flowi
|
||||||
flowi4_init_output(fl4, oif, sk ? sk->sk_mark : 0, tos,
|
flowi4_init_output(fl4, oif, sk ? sk->sk_mark : 0, tos,
|
||||||
RT_SCOPE_UNIVERSE, proto,
|
RT_SCOPE_UNIVERSE, proto,
|
||||||
sk ? inet_sk_flowi_flags(sk) : 0,
|
sk ? inet_sk_flowi_flags(sk) : 0,
|
||||||
daddr, saddr, dport, sport);
|
daddr, saddr, dport, sport, sock_net_uid(net, sk));
|
||||||
if (sk)
|
if (sk)
|
||||||
security_sk_classify_flow(sk, flowi4_to_flowi(fl4));
|
security_sk_classify_flow(sk, flowi4_to_flowi(fl4));
|
||||||
return ip_route_output_flow(net, fl4, sk);
|
return ip_route_output_flow(net, fl4, sk);
|
||||||
|
@ -250,7 +250,8 @@ static inline void ip_route_connect_init(struct flowi4 *fl4, __be32 dst, __be32
|
||||||
flow_flags |= FLOWI_FLAG_CAN_SLEEP;
|
flow_flags |= FLOWI_FLAG_CAN_SLEEP;
|
||||||
|
|
||||||
flowi4_init_output(fl4, oif, sk->sk_mark, tos, RT_SCOPE_UNIVERSE,
|
flowi4_init_output(fl4, oif, sk->sk_mark, tos, RT_SCOPE_UNIVERSE,
|
||||||
protocol, flow_flags, dst, src, dport, sport);
|
protocol, flow_flags, dst, src, dport, sport,
|
||||||
|
sk->sk_uid);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct rtable *ip_route_connect(struct flowi4 *fl4,
|
static inline struct rtable *ip_route_connect(struct flowi4 *fl4,
|
||||||
|
|
|
@ -98,7 +98,7 @@ void mem_cgroup_sockets_destroy(struct cgroup *cgrp)
|
||||||
#else
|
#else
|
||||||
/* Validate arguments and do nothing */
|
/* Validate arguments and do nothing */
|
||||||
static inline __printf(2, 3)
|
static inline __printf(2, 3)
|
||||||
void SOCK_DEBUG(struct sock *sk, const char *msg, ...)
|
void SOCK_DEBUG(const struct sock *sk, const char *msg, ...)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -374,13 +374,14 @@ struct sock {
|
||||||
void *sk_security;
|
void *sk_security;
|
||||||
#endif
|
#endif
|
||||||
__u32 sk_mark;
|
__u32 sk_mark;
|
||||||
|
kuid_t sk_uid;
|
||||||
u32 sk_classid;
|
u32 sk_classid;
|
||||||
struct cg_proto *sk_cgrp;
|
struct cg_proto *sk_cgrp;
|
||||||
void (*sk_state_change)(struct sock *sk);
|
void (*sk_state_change)(struct sock *sk);
|
||||||
void (*sk_data_ready)(struct sock *sk, int bytes);
|
void (*sk_data_ready)(struct sock *sk, int bytes);
|
||||||
void (*sk_write_space)(struct sock *sk);
|
void (*sk_write_space)(struct sock *sk);
|
||||||
void (*sk_error_report)(struct sock *sk);
|
void (*sk_error_report)(struct sock *sk);
|
||||||
int (*sk_backlog_rcv)(struct sock *sk,
|
int (*sk_backlog_rcv)(struct sock *sk,
|
||||||
struct sk_buff *skb);
|
struct sk_buff *skb);
|
||||||
void (*sk_destruct)(struct sock *sk);
|
void (*sk_destruct)(struct sock *sk);
|
||||||
};
|
};
|
||||||
|
@ -451,40 +452,40 @@ static inline struct sock *sk_nulls_next(const struct sock *sk)
|
||||||
NULL;
|
NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int sk_unhashed(const struct sock *sk)
|
static inline bool sk_unhashed(const struct sock *sk)
|
||||||
{
|
{
|
||||||
return hlist_unhashed(&sk->sk_node);
|
return hlist_unhashed(&sk->sk_node);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int sk_hashed(const struct sock *sk)
|
static inline bool sk_hashed(const struct sock *sk)
|
||||||
{
|
{
|
||||||
return !sk_unhashed(sk);
|
return !sk_unhashed(sk);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void sk_node_init(struct hlist_node *node)
|
static inline void sk_node_init(struct hlist_node *node)
|
||||||
{
|
{
|
||||||
node->pprev = NULL;
|
node->pprev = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void sk_nulls_node_init(struct hlist_nulls_node *node)
|
static inline void sk_nulls_node_init(struct hlist_nulls_node *node)
|
||||||
{
|
{
|
||||||
node->pprev = NULL;
|
node->pprev = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void __sk_del_node(struct sock *sk)
|
static inline void __sk_del_node(struct sock *sk)
|
||||||
{
|
{
|
||||||
__hlist_del(&sk->sk_node);
|
__hlist_del(&sk->sk_node);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* NB: equivalent to hlist_del_init_rcu */
|
/* NB: equivalent to hlist_del_init_rcu */
|
||||||
static __inline__ int __sk_del_node_init(struct sock *sk)
|
static inline bool __sk_del_node_init(struct sock *sk)
|
||||||
{
|
{
|
||||||
if (sk_hashed(sk)) {
|
if (sk_hashed(sk)) {
|
||||||
__sk_del_node(sk);
|
__sk_del_node(sk);
|
||||||
sk_node_init(&sk->sk_node);
|
sk_node_init(&sk->sk_node);
|
||||||
return 1;
|
return true;
|
||||||
}
|
}
|
||||||
return 0;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Grab socket reference count. This operation is valid only
|
/* Grab socket reference count. This operation is valid only
|
||||||
|
@ -506,9 +507,9 @@ static inline void __sock_put(struct sock *sk)
|
||||||
atomic_dec(&sk->sk_refcnt);
|
atomic_dec(&sk->sk_refcnt);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ int sk_del_node_init(struct sock *sk)
|
static inline bool sk_del_node_init(struct sock *sk)
|
||||||
{
|
{
|
||||||
int rc = __sk_del_node_init(sk);
|
bool rc = __sk_del_node_init(sk);
|
||||||
|
|
||||||
if (rc) {
|
if (rc) {
|
||||||
/* paranoid for a while -acme */
|
/* paranoid for a while -acme */
|
||||||
|
@ -519,18 +520,18 @@ static __inline__ int sk_del_node_init(struct sock *sk)
|
||||||
}
|
}
|
||||||
#define sk_del_node_init_rcu(sk) sk_del_node_init(sk)
|
#define sk_del_node_init_rcu(sk) sk_del_node_init(sk)
|
||||||
|
|
||||||
static __inline__ int __sk_nulls_del_node_init_rcu(struct sock *sk)
|
static inline bool __sk_nulls_del_node_init_rcu(struct sock *sk)
|
||||||
{
|
{
|
||||||
if (sk_hashed(sk)) {
|
if (sk_hashed(sk)) {
|
||||||
hlist_nulls_del_init_rcu(&sk->sk_nulls_node);
|
hlist_nulls_del_init_rcu(&sk->sk_nulls_node);
|
||||||
return 1;
|
return true;
|
||||||
}
|
}
|
||||||
return 0;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ int sk_nulls_del_node_init_rcu(struct sock *sk)
|
static inline bool sk_nulls_del_node_init_rcu(struct sock *sk)
|
||||||
{
|
{
|
||||||
int rc = __sk_nulls_del_node_init_rcu(sk);
|
bool rc = __sk_nulls_del_node_init_rcu(sk);
|
||||||
|
|
||||||
if (rc) {
|
if (rc) {
|
||||||
/* paranoid for a while -acme */
|
/* paranoid for a while -acme */
|
||||||
|
@ -540,40 +541,40 @@ static __inline__ int sk_nulls_del_node_init_rcu(struct sock *sk)
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void __sk_add_node(struct sock *sk, struct hlist_head *list)
|
static inline void __sk_add_node(struct sock *sk, struct hlist_head *list)
|
||||||
{
|
{
|
||||||
hlist_add_head(&sk->sk_node, list);
|
hlist_add_head(&sk->sk_node, list);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void sk_add_node(struct sock *sk, struct hlist_head *list)
|
static inline void sk_add_node(struct sock *sk, struct hlist_head *list)
|
||||||
{
|
{
|
||||||
sock_hold(sk);
|
sock_hold(sk);
|
||||||
__sk_add_node(sk, list);
|
__sk_add_node(sk, list);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void sk_add_node_rcu(struct sock *sk, struct hlist_head *list)
|
static inline void sk_add_node_rcu(struct sock *sk, struct hlist_head *list)
|
||||||
{
|
{
|
||||||
sock_hold(sk);
|
sock_hold(sk);
|
||||||
hlist_add_head_rcu(&sk->sk_node, list);
|
hlist_add_head_rcu(&sk->sk_node, list);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void __sk_nulls_add_node_rcu(struct sock *sk, struct hlist_nulls_head *list)
|
static inline void __sk_nulls_add_node_rcu(struct sock *sk, struct hlist_nulls_head *list)
|
||||||
{
|
{
|
||||||
hlist_nulls_add_head_rcu(&sk->sk_nulls_node, list);
|
hlist_nulls_add_head_rcu(&sk->sk_nulls_node, list);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void sk_nulls_add_node_rcu(struct sock *sk, struct hlist_nulls_head *list)
|
static inline void sk_nulls_add_node_rcu(struct sock *sk, struct hlist_nulls_head *list)
|
||||||
{
|
{
|
||||||
sock_hold(sk);
|
sock_hold(sk);
|
||||||
__sk_nulls_add_node_rcu(sk, list);
|
__sk_nulls_add_node_rcu(sk, list);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void __sk_del_bind_node(struct sock *sk)
|
static inline void __sk_del_bind_node(struct sock *sk)
|
||||||
{
|
{
|
||||||
__hlist_del(&sk->sk_bind_node);
|
__hlist_del(&sk->sk_bind_node);
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline__ void sk_add_bind_node(struct sock *sk,
|
static inline void sk_add_bind_node(struct sock *sk,
|
||||||
struct hlist_head *list)
|
struct hlist_head *list)
|
||||||
{
|
{
|
||||||
hlist_add_head(&sk->sk_bind_node, list);
|
hlist_add_head(&sk->sk_bind_node, list);
|
||||||
|
@ -662,7 +663,7 @@ static inline void sk_acceptq_added(struct sock *sk)
|
||||||
sk->sk_ack_backlog++;
|
sk->sk_ack_backlog++;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int sk_acceptq_is_full(struct sock *sk)
|
static inline bool sk_acceptq_is_full(const struct sock *sk)
|
||||||
{
|
{
|
||||||
return sk->sk_ack_backlog > sk->sk_max_ack_backlog;
|
return sk->sk_ack_backlog > sk->sk_max_ack_backlog;
|
||||||
}
|
}
|
||||||
|
@ -670,19 +671,19 @@ static inline int sk_acceptq_is_full(struct sock *sk)
|
||||||
/*
|
/*
|
||||||
* Compute minimal free write space needed to queue new packets.
|
* Compute minimal free write space needed to queue new packets.
|
||||||
*/
|
*/
|
||||||
static inline int sk_stream_min_wspace(struct sock *sk)
|
static inline int sk_stream_min_wspace(const struct sock *sk)
|
||||||
{
|
{
|
||||||
return sk->sk_wmem_queued >> 1;
|
return sk->sk_wmem_queued >> 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int sk_stream_wspace(struct sock *sk)
|
static inline int sk_stream_wspace(const struct sock *sk)
|
||||||
{
|
{
|
||||||
return sk->sk_sndbuf - sk->sk_wmem_queued;
|
return sk->sk_sndbuf - sk->sk_wmem_queued;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void sk_stream_write_space(struct sock *sk);
|
void sk_stream_write_space(struct sock *sk);
|
||||||
|
|
||||||
static inline int sk_stream_memory_free(struct sock *sk)
|
static inline bool sk_stream_memory_free(const struct sock *sk)
|
||||||
{
|
{
|
||||||
return sk->sk_wmem_queued < sk->sk_sndbuf;
|
return sk->sk_wmem_queued < sk->sk_sndbuf;
|
||||||
}
|
}
|
||||||
|
@ -785,13 +786,13 @@ static inline void sock_rps_reset_rxhash(struct sock *sk)
|
||||||
__rc; \
|
__rc; \
|
||||||
})
|
})
|
||||||
|
|
||||||
extern int sk_stream_wait_connect(struct sock *sk, long *timeo_p);
|
int sk_stream_wait_connect(struct sock *sk, long *timeo_p);
|
||||||
extern int sk_stream_wait_memory(struct sock *sk, long *timeo_p);
|
int sk_stream_wait_memory(struct sock *sk, long *timeo_p);
|
||||||
extern void sk_stream_wait_close(struct sock *sk, long timeo_p);
|
void sk_stream_wait_close(struct sock *sk, long timeo_p);
|
||||||
extern int sk_stream_error(struct sock *sk, int flags, int err);
|
int sk_stream_error(struct sock *sk, int flags, int err);
|
||||||
extern void sk_stream_kill_queues(struct sock *sk);
|
void sk_stream_kill_queues(struct sock *sk);
|
||||||
|
|
||||||
extern int sk_wait_data(struct sock *sk, long *timeo);
|
int sk_wait_data(struct sock *sk, long *timeo, const struct sk_buff *skb);
|
||||||
|
|
||||||
struct request_sock_ops;
|
struct request_sock_ops;
|
||||||
struct timewait_sock_ops;
|
struct timewait_sock_ops;
|
||||||
|
@ -819,11 +820,11 @@ struct proto {
|
||||||
void (*close)(struct sock *sk,
|
void (*close)(struct sock *sk,
|
||||||
long timeout);
|
long timeout);
|
||||||
int (*connect)(struct sock *sk,
|
int (*connect)(struct sock *sk,
|
||||||
struct sockaddr *uaddr,
|
struct sockaddr *uaddr,
|
||||||
int addr_len);
|
int addr_len);
|
||||||
int (*disconnect)(struct sock *sk, int flags);
|
int (*disconnect)(struct sock *sk, int flags);
|
||||||
|
|
||||||
struct sock * (*accept) (struct sock *sk, int flags, int *err);
|
struct sock * (*accept)(struct sock *sk, int flags, int *err);
|
||||||
|
|
||||||
int (*ioctl)(struct sock *sk, int cmd,
|
int (*ioctl)(struct sock *sk, int cmd,
|
||||||
unsigned long arg);
|
unsigned long arg);
|
||||||
|
@ -852,8 +853,8 @@ struct proto {
|
||||||
struct msghdr *msg, size_t len);
|
struct msghdr *msg, size_t len);
|
||||||
int (*recvmsg)(struct kiocb *iocb, struct sock *sk,
|
int (*recvmsg)(struct kiocb *iocb, struct sock *sk,
|
||||||
struct msghdr *msg,
|
struct msghdr *msg,
|
||||||
size_t len, int noblock, int flags,
|
size_t len, int noblock, int flags,
|
||||||
int *addr_len);
|
int *addr_len);
|
||||||
int (*sendpage)(struct sock *sk, struct page *page,
|
int (*sendpage)(struct sock *sk, struct page *page,
|
||||||
int offset, size_t size, int flags);
|
int offset, size_t size, int flags);
|
||||||
int (*bind)(struct sock *sk,
|
int (*bind)(struct sock *sk,
|
||||||
|
@ -949,8 +950,8 @@ struct cg_proto {
|
||||||
struct mem_cgroup *memcg;
|
struct mem_cgroup *memcg;
|
||||||
};
|
};
|
||||||
|
|
||||||
extern int proto_register(struct proto *prot, int alloc_slab);
|
int proto_register(struct proto *prot, int alloc_slab);
|
||||||
extern void proto_unregister(struct proto *prot);
|
void proto_unregister(struct proto *prot);
|
||||||
|
|
||||||
#ifdef SOCK_REFCNT_DEBUG
|
#ifdef SOCK_REFCNT_DEBUG
|
||||||
static inline void sk_refcnt_debug_inc(struct sock *sk)
|
static inline void sk_refcnt_debug_inc(struct sock *sk)
|
||||||
|
@ -1180,10 +1181,10 @@ proto_memory_pressure(struct proto *prot)
|
||||||
|
|
||||||
#ifdef CONFIG_PROC_FS
|
#ifdef CONFIG_PROC_FS
|
||||||
/* Called with local bh disabled */
|
/* Called with local bh disabled */
|
||||||
extern void sock_prot_inuse_add(struct net *net, struct proto *prot, int inc);
|
void sock_prot_inuse_add(struct net *net, struct proto *prot, int inc);
|
||||||
extern int sock_prot_inuse_get(struct net *net, struct proto *proto);
|
int sock_prot_inuse_get(struct net *net, struct proto *proto);
|
||||||
#else
|
#else
|
||||||
static void inline sock_prot_inuse_add(struct net *net, struct proto *prot,
|
static inline void sock_prot_inuse_add(struct net *net, struct proto *prot,
|
||||||
int inc)
|
int inc)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -1257,8 +1258,8 @@ static inline struct inode *SOCK_INODE(struct socket *socket)
|
||||||
/*
|
/*
|
||||||
* Functions for memory accounting
|
* Functions for memory accounting
|
||||||
*/
|
*/
|
||||||
extern int __sk_mem_schedule(struct sock *sk, int size, int kind);
|
int __sk_mem_schedule(struct sock *sk, int size, int kind);
|
||||||
extern void __sk_mem_reclaim(struct sock *sk);
|
void __sk_mem_reclaim(struct sock *sk);
|
||||||
|
|
||||||
#define SK_MEM_QUANTUM ((int)PAGE_SIZE)
|
#define SK_MEM_QUANTUM ((int)PAGE_SIZE)
|
||||||
#define SK_MEM_QUANTUM_SHIFT ilog2(SK_MEM_QUANTUM)
|
#define SK_MEM_QUANTUM_SHIFT ilog2(SK_MEM_QUANTUM)
|
||||||
|
@ -1270,24 +1271,24 @@ static inline int sk_mem_pages(int amt)
|
||||||
return (amt + SK_MEM_QUANTUM - 1) >> SK_MEM_QUANTUM_SHIFT;
|
return (amt + SK_MEM_QUANTUM - 1) >> SK_MEM_QUANTUM_SHIFT;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int sk_has_account(struct sock *sk)
|
static inline bool sk_has_account(struct sock *sk)
|
||||||
{
|
{
|
||||||
/* return true if protocol supports memory accounting */
|
/* return true if protocol supports memory accounting */
|
||||||
return !!sk->sk_prot->memory_allocated;
|
return !!sk->sk_prot->memory_allocated;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int sk_wmem_schedule(struct sock *sk, int size)
|
static inline bool sk_wmem_schedule(struct sock *sk, int size)
|
||||||
{
|
{
|
||||||
if (!sk_has_account(sk))
|
if (!sk_has_account(sk))
|
||||||
return 1;
|
return true;
|
||||||
return size <= sk->sk_forward_alloc ||
|
return size <= sk->sk_forward_alloc ||
|
||||||
__sk_mem_schedule(sk, size, SK_MEM_SEND);
|
__sk_mem_schedule(sk, size, SK_MEM_SEND);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int sk_rmem_schedule(struct sock *sk, int size)
|
static inline bool sk_rmem_schedule(struct sock *sk, int size)
|
||||||
{
|
{
|
||||||
if (!sk_has_account(sk))
|
if (!sk_has_account(sk))
|
||||||
return 1;
|
return true;
|
||||||
return size <= sk->sk_forward_alloc ||
|
return size <= sk->sk_forward_alloc ||
|
||||||
__sk_mem_schedule(sk, size, SK_MEM_RECV);
|
__sk_mem_schedule(sk, size, SK_MEM_RECV);
|
||||||
}
|
}
|
||||||
|
@ -1352,7 +1353,7 @@ static inline void sk_wmem_free_skb(struct sock *sk, struct sk_buff *skb)
|
||||||
* Mark both the sk_lock and the sk_lock.slock as a
|
* Mark both the sk_lock and the sk_lock.slock as a
|
||||||
* per-address-family lock class.
|
* per-address-family lock class.
|
||||||
*/
|
*/
|
||||||
#define sock_lock_init_class_and_name(sk, sname, skey, name, key) \
|
#define sock_lock_init_class_and_name(sk, sname, skey, name, key) \
|
||||||
do { \
|
do { \
|
||||||
sk->sk_lock.owned = 0; \
|
sk->sk_lock.owned = 0; \
|
||||||
init_waitqueue_head(&sk->sk_lock.wq); \
|
init_waitqueue_head(&sk->sk_lock.wq); \
|
||||||
|
@ -1360,18 +1361,18 @@ do { \
|
||||||
debug_check_no_locks_freed((void *)&(sk)->sk_lock, \
|
debug_check_no_locks_freed((void *)&(sk)->sk_lock, \
|
||||||
sizeof((sk)->sk_lock)); \
|
sizeof((sk)->sk_lock)); \
|
||||||
lockdep_set_class_and_name(&(sk)->sk_lock.slock, \
|
lockdep_set_class_and_name(&(sk)->sk_lock.slock, \
|
||||||
(skey), (sname)); \
|
(skey), (sname)); \
|
||||||
lockdep_init_map(&(sk)->sk_lock.dep_map, (name), (key), 0); \
|
lockdep_init_map(&(sk)->sk_lock.dep_map, (name), (key), 0); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
extern void lock_sock_nested(struct sock *sk, int subclass);
|
void lock_sock_nested(struct sock *sk, int subclass);
|
||||||
|
|
||||||
static inline void lock_sock(struct sock *sk)
|
static inline void lock_sock(struct sock *sk)
|
||||||
{
|
{
|
||||||
lock_sock_nested(sk, 0);
|
lock_sock_nested(sk, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void release_sock(struct sock *sk);
|
void release_sock(struct sock *sk);
|
||||||
|
|
||||||
/* BH context may only use the following locking interface. */
|
/* BH context may only use the following locking interface. */
|
||||||
#define bh_lock_sock(__sk) spin_lock(&((__sk)->sk_lock.slock))
|
#define bh_lock_sock(__sk) spin_lock(&((__sk)->sk_lock.slock))
|
||||||
|
@ -1380,7 +1381,7 @@ extern void release_sock(struct sock *sk);
|
||||||
SINGLE_DEPTH_NESTING)
|
SINGLE_DEPTH_NESTING)
|
||||||
#define bh_unlock_sock(__sk) spin_unlock(&((__sk)->sk_lock.slock))
|
#define bh_unlock_sock(__sk) spin_unlock(&((__sk)->sk_lock.slock))
|
||||||
|
|
||||||
extern bool lock_sock_fast(struct sock *sk);
|
bool lock_sock_fast(struct sock *sk);
|
||||||
/**
|
/**
|
||||||
* unlock_sock_fast - complement of lock_sock_fast
|
* unlock_sock_fast - complement of lock_sock_fast
|
||||||
* @sk: socket
|
* @sk: socket
|
||||||
|
@ -1398,46 +1399,35 @@ static inline void unlock_sock_fast(struct sock *sk, bool slow)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
extern struct sock *sk_alloc(struct net *net, int family,
|
struct sock *sk_alloc(struct net *net, int family, gfp_t priority,
|
||||||
gfp_t priority,
|
struct proto *prot);
|
||||||
struct proto *prot);
|
void sk_free(struct sock *sk);
|
||||||
extern void sk_free(struct sock *sk);
|
void sk_release_kernel(struct sock *sk);
|
||||||
extern void sk_release_kernel(struct sock *sk);
|
struct sock *sk_clone_lock(const struct sock *sk, const gfp_t priority);
|
||||||
extern struct sock *sk_clone_lock(const struct sock *sk,
|
|
||||||
const gfp_t priority);
|
|
||||||
|
|
||||||
extern struct sk_buff *sock_wmalloc(struct sock *sk,
|
struct sk_buff *sock_wmalloc(struct sock *sk, unsigned long size, int force,
|
||||||
unsigned long size, int force,
|
gfp_t priority);
|
||||||
gfp_t priority);
|
struct sk_buff *sock_rmalloc(struct sock *sk, unsigned long size, int force,
|
||||||
extern struct sk_buff *sock_rmalloc(struct sock *sk,
|
gfp_t priority);
|
||||||
unsigned long size, int force,
|
void sock_wfree(struct sk_buff *skb);
|
||||||
gfp_t priority);
|
void sock_rfree(struct sk_buff *skb);
|
||||||
extern void sock_wfree(struct sk_buff *skb);
|
|
||||||
extern void sock_rfree(struct sk_buff *skb);
|
|
||||||
|
|
||||||
extern int sock_setsockopt(struct socket *sock, int level,
|
int sock_setsockopt(struct socket *sock, int level, int op,
|
||||||
int op, char __user *optval,
|
char __user *optval, unsigned int optlen);
|
||||||
unsigned int optlen);
|
|
||||||
|
|
||||||
extern int sock_getsockopt(struct socket *sock, int level,
|
int sock_getsockopt(struct socket *sock, int level, int op,
|
||||||
int op, char __user *optval,
|
char __user *optval, int __user *optlen);
|
||||||
int __user *optlen);
|
struct sk_buff *sock_alloc_send_skb(struct sock *sk, unsigned long size,
|
||||||
extern struct sk_buff *sock_alloc_send_skb(struct sock *sk,
|
int noblock, int *errcode);
|
||||||
unsigned long size,
|
struct sk_buff *sock_alloc_send_pskb(struct sock *sk, unsigned long header_len,
|
||||||
int noblock,
|
unsigned long data_len, int noblock,
|
||||||
int *errcode);
|
int *errcode);
|
||||||
extern struct sk_buff *sock_alloc_send_pskb(struct sock *sk,
|
void *sock_kmalloc(struct sock *sk, int size, gfp_t priority);
|
||||||
unsigned long header_len,
|
void sock_kfree_s(struct sock *sk, void *mem, int size);
|
||||||
unsigned long data_len,
|
void sk_send_sigurg(struct sock *sk);
|
||||||
int noblock,
|
|
||||||
int *errcode);
|
|
||||||
extern void *sock_kmalloc(struct sock *sk, int size,
|
|
||||||
gfp_t priority);
|
|
||||||
extern void sock_kfree_s(struct sock *sk, void *mem, int size);
|
|
||||||
extern void sk_send_sigurg(struct sock *sk);
|
|
||||||
|
|
||||||
#ifdef CONFIG_CGROUPS
|
#ifdef CONFIG_CGROUPS
|
||||||
extern void sock_update_classid(struct sock *sk);
|
void sock_update_classid(struct sock *sk);
|
||||||
#else
|
#else
|
||||||
static inline void sock_update_classid(struct sock *sk)
|
static inline void sock_update_classid(struct sock *sk)
|
||||||
{
|
{
|
||||||
|
@ -1448,63 +1438,51 @@ static inline void sock_update_classid(struct sock *sk)
|
||||||
* Functions to fill in entries in struct proto_ops when a protocol
|
* Functions to fill in entries in struct proto_ops when a protocol
|
||||||
* does not implement a particular function.
|
* does not implement a particular function.
|
||||||
*/
|
*/
|
||||||
extern int sock_no_bind(struct socket *,
|
int sock_no_bind(struct socket *, struct sockaddr *, int);
|
||||||
struct sockaddr *, int);
|
int sock_no_connect(struct socket *, struct sockaddr *, int, int);
|
||||||
extern int sock_no_connect(struct socket *,
|
int sock_no_socketpair(struct socket *, struct socket *);
|
||||||
struct sockaddr *, int, int);
|
int sock_no_accept(struct socket *, struct socket *, int);
|
||||||
extern int sock_no_socketpair(struct socket *,
|
int sock_no_getname(struct socket *, struct sockaddr *, int *, int);
|
||||||
struct socket *);
|
unsigned int sock_no_poll(struct file *, struct socket *,
|
||||||
extern int sock_no_accept(struct socket *,
|
struct poll_table_struct *);
|
||||||
struct socket *, int);
|
int sock_no_ioctl(struct socket *, unsigned int, unsigned long);
|
||||||
extern int sock_no_getname(struct socket *,
|
int sock_no_listen(struct socket *, int);
|
||||||
struct sockaddr *, int *, int);
|
int sock_no_shutdown(struct socket *, int);
|
||||||
extern unsigned int sock_no_poll(struct file *, struct socket *,
|
int sock_no_getsockopt(struct socket *, int , int, char __user *, int __user *);
|
||||||
struct poll_table_struct *);
|
int sock_no_setsockopt(struct socket *, int, int, char __user *, unsigned int);
|
||||||
extern int sock_no_ioctl(struct socket *, unsigned int,
|
int sock_no_sendmsg(struct kiocb *, struct socket *, struct msghdr *, size_t);
|
||||||
unsigned long);
|
int sock_no_recvmsg(struct kiocb *, struct socket *, struct msghdr *, size_t,
|
||||||
extern int sock_no_listen(struct socket *, int);
|
int);
|
||||||
extern int sock_no_shutdown(struct socket *, int);
|
int sock_no_mmap(struct file *file, struct socket *sock,
|
||||||
extern int sock_no_getsockopt(struct socket *, int , int,
|
struct vm_area_struct *vma);
|
||||||
char __user *, int __user *);
|
ssize_t sock_no_sendpage(struct socket *sock, struct page *page, int offset,
|
||||||
extern int sock_no_setsockopt(struct socket *, int, int,
|
size_t size, int flags);
|
||||||
char __user *, unsigned int);
|
|
||||||
extern int sock_no_sendmsg(struct kiocb *, struct socket *,
|
|
||||||
struct msghdr *, size_t);
|
|
||||||
extern int sock_no_recvmsg(struct kiocb *, struct socket *,
|
|
||||||
struct msghdr *, size_t, int);
|
|
||||||
extern int sock_no_mmap(struct file *file,
|
|
||||||
struct socket *sock,
|
|
||||||
struct vm_area_struct *vma);
|
|
||||||
extern ssize_t sock_no_sendpage(struct socket *sock,
|
|
||||||
struct page *page,
|
|
||||||
int offset, size_t size,
|
|
||||||
int flags);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Functions to fill in entries in struct proto_ops when a protocol
|
* Functions to fill in entries in struct proto_ops when a protocol
|
||||||
* uses the inet style.
|
* uses the inet style.
|
||||||
*/
|
*/
|
||||||
extern int sock_common_getsockopt(struct socket *sock, int level, int optname,
|
int sock_common_getsockopt(struct socket *sock, int level, int optname,
|
||||||
char __user *optval, int __user *optlen);
|
char __user *optval, int __user *optlen);
|
||||||
extern int sock_common_recvmsg(struct kiocb *iocb, struct socket *sock,
|
int sock_common_recvmsg(struct kiocb *iocb, struct socket *sock,
|
||||||
struct msghdr *msg, size_t size, int flags);
|
struct msghdr *msg, size_t size, int flags);
|
||||||
extern int sock_common_setsockopt(struct socket *sock, int level, int optname,
|
int sock_common_setsockopt(struct socket *sock, int level, int optname,
|
||||||
char __user *optval, unsigned int optlen);
|
char __user *optval, unsigned int optlen);
|
||||||
extern int compat_sock_common_getsockopt(struct socket *sock, int level,
|
int compat_sock_common_getsockopt(struct socket *sock, int level,
|
||||||
int optname, char __user *optval, int __user *optlen);
|
int optname, char __user *optval, int __user *optlen);
|
||||||
extern int compat_sock_common_setsockopt(struct socket *sock, int level,
|
int compat_sock_common_setsockopt(struct socket *sock, int level,
|
||||||
int optname, char __user *optval, unsigned int optlen);
|
int optname, char __user *optval, unsigned int optlen);
|
||||||
|
|
||||||
extern void sk_common_release(struct sock *sk);
|
void sk_common_release(struct sock *sk);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Default socket callbacks and setup code
|
* Default socket callbacks and setup code
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* Initialise core socket variables */
|
/* Initialise core socket variables */
|
||||||
extern void sock_init_data(struct socket *sock, struct sock *sk);
|
void sock_init_data(struct socket *sock, struct sock *sk);
|
||||||
|
|
||||||
extern void sk_filter_release_rcu(struct rcu_head *rcu);
|
void sk_filter_release_rcu(struct rcu_head *rcu);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* sk_filter_release - release a socket filter
|
* sk_filter_release - release a socket filter
|
||||||
|
@ -1565,8 +1543,7 @@ static inline void sock_put(struct sock *sk)
|
||||||
sk_free(sk);
|
sk_free(sk);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern int sk_receive_skb(struct sock *sk, struct sk_buff *skb,
|
int sk_receive_skb(struct sock *sk, struct sk_buff *skb, const int nested);
|
||||||
const int nested);
|
|
||||||
|
|
||||||
static inline void sk_tx_queue_set(struct sock *sk, int tx_queue)
|
static inline void sk_tx_queue_set(struct sock *sk, int tx_queue)
|
||||||
{
|
{
|
||||||
|
@ -1616,12 +1593,18 @@ static inline void sock_graft(struct sock *sk, struct socket *parent)
|
||||||
sk->sk_wq = parent->wq;
|
sk->sk_wq = parent->wq;
|
||||||
parent->sk = sk;
|
parent->sk = sk;
|
||||||
sk_set_socket(sk, parent);
|
sk_set_socket(sk, parent);
|
||||||
|
sk->sk_uid = SOCK_INODE(parent)->i_uid;
|
||||||
security_sock_graft(sk, parent);
|
security_sock_graft(sk, parent);
|
||||||
write_unlock_bh(&sk->sk_callback_lock);
|
write_unlock_bh(&sk->sk_callback_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern int sock_i_uid(struct sock *sk);
|
int sock_i_uid(struct sock *sk);
|
||||||
extern unsigned long sock_i_ino(struct sock *sk);
|
unsigned long sock_i_ino(struct sock *sk);
|
||||||
|
|
||||||
|
static inline kuid_t sock_net_uid(const struct net *net, const struct sock *sk)
|
||||||
|
{
|
||||||
|
return sk ? sk->sk_uid : make_kuid(net->user_ns, 0);
|
||||||
|
}
|
||||||
|
|
||||||
static inline struct dst_entry *
|
static inline struct dst_entry *
|
||||||
__sk_dst_get(struct sock *sk)
|
__sk_dst_get(struct sock *sk)
|
||||||
|
@ -1643,7 +1626,7 @@ sk_dst_get(struct sock *sk)
|
||||||
return dst;
|
return dst;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void sk_reset_txq(struct sock *sk);
|
void sk_reset_txq(struct sock *sk);
|
||||||
|
|
||||||
static inline void dst_negative_advice(struct sock *sk)
|
static inline void dst_negative_advice(struct sock *sk)
|
||||||
{
|
{
|
||||||
|
@ -1696,16 +1679,16 @@ sk_dst_reset(struct sock *sk)
|
||||||
spin_unlock(&sk->sk_dst_lock);
|
spin_unlock(&sk->sk_dst_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern struct dst_entry *__sk_dst_check(struct sock *sk, u32 cookie);
|
struct dst_entry *__sk_dst_check(struct sock *sk, u32 cookie);
|
||||||
|
|
||||||
extern struct dst_entry *sk_dst_check(struct sock *sk, u32 cookie);
|
struct dst_entry *sk_dst_check(struct sock *sk, u32 cookie);
|
||||||
|
|
||||||
static inline int sk_can_gso(const struct sock *sk)
|
static inline bool sk_can_gso(const struct sock *sk)
|
||||||
{
|
{
|
||||||
return net_gso_ok(sk->sk_route_caps, sk->sk_gso_type);
|
return net_gso_ok(sk->sk_route_caps, sk->sk_gso_type);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void sk_setup_caps(struct sock *sk, struct dst_entry *dst);
|
void sk_setup_caps(struct sock *sk, struct dst_entry *dst);
|
||||||
|
|
||||||
static inline void sk_nocaps_add(struct sock *sk, netdev_features_t flags)
|
static inline void sk_nocaps_add(struct sock *sk, netdev_features_t flags)
|
||||||
{
|
{
|
||||||
|
@ -1817,7 +1800,7 @@ static inline int sk_rmem_alloc_get(const struct sock *sk)
|
||||||
*
|
*
|
||||||
* Returns true if socket has write or read allocations
|
* Returns true if socket has write or read allocations
|
||||||
*/
|
*/
|
||||||
static inline int sk_has_allocations(const struct sock *sk)
|
static inline bool sk_has_allocations(const struct sock *sk)
|
||||||
{
|
{
|
||||||
return sk_wmem_alloc_get(sk) || sk_rmem_alloc_get(sk);
|
return sk_wmem_alloc_get(sk) || sk_rmem_alloc_get(sk);
|
||||||
}
|
}
|
||||||
|
@ -1856,9 +1839,7 @@ static inline int sk_has_allocations(const struct sock *sk)
|
||||||
*/
|
*/
|
||||||
static inline bool wq_has_sleeper(struct socket_wq *wq)
|
static inline bool wq_has_sleeper(struct socket_wq *wq)
|
||||||
{
|
{
|
||||||
|
/* We need to be sure we are in sync with the
|
||||||
/*
|
|
||||||
* We need to be sure we are in sync with the
|
|
||||||
* add_wait_queue modifications to the wait queue.
|
* add_wait_queue modifications to the wait queue.
|
||||||
*
|
*
|
||||||
* This memory barrier is paired in the sock_poll_wait.
|
* This memory barrier is paired in the sock_poll_wait.
|
||||||
|
@ -1880,22 +1861,21 @@ static inline void sock_poll_wait(struct file *filp,
|
||||||
{
|
{
|
||||||
if (!poll_does_not_wait(p) && wait_address) {
|
if (!poll_does_not_wait(p) && wait_address) {
|
||||||
poll_wait(filp, wait_address, p);
|
poll_wait(filp, wait_address, p);
|
||||||
/*
|
/* We need to be sure we are in sync with the
|
||||||
* We need to be sure we are in sync with the
|
|
||||||
* socket flags modification.
|
* socket flags modification.
|
||||||
*
|
*
|
||||||
* This memory barrier is paired in the wq_has_sleeper.
|
* This memory barrier is paired in the wq_has_sleeper.
|
||||||
*/
|
*/
|
||||||
smp_mb();
|
smp_mb();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Queue a received datagram if it will fit. Stream and sequenced
|
* Queue a received datagram if it will fit. Stream and sequenced
|
||||||
* protocols can't normally use this as they need to fit buffers in
|
* protocols can't normally use this as they need to fit buffers in
|
||||||
* and play with them.
|
* and play with them.
|
||||||
*
|
*
|
||||||
* Inlined as it's very short and called for pretty much every
|
* Inlined as it's very short and called for pretty much every
|
||||||
* packet ever received.
|
* packet ever received.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@ -1921,14 +1901,14 @@ static inline void skb_set_owner_r(struct sk_buff *skb, struct sock *sk)
|
||||||
sk_mem_charge(sk, skb->truesize);
|
sk_mem_charge(sk, skb->truesize);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void sk_reset_timer(struct sock *sk, struct timer_list* timer,
|
void sk_reset_timer(struct sock *sk, struct timer_list *timer,
|
||||||
unsigned long expires);
|
unsigned long expires);
|
||||||
|
|
||||||
extern void sk_stop_timer(struct sock *sk, struct timer_list* timer);
|
void sk_stop_timer(struct sock *sk, struct timer_list *timer);
|
||||||
|
|
||||||
extern int sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb);
|
int sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb);
|
||||||
|
|
||||||
extern int sock_queue_err_skb(struct sock *sk, struct sk_buff *skb);
|
int sock_queue_err_skb(struct sock *sk, struct sk_buff *skb);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Recover an error report and clear atomically
|
* Recover an error report and clear atomically
|
||||||
|
@ -1993,7 +1973,7 @@ static inline struct page *sk_stream_alloc_page(struct sock *sk)
|
||||||
/*
|
/*
|
||||||
* Default write policy as shown to user space via poll/select/SIGIO
|
* Default write policy as shown to user space via poll/select/SIGIO
|
||||||
*/
|
*/
|
||||||
static inline int sock_writeable(const struct sock *sk)
|
static inline bool sock_writeable(const struct sock *sk)
|
||||||
{
|
{
|
||||||
return atomic_read(&sk->sk_wmem_alloc) < (sk->sk_sndbuf >> 1);
|
return atomic_read(&sk->sk_wmem_alloc) < (sk->sk_sndbuf >> 1);
|
||||||
}
|
}
|
||||||
|
@ -2003,12 +1983,12 @@ static inline gfp_t gfp_any(void)
|
||||||
return in_softirq() ? GFP_ATOMIC : GFP_KERNEL;
|
return in_softirq() ? GFP_ATOMIC : GFP_KERNEL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline long sock_rcvtimeo(const struct sock *sk, int noblock)
|
static inline long sock_rcvtimeo(const struct sock *sk, bool noblock)
|
||||||
{
|
{
|
||||||
return noblock ? 0 : sk->sk_rcvtimeo;
|
return noblock ? 0 : sk->sk_rcvtimeo;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline long sock_sndtimeo(const struct sock *sk, int noblock)
|
static inline long sock_sndtimeo(const struct sock *sk, bool noblock)
|
||||||
{
|
{
|
||||||
return noblock ? 0 : sk->sk_sndtimeo;
|
return noblock ? 0 : sk->sk_sndtimeo;
|
||||||
}
|
}
|
||||||
|
@ -2026,12 +2006,12 @@ static inline int sock_intr_errno(long timeo)
|
||||||
return timeo == MAX_SCHEDULE_TIMEOUT ? -ERESTARTSYS : -EINTR;
|
return timeo == MAX_SCHEDULE_TIMEOUT ? -ERESTARTSYS : -EINTR;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void __sock_recv_timestamp(struct msghdr *msg, struct sock *sk,
|
void __sock_recv_timestamp(struct msghdr *msg, struct sock *sk,
|
||||||
struct sk_buff *skb);
|
struct sk_buff *skb);
|
||||||
extern void __sock_recv_wifi_status(struct msghdr *msg, struct sock *sk,
|
void __sock_recv_wifi_status(struct msghdr *msg, struct sock *sk,
|
||||||
struct sk_buff *skb);
|
struct sk_buff *skb);
|
||||||
|
|
||||||
static __inline__ void
|
static inline void
|
||||||
sock_recv_timestamp(struct msghdr *msg, struct sock *sk, struct sk_buff *skb)
|
sock_recv_timestamp(struct msghdr *msg, struct sock *sk, struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
ktime_t kt = skb->tstamp;
|
ktime_t kt = skb->tstamp;
|
||||||
|
@ -2062,8 +2042,8 @@ sock_recv_timestamp(struct msghdr *msg, struct sock *sk, struct sk_buff *skb)
|
||||||
__sock_recv_wifi_status(msg, sk, skb);
|
__sock_recv_wifi_status(msg, sk, skb);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void __sock_recv_ts_and_drops(struct msghdr *msg, struct sock *sk,
|
void __sock_recv_ts_and_drops(struct msghdr *msg, struct sock *sk,
|
||||||
struct sk_buff *skb);
|
struct sk_buff *skb);
|
||||||
|
|
||||||
static inline void sock_recv_ts_and_drops(struct msghdr *msg, struct sock *sk,
|
static inline void sock_recv_ts_and_drops(struct msghdr *msg, struct sock *sk,
|
||||||
struct sk_buff *skb)
|
struct sk_buff *skb)
|
||||||
|
@ -2072,7 +2052,7 @@ static inline void sock_recv_ts_and_drops(struct msghdr *msg, struct sock *sk,
|
||||||
(1UL << SOCK_RCVTSTAMP) | \
|
(1UL << SOCK_RCVTSTAMP) | \
|
||||||
(1UL << SOCK_TIMESTAMPING_RX_SOFTWARE) | \
|
(1UL << SOCK_TIMESTAMPING_RX_SOFTWARE) | \
|
||||||
(1UL << SOCK_TIMESTAMPING_SOFTWARE) | \
|
(1UL << SOCK_TIMESTAMPING_SOFTWARE) | \
|
||||||
(1UL << SOCK_TIMESTAMPING_RAW_HARDWARE) | \
|
(1UL << SOCK_TIMESTAMPING_RAW_HARDWARE) | \
|
||||||
(1UL << SOCK_TIMESTAMPING_SYS_HARDWARE))
|
(1UL << SOCK_TIMESTAMPING_SYS_HARDWARE))
|
||||||
|
|
||||||
if (sk->sk_flags & FLAGS_TS_OR_DROPS)
|
if (sk->sk_flags & FLAGS_TS_OR_DROPS)
|
||||||
|
@ -2086,10 +2066,9 @@ static inline void sock_recv_ts_and_drops(struct msghdr *msg, struct sock *sk,
|
||||||
* @sk: socket sending this packet
|
* @sk: socket sending this packet
|
||||||
* @tx_flags: filled with instructions for time stamping
|
* @tx_flags: filled with instructions for time stamping
|
||||||
*
|
*
|
||||||
* Currently only depends on SOCK_TIMESTAMPING* flags. Returns error code if
|
* Currently only depends on SOCK_TIMESTAMPING* flags.
|
||||||
* parameters are invalid.
|
|
||||||
*/
|
*/
|
||||||
extern int sock_tx_timestamp(struct sock *sk, __u8 *tx_flags);
|
void sock_tx_timestamp(struct sock *sk, __u8 *tx_flags);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* sk_eat_skb - Release a skb if it is no longer needed
|
* sk_eat_skb - Release a skb if it is no longer needed
|
||||||
|
@ -2101,7 +2080,7 @@ extern int sock_tx_timestamp(struct sock *sk, __u8 *tx_flags);
|
||||||
* locked so that the sk_buff queue operation is ok.
|
* locked so that the sk_buff queue operation is ok.
|
||||||
*/
|
*/
|
||||||
#ifdef CONFIG_NET_DMA
|
#ifdef CONFIG_NET_DMA
|
||||||
static inline void sk_eat_skb(struct sock *sk, struct sk_buff *skb, int copied_early)
|
static inline void sk_eat_skb(struct sock *sk, struct sk_buff *skb, bool copied_early)
|
||||||
{
|
{
|
||||||
__skb_unlink(skb, &sk->sk_receive_queue);
|
__skb_unlink(skb, &sk->sk_receive_queue);
|
||||||
if (!copied_early)
|
if (!copied_early)
|
||||||
|
@ -2110,7 +2089,7 @@ static inline void sk_eat_skb(struct sock *sk, struct sk_buff *skb, int copied_e
|
||||||
__skb_queue_tail(&sk->sk_async_wait_queue, skb);
|
__skb_queue_tail(&sk->sk_async_wait_queue, skb);
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static inline void sk_eat_skb(struct sock *sk, struct sk_buff *skb, int copied_early)
|
static inline void sk_eat_skb(struct sock *sk, struct sk_buff *skb, bool copied_early)
|
||||||
{
|
{
|
||||||
__skb_unlink(skb, &sk->sk_receive_queue);
|
__skb_unlink(skb, &sk->sk_receive_queue);
|
||||||
__kfree_skb(skb);
|
__kfree_skb(skb);
|
||||||
|
@ -2162,9 +2141,9 @@ static inline bool sk_fullsock(const struct sock *sk)
|
||||||
return (1 << sk->sk_state) & ~(TCPF_TIME_WAIT);
|
return (1 << sk->sk_state) & ~(TCPF_TIME_WAIT);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void sock_enable_timestamp(struct sock *sk, int flag);
|
void sock_enable_timestamp(struct sock *sk, int flag);
|
||||||
extern int sock_get_timestamp(struct sock *, struct timeval __user *);
|
int sock_get_timestamp(struct sock *, struct timeval __user *);
|
||||||
extern int sock_get_timestampns(struct sock *, struct timespec __user *);
|
int sock_get_timestampns(struct sock *, struct timespec __user *);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Enable debug/info messages
|
* Enable debug/info messages
|
||||||
|
@ -2179,7 +2158,7 @@ extern int net_msg_warn;
|
||||||
extern __u32 sysctl_wmem_max;
|
extern __u32 sysctl_wmem_max;
|
||||||
extern __u32 sysctl_rmem_max;
|
extern __u32 sysctl_rmem_max;
|
||||||
|
|
||||||
extern void sk_init(void);
|
void sk_init(void);
|
||||||
|
|
||||||
extern int sysctl_optmem_max;
|
extern int sysctl_optmem_max;
|
||||||
|
|
||||||
|
|
|
@ -514,6 +514,10 @@ long compat_sys_msgctl(int first, int second, void __user *uptr)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef COMPAT_SHMLBA
|
||||||
|
#define COMPAT_SHMLBA SHMLBA
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_ARCH_WANT_OLD_COMPAT_IPC
|
#ifdef CONFIG_ARCH_WANT_OLD_COMPAT_IPC
|
||||||
long compat_sys_shmat(int first, int second, compat_uptr_t third, int version,
|
long compat_sys_shmat(int first, int second, compat_uptr_t third, int version,
|
||||||
void __user *uptr)
|
void __user *uptr)
|
||||||
|
@ -524,7 +528,7 @@ long compat_sys_shmat(int first, int second, compat_uptr_t third, int version,
|
||||||
|
|
||||||
if (version == 1)
|
if (version == 1)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
err = do_shmat(first, uptr, second, &raddr);
|
err = do_shmat(first, uptr, second, &raddr, COMPAT_SHMLBA);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return err;
|
return err;
|
||||||
uaddr = compat_ptr(third);
|
uaddr = compat_ptr(third);
|
||||||
|
@ -536,7 +540,7 @@ long compat_sys_shmat(int shmid, compat_uptr_t shmaddr, int shmflg)
|
||||||
unsigned long ret;
|
unsigned long ret;
|
||||||
long err;
|
long err;
|
||||||
|
|
||||||
err = do_shmat(shmid, compat_ptr(shmaddr), shmflg, &ret);
|
err = do_shmat(shmid, compat_ptr(shmaddr), shmflg, &ret, COMPAT_SHMLBA);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
force_successful_syscall_return();
|
force_successful_syscall_return();
|
||||||
|
|
11
ipc/shm.c
11
ipc/shm.c
|
@ -944,7 +944,8 @@ out:
|
||||||
* "raddr" thing points to kernel space, and there has to be a wrapper around
|
* "raddr" thing points to kernel space, and there has to be a wrapper around
|
||||||
* this.
|
* this.
|
||||||
*/
|
*/
|
||||||
long do_shmat(int shmid, char __user *shmaddr, int shmflg, ulong *raddr)
|
long do_shmat(int shmid, char __user *shmaddr, int shmflg, ulong *raddr,
|
||||||
|
unsigned long shmlba)
|
||||||
{
|
{
|
||||||
struct shmid_kernel *shp;
|
struct shmid_kernel *shp;
|
||||||
unsigned long addr;
|
unsigned long addr;
|
||||||
|
@ -964,14 +965,14 @@ long do_shmat(int shmid, char __user *shmaddr, int shmflg, ulong *raddr)
|
||||||
if (shmid < 0)
|
if (shmid < 0)
|
||||||
goto out;
|
goto out;
|
||||||
else if ((addr = (ulong)shmaddr)) {
|
else if ((addr = (ulong)shmaddr)) {
|
||||||
if (addr & (SHMLBA-1)) {
|
if (addr & (shmlba - 1)) {
|
||||||
/*
|
/*
|
||||||
* Round down to the nearest multiple of shmlba.
|
* Round down to the nearest multiple of shmlba.
|
||||||
* For sane do_mmap_pgoff() parameters, avoid
|
* For sane do_mmap_pgoff() parameters, avoid
|
||||||
* round downs that trigger nil-page and MAP_FIXED.
|
* round downs that trigger nil-page and MAP_FIXED.
|
||||||
*/
|
*/
|
||||||
if ((shmflg & SHM_RND) && addr >= SHMLBA)
|
if ((shmflg & SHM_RND) && addr >= shmlba)
|
||||||
addr &= ~(SHMLBA - 1);
|
addr &= ~(shmlba - 1);
|
||||||
else
|
else
|
||||||
#ifndef __ARCH_FORCE_SHMLBA
|
#ifndef __ARCH_FORCE_SHMLBA
|
||||||
if (addr & ~PAGE_MASK)
|
if (addr & ~PAGE_MASK)
|
||||||
|
@ -1098,7 +1099,7 @@ SYSCALL_DEFINE3(shmat, int, shmid, char __user *, shmaddr, int, shmflg)
|
||||||
unsigned long ret;
|
unsigned long ret;
|
||||||
long err;
|
long err;
|
||||||
|
|
||||||
err = do_shmat(shmid, shmaddr, shmflg, &ret);
|
err = do_shmat(shmid, shmaddr, shmflg, &ret, SHMLBA);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
force_successful_syscall_return();
|
force_successful_syscall_return();
|
||||||
|
|
|
@ -73,7 +73,7 @@ SYSCALL_DEFINE6(ipc, unsigned int, call, int, first, unsigned long, second,
|
||||||
default: {
|
default: {
|
||||||
unsigned long raddr;
|
unsigned long raddr;
|
||||||
ret = do_shmat(first, (char __user *)ptr,
|
ret = do_shmat(first, (char __user *)ptr,
|
||||||
second, &raddr);
|
second, &raddr, SHMLBA);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
return put_user(raddr, (unsigned long __user *) third);
|
return put_user(raddr, (unsigned long __user *) third);
|
||||||
|
|
|
@ -677,9 +677,8 @@ static int raw_sendmsg(struct kiocb *iocb, struct socket *sock,
|
||||||
err = memcpy_fromiovec(skb_put(skb, size), msg->msg_iov, size);
|
err = memcpy_fromiovec(skb_put(skb, size), msg->msg_iov, size);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
goto free_skb;
|
goto free_skb;
|
||||||
err = sock_tx_timestamp(sk, &skb_shinfo(skb)->tx_flags);
|
|
||||||
if (err < 0)
|
sock_tx_timestamp(sk, &skb_shinfo(skb)->tx_flags);
|
||||||
goto free_skb;
|
|
||||||
|
|
||||||
skb->dev = dev;
|
skb->dev = dev;
|
||||||
skb->sk = sk;
|
skb->sk = sk;
|
||||||
|
|
|
@ -1739,20 +1739,21 @@ static void __release_sock(struct sock *sk)
|
||||||
* sk_wait_data - wait for data to arrive at sk_receive_queue
|
* sk_wait_data - wait for data to arrive at sk_receive_queue
|
||||||
* @sk: sock to wait on
|
* @sk: sock to wait on
|
||||||
* @timeo: for how long
|
* @timeo: for how long
|
||||||
|
* @skb: last skb seen on sk_receive_queue
|
||||||
*
|
*
|
||||||
* Now socket state including sk->sk_err is changed only under lock,
|
* Now socket state including sk->sk_err is changed only under lock,
|
||||||
* hence we may omit checks after joining wait queue.
|
* hence we may omit checks after joining wait queue.
|
||||||
* We check receive queue before schedule() only as optimization;
|
* We check receive queue before schedule() only as optimization;
|
||||||
* it is very likely that release_sock() added new data.
|
* it is very likely that release_sock() added new data.
|
||||||
*/
|
*/
|
||||||
int sk_wait_data(struct sock *sk, long *timeo)
|
int sk_wait_data(struct sock *sk, long *timeo, const struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
DEFINE_WAIT(wait);
|
DEFINE_WAIT(wait);
|
||||||
|
|
||||||
prepare_to_wait(sk_sleep(sk), &wait, TASK_INTERRUPTIBLE);
|
prepare_to_wait(sk_sleep(sk), &wait, TASK_INTERRUPTIBLE);
|
||||||
set_bit(SOCK_ASYNC_WAITDATA, &sk->sk_socket->flags);
|
set_bit(SOCK_ASYNC_WAITDATA, &sk->sk_socket->flags);
|
||||||
rc = sk_wait_event(sk, timeo, !skb_queue_empty(&sk->sk_receive_queue));
|
rc = sk_wait_event(sk, timeo, skb_peek_tail(&sk->sk_receive_queue) != skb);
|
||||||
clear_bit(SOCK_ASYNC_WAITDATA, &sk->sk_socket->flags);
|
clear_bit(SOCK_ASYNC_WAITDATA, &sk->sk_socket->flags);
|
||||||
finish_wait(sk_sleep(sk), &wait);
|
finish_wait(sk_sleep(sk), &wait);
|
||||||
return rc;
|
return rc;
|
||||||
|
@ -2092,8 +2093,11 @@ void sock_init_data(struct socket *sock, struct sock *sk)
|
||||||
sk->sk_type = sock->type;
|
sk->sk_type = sock->type;
|
||||||
sk->sk_wq = sock->wq;
|
sk->sk_wq = sock->wq;
|
||||||
sock->sk = sk;
|
sock->sk = sk;
|
||||||
} else
|
sk->sk_uid = SOCK_INODE(sock)->i_uid;
|
||||||
|
} else {
|
||||||
sk->sk_wq = NULL;
|
sk->sk_wq = NULL;
|
||||||
|
sk->sk_uid = make_kuid(sock_net(sk)->user_ns, 0);
|
||||||
|
}
|
||||||
|
|
||||||
spin_lock_init(&sk->sk_dst_lock);
|
spin_lock_init(&sk->sk_dst_lock);
|
||||||
rwlock_init(&sk->sk_callback_lock);
|
rwlock_init(&sk->sk_callback_lock);
|
||||||
|
|
|
@ -848,7 +848,7 @@ int dccp_recvmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
default:
|
default:
|
||||||
dccp_pr_debug("packet_type=%s\n",
|
dccp_pr_debug("packet_type=%s\n",
|
||||||
dccp_packet_name(dh->dccph_type));
|
dccp_packet_name(dh->dccph_type));
|
||||||
sk_eat_skb(sk, skb, 0);
|
sk_eat_skb(sk, skb, false);
|
||||||
}
|
}
|
||||||
verify_sock_status:
|
verify_sock_status:
|
||||||
if (sock_flag(sk, SOCK_DONE)) {
|
if (sock_flag(sk, SOCK_DONE)) {
|
||||||
|
@ -888,7 +888,7 @@ verify_sock_status:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
sk_wait_data(sk, &timeo);
|
sk_wait_data(sk, &timeo, NULL);
|
||||||
continue;
|
continue;
|
||||||
found_ok_skb:
|
found_ok_skb:
|
||||||
if (len > skb->len)
|
if (len > skb->len)
|
||||||
|
@ -905,7 +905,7 @@ verify_sock_status:
|
||||||
len = skb->len;
|
len = skb->len;
|
||||||
found_fin_ok:
|
found_fin_ok:
|
||||||
if (!(flags & MSG_PEEK))
|
if (!(flags & MSG_PEEK))
|
||||||
sk_eat_skb(sk, skb, 0);
|
sk_eat_skb(sk, skb, false);
|
||||||
break;
|
break;
|
||||||
} while (1);
|
} while (1);
|
||||||
out:
|
out:
|
||||||
|
|
|
@ -361,6 +361,7 @@ static void icmp_reply(struct icmp_bxm *icmp_param, struct sk_buff *skb)
|
||||||
fl4.daddr = daddr;
|
fl4.daddr = daddr;
|
||||||
fl4.saddr = rt->rt_spec_dst;
|
fl4.saddr = rt->rt_spec_dst;
|
||||||
fl4.flowi4_mark = mark;
|
fl4.flowi4_mark = mark;
|
||||||
|
fl4.flowi4_uid = sock_net_uid(net, NULL);
|
||||||
fl4.flowi4_tos = RT_TOS(ip_hdr(skb)->tos);
|
fl4.flowi4_tos = RT_TOS(ip_hdr(skb)->tos);
|
||||||
fl4.flowi4_proto = IPPROTO_ICMP;
|
fl4.flowi4_proto = IPPROTO_ICMP;
|
||||||
security_skb_classify_flow(skb, flowi4_to_flowi(&fl4));
|
security_skb_classify_flow(skb, flowi4_to_flowi(&fl4));
|
||||||
|
@ -392,6 +393,7 @@ static struct rtable *icmp_route_lookup(struct net *net,
|
||||||
param->replyopts.opt.opt.faddr : iph->saddr);
|
param->replyopts.opt.opt.faddr : iph->saddr);
|
||||||
fl4->saddr = saddr;
|
fl4->saddr = saddr;
|
||||||
fl4->flowi4_mark = mark;
|
fl4->flowi4_mark = mark;
|
||||||
|
fl4->flowi4_uid = sock_net_uid(net, NULL);
|
||||||
fl4->flowi4_tos = RT_TOS(tos);
|
fl4->flowi4_tos = RT_TOS(tos);
|
||||||
fl4->flowi4_proto = IPPROTO_ICMP;
|
fl4->flowi4_proto = IPPROTO_ICMP;
|
||||||
fl4->fl4_icmp_type = type;
|
fl4->fl4_icmp_type = type;
|
||||||
|
|
|
@ -365,7 +365,8 @@ struct dst_entry *inet_csk_route_req(struct sock *sk,
|
||||||
RT_CONN_FLAGS(sk), RT_SCOPE_UNIVERSE,
|
RT_CONN_FLAGS(sk), RT_SCOPE_UNIVERSE,
|
||||||
sk->sk_protocol, inet_sk_flowi_flags(sk),
|
sk->sk_protocol, inet_sk_flowi_flags(sk),
|
||||||
(opt && opt->opt.srr) ? opt->opt.faddr : ireq->rmt_addr,
|
(opt && opt->opt.srr) ? opt->opt.faddr : ireq->rmt_addr,
|
||||||
ireq->loc_addr, ireq->rmt_port, inet_sk(sk)->inet_sport);
|
ireq->loc_addr, ireq->rmt_port, inet_sk(sk)->inet_sport,
|
||||||
|
sk->sk_uid);
|
||||||
security_req_classify_flow(req, flowi4_to_flowi(fl4));
|
security_req_classify_flow(req, flowi4_to_flowi(fl4));
|
||||||
rt = ip_route_output_flow(net, fl4, sk);
|
rt = ip_route_output_flow(net, fl4, sk);
|
||||||
if (IS_ERR(rt))
|
if (IS_ERR(rt))
|
||||||
|
@ -398,7 +399,8 @@ struct dst_entry *inet_csk_route_child_sock(struct sock *sk,
|
||||||
RT_CONN_FLAGS(sk), RT_SCOPE_UNIVERSE,
|
RT_CONN_FLAGS(sk), RT_SCOPE_UNIVERSE,
|
||||||
sk->sk_protocol, inet_sk_flowi_flags(sk),
|
sk->sk_protocol, inet_sk_flowi_flags(sk),
|
||||||
(opt && opt->opt.srr) ? opt->opt.faddr : ireq->rmt_addr,
|
(opt && opt->opt.srr) ? opt->opt.faddr : ireq->rmt_addr,
|
||||||
ireq->loc_addr, ireq->rmt_port, inet_sk(sk)->inet_sport);
|
ireq->loc_addr, ireq->rmt_port, inet_sk(sk)->inet_sport,
|
||||||
|
sk->sk_uid);
|
||||||
security_req_classify_flow(req, flowi4_to_flowi(fl4));
|
security_req_classify_flow(req, flowi4_to_flowi(fl4));
|
||||||
rt = ip_route_output_flow(net, fl4, sk);
|
rt = ip_route_output_flow(net, fl4, sk);
|
||||||
if (IS_ERR(rt))
|
if (IS_ERR(rt))
|
||||||
|
|
|
@ -1506,7 +1506,8 @@ void ip_send_reply(struct sock *sk, struct sk_buff *skb, __be32 daddr,
|
||||||
RT_SCOPE_UNIVERSE, sk->sk_protocol,
|
RT_SCOPE_UNIVERSE, sk->sk_protocol,
|
||||||
ip_reply_arg_flowi_flags(arg),
|
ip_reply_arg_flowi_flags(arg),
|
||||||
daddr, rt->rt_spec_dst,
|
daddr, rt->rt_spec_dst,
|
||||||
tcp_hdr(skb)->source, tcp_hdr(skb)->dest);
|
tcp_hdr(skb)->source, tcp_hdr(skb)->dest,
|
||||||
|
arg->uid);
|
||||||
security_skb_classify_flow(skb, flowi4_to_flowi(&fl4));
|
security_skb_classify_flow(skb, flowi4_to_flowi(&fl4));
|
||||||
rt = ip_route_output_key(sock_net(sk), &fl4);
|
rt = ip_route_output_key(sock_net(sk), &fl4);
|
||||||
if (IS_ERR(rt))
|
if (IS_ERR(rt))
|
||||||
|
|
|
@ -1211,6 +1211,10 @@ int ip_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, unsi
|
||||||
struct net *net = sock_net(sk);
|
struct net *net = sock_net(sk);
|
||||||
struct mr_table *mrt;
|
struct mr_table *mrt;
|
||||||
|
|
||||||
|
if (sk->sk_type != SOCK_RAW ||
|
||||||
|
inet_sk(sk)->inet_num != IPPROTO_IGMP)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
mrt = ipmr_get_table(net, raw_sk(sk)->ipmr_table ? : RT_TABLE_DEFAULT);
|
mrt = ipmr_get_table(net, raw_sk(sk)->ipmr_table ? : RT_TABLE_DEFAULT);
|
||||||
if (mrt == NULL)
|
if (mrt == NULL)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
@ -1223,11 +1227,8 @@ int ip_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, unsi
|
||||||
|
|
||||||
switch (optname) {
|
switch (optname) {
|
||||||
case MRT_INIT:
|
case MRT_INIT:
|
||||||
if (sk->sk_type != SOCK_RAW ||
|
|
||||||
inet_sk(sk)->inet_num != IPPROTO_IGMP)
|
|
||||||
return -EOPNOTSUPP;
|
|
||||||
if (optlen != sizeof(int))
|
if (optlen != sizeof(int))
|
||||||
return -ENOPROTOOPT;
|
return -EINVAL;
|
||||||
|
|
||||||
rtnl_lock();
|
rtnl_lock();
|
||||||
if (rtnl_dereference(mrt->mroute_sk)) {
|
if (rtnl_dereference(mrt->mroute_sk)) {
|
||||||
|
@ -1288,9 +1289,11 @@ int ip_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, unsi
|
||||||
case MRT_ASSERT:
|
case MRT_ASSERT:
|
||||||
{
|
{
|
||||||
int v;
|
int v;
|
||||||
|
if (optlen != sizeof(v))
|
||||||
|
return -EINVAL;
|
||||||
if (get_user(v, (int __user *)optval))
|
if (get_user(v, (int __user *)optval))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
mrt->mroute_do_assert = (v) ? 1 : 0;
|
mrt->mroute_do_assert = !!v;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#ifdef CONFIG_IP_PIMSM
|
#ifdef CONFIG_IP_PIMSM
|
||||||
|
@ -1298,9 +1301,11 @@ int ip_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, unsi
|
||||||
{
|
{
|
||||||
int v;
|
int v;
|
||||||
|
|
||||||
|
if (optlen != sizeof(v))
|
||||||
|
return -EINVAL;
|
||||||
if (get_user(v, (int __user *)optval))
|
if (get_user(v, (int __user *)optval))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
v = (v) ? 1 : 0;
|
v = !!v;
|
||||||
|
|
||||||
rtnl_lock();
|
rtnl_lock();
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
@ -1329,7 +1334,8 @@ int ip_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, unsi
|
||||||
} else {
|
} else {
|
||||||
if (!ipmr_new_table(net, v))
|
if (!ipmr_new_table(net, v))
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
raw_sk(sk)->ipmr_table = v;
|
else
|
||||||
|
raw_sk(sk)->ipmr_table = v;
|
||||||
}
|
}
|
||||||
rtnl_unlock();
|
rtnl_unlock();
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -1355,6 +1361,10 @@ int ip_mroute_getsockopt(struct sock *sk, int optname, char __user *optval, int
|
||||||
struct net *net = sock_net(sk);
|
struct net *net = sock_net(sk);
|
||||||
struct mr_table *mrt;
|
struct mr_table *mrt;
|
||||||
|
|
||||||
|
if (sk->sk_type != SOCK_RAW ||
|
||||||
|
inet_sk(sk)->inet_num != IPPROTO_IGMP)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
mrt = ipmr_get_table(net, raw_sk(sk)->ipmr_table ? : RT_TABLE_DEFAULT);
|
mrt = ipmr_get_table(net, raw_sk(sk)->ipmr_table ? : RT_TABLE_DEFAULT);
|
||||||
if (mrt == NULL)
|
if (mrt == NULL)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
|
|
@ -738,9 +738,8 @@ int ping_v4_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
ipc.opt = NULL;
|
ipc.opt = NULL;
|
||||||
ipc.oif = sk->sk_bound_dev_if;
|
ipc.oif = sk->sk_bound_dev_if;
|
||||||
ipc.tx_flags = 0;
|
ipc.tx_flags = 0;
|
||||||
err = sock_tx_timestamp(sk, &ipc.tx_flags);
|
|
||||||
if (err)
|
sock_tx_timestamp(sk, &ipc.tx_flags);
|
||||||
return err;
|
|
||||||
|
|
||||||
if (msg->msg_controllen) {
|
if (msg->msg_controllen) {
|
||||||
err = ip_cmsg_send(sock_net(sk), msg, &ipc);
|
err = ip_cmsg_send(sock_net(sk), msg, &ipc);
|
||||||
|
@ -787,7 +786,8 @@ int ping_v4_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
|
|
||||||
flowi4_init_output(&fl4, ipc.oif, sk->sk_mark, tos,
|
flowi4_init_output(&fl4, ipc.oif, sk->sk_mark, tos,
|
||||||
RT_SCOPE_UNIVERSE, sk->sk_protocol,
|
RT_SCOPE_UNIVERSE, sk->sk_protocol,
|
||||||
inet_sk_flowi_flags(sk), faddr, saddr, 0, 0);
|
inet_sk_flowi_flags(sk), faddr, saddr, 0, 0,
|
||||||
|
sk->sk_uid);
|
||||||
|
|
||||||
security_sk_classify_flow(sk, flowi4_to_flowi(&fl4));
|
security_sk_classify_flow(sk, flowi4_to_flowi(&fl4));
|
||||||
rt = ip_route_output_flow(net, &fl4, sk);
|
rt = ip_route_output_flow(net, &fl4, sk);
|
||||||
|
|
|
@ -569,7 +569,7 @@ static int raw_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
RT_SCOPE_UNIVERSE,
|
RT_SCOPE_UNIVERSE,
|
||||||
inet->hdrincl ? IPPROTO_RAW : sk->sk_protocol,
|
inet->hdrincl ? IPPROTO_RAW : sk->sk_protocol,
|
||||||
inet_sk_flowi_flags(sk) | FLOWI_FLAG_CAN_SLEEP,
|
inet_sk_flowi_flags(sk) | FLOWI_FLAG_CAN_SLEEP,
|
||||||
daddr, saddr, 0, 0);
|
daddr, saddr, 0, 0, sk->sk_uid);
|
||||||
|
|
||||||
if (!inet->hdrincl) {
|
if (!inet->hdrincl) {
|
||||||
err = raw_probe_proto_opt(&fl4, msg);
|
err = raw_probe_proto_opt(&fl4, msg);
|
||||||
|
|
|
@ -352,7 +352,7 @@ struct sock *cookie_v4_check(struct sock *sk, struct sk_buff *skb,
|
||||||
RT_CONN_FLAGS(sk), RT_SCOPE_UNIVERSE, IPPROTO_TCP,
|
RT_CONN_FLAGS(sk), RT_SCOPE_UNIVERSE, IPPROTO_TCP,
|
||||||
inet_sk_flowi_flags(sk),
|
inet_sk_flowi_flags(sk),
|
||||||
(opt && opt->srr) ? opt->faddr : ireq->rmt_addr,
|
(opt && opt->srr) ? opt->faddr : ireq->rmt_addr,
|
||||||
ireq->loc_addr, th->source, th->dest);
|
ireq->loc_addr, th->source, th->dest, sk->sk_uid);
|
||||||
security_req_classify_flow(req, flowi4_to_flowi(&fl4));
|
security_req_classify_flow(req, flowi4_to_flowi(&fl4));
|
||||||
rt = ip_route_output_key(sock_net(sk), &fl4);
|
rt = ip_route_output_key(sock_net(sk), &fl4);
|
||||||
if (IS_ERR(rt)) {
|
if (IS_ERR(rt)) {
|
||||||
|
|
|
@ -677,7 +677,7 @@ ssize_t tcp_splice_read(struct socket *sock, loff_t *ppos,
|
||||||
*/
|
*/
|
||||||
if (!skb_queue_empty(&sk->sk_receive_queue))
|
if (!skb_queue_empty(&sk->sk_receive_queue))
|
||||||
break;
|
break;
|
||||||
sk_wait_data(sk, &timeo);
|
sk_wait_data(sk, &timeo, NULL);
|
||||||
if (signal_pending(current)) {
|
if (signal_pending(current)) {
|
||||||
ret = sock_intr_errno(timeo);
|
ret = sock_intr_errno(timeo);
|
||||||
break;
|
break;
|
||||||
|
@ -1413,11 +1413,11 @@ int tcp_read_sock(struct sock *sk, read_descriptor_t *desc,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (tcp_hdr(skb)->fin) {
|
if (tcp_hdr(skb)->fin) {
|
||||||
sk_eat_skb(sk, skb, 0);
|
sk_eat_skb(sk, skb, false);
|
||||||
++seq;
|
++seq;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
sk_eat_skb(sk, skb, 0);
|
sk_eat_skb(sk, skb, false);
|
||||||
if (!desc->count)
|
if (!desc->count)
|
||||||
break;
|
break;
|
||||||
tp->copied_seq = seq;
|
tp->copied_seq = seq;
|
||||||
|
@ -1456,8 +1456,8 @@ int tcp_recvmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
int target; /* Read at least this many bytes */
|
int target; /* Read at least this many bytes */
|
||||||
long timeo;
|
long timeo;
|
||||||
struct task_struct *user_recv = NULL;
|
struct task_struct *user_recv = NULL;
|
||||||
int copied_early = 0;
|
bool copied_early = false;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb, *last;
|
||||||
u32 urg_hole = 0;
|
u32 urg_hole = 0;
|
||||||
|
|
||||||
lock_sock(sk);
|
lock_sock(sk);
|
||||||
|
@ -1517,7 +1517,9 @@ int tcp_recvmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
|
|
||||||
/* Next get a buffer. */
|
/* Next get a buffer. */
|
||||||
|
|
||||||
|
last = skb_peek_tail(&sk->sk_receive_queue);
|
||||||
skb_queue_walk(&sk->sk_receive_queue, skb) {
|
skb_queue_walk(&sk->sk_receive_queue, skb) {
|
||||||
|
last = skb;
|
||||||
/* Now that we have two receive queues this
|
/* Now that we have two receive queues this
|
||||||
* shouldn't happen.
|
* shouldn't happen.
|
||||||
*/
|
*/
|
||||||
|
@ -1646,8 +1648,9 @@ int tcp_recvmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
/* Do not sleep, just process backlog. */
|
/* Do not sleep, just process backlog. */
|
||||||
release_sock(sk);
|
release_sock(sk);
|
||||||
lock_sock(sk);
|
lock_sock(sk);
|
||||||
} else
|
} else {
|
||||||
sk_wait_data(sk, &timeo);
|
sk_wait_data(sk, &timeo, last);
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_NET_DMA
|
#ifdef CONFIG_NET_DMA
|
||||||
tcp_service_net_dma(sk, false); /* Don't block */
|
tcp_service_net_dma(sk, false); /* Don't block */
|
||||||
|
@ -1735,7 +1738,7 @@ do_prequeue:
|
||||||
dma_async_memcpy_issue_pending(tp->ucopy.dma_chan);
|
dma_async_memcpy_issue_pending(tp->ucopy.dma_chan);
|
||||||
|
|
||||||
if ((offset + used) == skb->len)
|
if ((offset + used) == skb->len)
|
||||||
copied_early = 1;
|
copied_early = true;
|
||||||
|
|
||||||
} else
|
} else
|
||||||
#endif
|
#endif
|
||||||
|
@ -1769,7 +1772,7 @@ skip_copy:
|
||||||
goto found_fin_ok;
|
goto found_fin_ok;
|
||||||
if (!(flags & MSG_PEEK)) {
|
if (!(flags & MSG_PEEK)) {
|
||||||
sk_eat_skb(sk, skb, copied_early);
|
sk_eat_skb(sk, skb, copied_early);
|
||||||
copied_early = 0;
|
copied_early = false;
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
@ -1778,7 +1781,7 @@ skip_copy:
|
||||||
++*seq;
|
++*seq;
|
||||||
if (!(flags & MSG_PEEK)) {
|
if (!(flags & MSG_PEEK)) {
|
||||||
sk_eat_skb(sk, skb, copied_early);
|
sk_eat_skb(sk, skb, copied_early);
|
||||||
copied_early = 0;
|
copied_early = false;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
} while (len > 0);
|
} while (len > 0);
|
||||||
|
|
|
@ -689,6 +689,7 @@ static void tcp_v4_send_reset(struct sock *sk, struct sk_buff *skb)
|
||||||
arg.bound_dev_if = sk->sk_bound_dev_if;
|
arg.bound_dev_if = sk->sk_bound_dev_if;
|
||||||
|
|
||||||
arg.tos = ip_hdr(skb)->tos;
|
arg.tos = ip_hdr(skb)->tos;
|
||||||
|
arg.uid = sock_net_uid(net, sk && sk_fullsock(sk) ? sk : NULL);
|
||||||
ip_send_reply(net->ipv4.tcp_sock, skb, ip_hdr(skb)->saddr,
|
ip_send_reply(net->ipv4.tcp_sock, skb, ip_hdr(skb)->saddr,
|
||||||
&arg, arg.iov[0].iov_len);
|
&arg, arg.iov[0].iov_len);
|
||||||
|
|
||||||
|
@ -708,7 +709,8 @@ release_sk1:
|
||||||
outside socket context is ugly, certainly. What can I do?
|
outside socket context is ugly, certainly. What can I do?
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static void tcp_v4_send_ack(struct sk_buff *skb, u32 seq, u32 ack,
|
static void tcp_v4_send_ack(const struct sock *sk, struct sk_buff *skb,
|
||||||
|
u32 seq, u32 ack,
|
||||||
u32 win, u32 ts, int oif,
|
u32 win, u32 ts, int oif,
|
||||||
struct tcp_md5sig_key *key,
|
struct tcp_md5sig_key *key,
|
||||||
int reply_flags, u8 tos)
|
int reply_flags, u8 tos)
|
||||||
|
@ -723,7 +725,7 @@ static void tcp_v4_send_ack(struct sk_buff *skb, u32 seq, u32 ack,
|
||||||
];
|
];
|
||||||
} rep;
|
} rep;
|
||||||
struct ip_reply_arg arg;
|
struct ip_reply_arg arg;
|
||||||
struct net *net = dev_net(skb_dst(skb)->dev);
|
struct net *net = sock_net(sk);
|
||||||
|
|
||||||
memset(&rep.th, 0, sizeof(struct tcphdr));
|
memset(&rep.th, 0, sizeof(struct tcphdr));
|
||||||
memset(&arg, 0, sizeof(arg));
|
memset(&arg, 0, sizeof(arg));
|
||||||
|
@ -772,6 +774,7 @@ static void tcp_v4_send_ack(struct sk_buff *skb, u32 seq, u32 ack,
|
||||||
if (oif)
|
if (oif)
|
||||||
arg.bound_dev_if = oif;
|
arg.bound_dev_if = oif;
|
||||||
arg.tos = tos;
|
arg.tos = tos;
|
||||||
|
arg.uid = sock_net_uid(net, sk_fullsock(sk) ? sk : NULL);
|
||||||
ip_send_reply(net->ipv4.tcp_sock, skb, ip_hdr(skb)->saddr,
|
ip_send_reply(net->ipv4.tcp_sock, skb, ip_hdr(skb)->saddr,
|
||||||
&arg, arg.iov[0].iov_len);
|
&arg, arg.iov[0].iov_len);
|
||||||
|
|
||||||
|
@ -783,7 +786,7 @@ static void tcp_v4_timewait_ack(struct sock *sk, struct sk_buff *skb)
|
||||||
struct inet_timewait_sock *tw = inet_twsk(sk);
|
struct inet_timewait_sock *tw = inet_twsk(sk);
|
||||||
struct tcp_timewait_sock *tcptw = tcp_twsk(sk);
|
struct tcp_timewait_sock *tcptw = tcp_twsk(sk);
|
||||||
|
|
||||||
tcp_v4_send_ack(skb, tcptw->tw_snd_nxt, tcptw->tw_rcv_nxt,
|
tcp_v4_send_ack(sk, skb, tcptw->tw_snd_nxt, tcptw->tw_rcv_nxt,
|
||||||
tcptw->tw_rcv_wnd >> tw->tw_rcv_wscale,
|
tcptw->tw_rcv_wnd >> tw->tw_rcv_wscale,
|
||||||
tcptw->tw_ts_recent,
|
tcptw->tw_ts_recent,
|
||||||
tw->tw_bound_dev_if,
|
tw->tw_bound_dev_if,
|
||||||
|
@ -798,7 +801,7 @@ static void tcp_v4_timewait_ack(struct sock *sk, struct sk_buff *skb)
|
||||||
static void tcp_v4_reqsk_send_ack(struct sock *sk, struct sk_buff *skb,
|
static void tcp_v4_reqsk_send_ack(struct sock *sk, struct sk_buff *skb,
|
||||||
struct request_sock *req)
|
struct request_sock *req)
|
||||||
{
|
{
|
||||||
tcp_v4_send_ack(skb, tcp_rsk(req)->snt_isn + 1,
|
tcp_v4_send_ack(sk, skb, tcp_rsk(req)->snt_isn + 1,
|
||||||
tcp_rsk(req)->rcv_isn + 1, req->rcv_wnd,
|
tcp_rsk(req)->rcv_isn + 1, req->rcv_wnd,
|
||||||
req->ts_recent,
|
req->ts_recent,
|
||||||
0,
|
0,
|
||||||
|
@ -1589,7 +1592,7 @@ static __sum16 tcp_v4_checksum_init(struct sk_buff *skb)
|
||||||
|
|
||||||
|
|
||||||
/* The socket must have it's spinlock held when we get
|
/* The socket must have it's spinlock held when we get
|
||||||
* here, unless it is a TCP_LISTEN socket.
|
* here.
|
||||||
*
|
*
|
||||||
* We have a potential double-lock case here, so even when
|
* We have a potential double-lock case here, so even when
|
||||||
* doing backlog processing we use the BH locking scheme.
|
* doing backlog processing we use the BH locking scheme.
|
||||||
|
@ -1728,11 +1731,6 @@ process:
|
||||||
|
|
||||||
skb->dev = NULL;
|
skb->dev = NULL;
|
||||||
|
|
||||||
if (sk->sk_state == TCP_LISTEN) {
|
|
||||||
ret = tcp_v4_do_rcv(sk, skb);
|
|
||||||
goto put_and_return;
|
|
||||||
}
|
|
||||||
|
|
||||||
bh_lock_sock_nested(sk);
|
bh_lock_sock_nested(sk);
|
||||||
tcp_sk(sk)->segs_in += max_t(u16, 1, skb_shinfo(skb)->gso_segs);
|
tcp_sk(sk)->segs_in += max_t(u16, 1, skb_shinfo(skb)->gso_segs);
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
@ -1756,7 +1754,6 @@ process:
|
||||||
}
|
}
|
||||||
bh_unlock_sock(sk);
|
bh_unlock_sock(sk);
|
||||||
|
|
||||||
put_and_return:
|
|
||||||
sock_put(sk);
|
sock_put(sk);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
|
|
|
@ -872,9 +872,9 @@ int udp_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
ipc.addr = inet->inet_saddr;
|
ipc.addr = inet->inet_saddr;
|
||||||
|
|
||||||
ipc.oif = sk->sk_bound_dev_if;
|
ipc.oif = sk->sk_bound_dev_if;
|
||||||
err = sock_tx_timestamp(sk, &ipc.tx_flags);
|
|
||||||
if (err)
|
sock_tx_timestamp(sk, &ipc.tx_flags);
|
||||||
return err;
|
|
||||||
if (msg->msg_controllen) {
|
if (msg->msg_controllen) {
|
||||||
err = ip_cmsg_send(sock_net(sk), msg, &ipc);
|
err = ip_cmsg_send(sock_net(sk), msg, &ipc);
|
||||||
if (err)
|
if (err)
|
||||||
|
@ -932,7 +932,8 @@ int udp_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
flowi4_init_output(fl4, ipc.oif, sk->sk_mark, tos,
|
flowi4_init_output(fl4, ipc.oif, sk->sk_mark, tos,
|
||||||
RT_SCOPE_UNIVERSE, sk->sk_protocol,
|
RT_SCOPE_UNIVERSE, sk->sk_protocol,
|
||||||
inet_sk_flowi_flags(sk)|FLOWI_FLAG_CAN_SLEEP,
|
inet_sk_flowi_flags(sk)|FLOWI_FLAG_CAN_SLEEP,
|
||||||
faddr, saddr, dport, inet->inet_sport);
|
faddr, saddr, dport, inet->inet_sport,
|
||||||
|
sk->sk_uid);
|
||||||
|
|
||||||
security_sk_classify_flow(sk, flowi4_to_flowi(fl4));
|
security_sk_classify_flow(sk, flowi4_to_flowi(fl4));
|
||||||
rt = ip_route_output_flow(net, fl4, sk);
|
rt = ip_route_output_flow(net, fl4, sk);
|
||||||
|
|
|
@ -711,6 +711,7 @@ int inet6_sk_rebuild_header(struct sock *sk)
|
||||||
fl6.flowi6_mark = sk->sk_mark;
|
fl6.flowi6_mark = sk->sk_mark;
|
||||||
fl6.fl6_dport = inet->inet_dport;
|
fl6.fl6_dport = inet->inet_dport;
|
||||||
fl6.fl6_sport = inet->inet_sport;
|
fl6.fl6_sport = inet->inet_sport;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
security_sk_classify_flow(sk, flowi6_to_flowi(&fl6));
|
security_sk_classify_flow(sk, flowi6_to_flowi(&fl6));
|
||||||
|
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
|
|
|
@ -620,7 +620,7 @@ static void ah6_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
|
||||||
|
|
||||||
NETDEBUG(KERN_DEBUG "pmtu discovery on SA AH/%08x/%pI6\n",
|
NETDEBUG(KERN_DEBUG "pmtu discovery on SA AH/%08x/%pI6\n",
|
||||||
ntohl(ah->spi), &iph->daddr);
|
ntohl(ah->spi), &iph->daddr);
|
||||||
ip6_update_pmtu(skb, net, info, 0, 0);
|
ip6_update_pmtu(skb, net, info, 0, 0, sock_net_uid(net, NULL));
|
||||||
xfrm_state_put(x);
|
xfrm_state_put(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -160,6 +160,7 @@ ipv4_connected:
|
||||||
fl6.flowi6_mark = sk->sk_mark;
|
fl6.flowi6_mark = sk->sk_mark;
|
||||||
fl6.fl6_dport = inet->inet_dport;
|
fl6.fl6_dport = inet->inet_dport;
|
||||||
fl6.fl6_sport = inet->inet_sport;
|
fl6.fl6_sport = inet->inet_sport;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
|
|
||||||
if (!fl6.flowi6_oif && (addr_type&IPV6_ADDR_MULTICAST))
|
if (!fl6.flowi6_oif && (addr_type&IPV6_ADDR_MULTICAST))
|
||||||
fl6.flowi6_oif = np->mcast_oif;
|
fl6.flowi6_oif = np->mcast_oif;
|
||||||
|
|
|
@ -441,7 +441,7 @@ static void esp6_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
|
||||||
return;
|
return;
|
||||||
printk(KERN_DEBUG "pmtu discovery on SA ESP/%08x/%pI6\n",
|
printk(KERN_DEBUG "pmtu discovery on SA ESP/%08x/%pI6\n",
|
||||||
ntohl(esph->spi), &iph->daddr);
|
ntohl(esph->spi), &iph->daddr);
|
||||||
ip6_update_pmtu(skb, net, info, 0, 0);
|
ip6_update_pmtu(skb, net, info, 0, 0, sock_net_uid(net, NULL));
|
||||||
xfrm_state_put(x);
|
xfrm_state_put(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -466,6 +466,7 @@ void icmpv6_send(struct sk_buff *skb, u8 type, u8 code, __u32 info)
|
||||||
fl6.flowi6_oif = iif;
|
fl6.flowi6_oif = iif;
|
||||||
fl6.fl6_icmp_type = type;
|
fl6.fl6_icmp_type = type;
|
||||||
fl6.fl6_icmp_code = code;
|
fl6.fl6_icmp_code = code;
|
||||||
|
fl6.flowi6_uid = sock_net_uid(net, NULL);
|
||||||
security_skb_classify_flow(skb, flowi6_to_flowi(&fl6));
|
security_skb_classify_flow(skb, flowi6_to_flowi(&fl6));
|
||||||
|
|
||||||
sk = icmpv6_xmit_lock(net);
|
sk = icmpv6_xmit_lock(net);
|
||||||
|
@ -564,6 +565,7 @@ static void icmpv6_echo_reply(struct sk_buff *skb)
|
||||||
fl6.flowi6_oif = skb->dev->ifindex;
|
fl6.flowi6_oif = skb->dev->ifindex;
|
||||||
fl6.fl6_icmp_type = ICMPV6_ECHO_REPLY;
|
fl6.fl6_icmp_type = ICMPV6_ECHO_REPLY;
|
||||||
fl6.flowi6_mark = mark;
|
fl6.flowi6_mark = mark;
|
||||||
|
fl6.flowi6_uid = sock_net_uid(net, NULL);
|
||||||
security_skb_classify_flow(skb, flowi6_to_flowi(&fl6));
|
security_skb_classify_flow(skb, flowi6_to_flowi(&fl6));
|
||||||
|
|
||||||
sk = icmpv6_xmit_lock(net);
|
sk = icmpv6_xmit_lock(net);
|
||||||
|
|
|
@ -74,6 +74,7 @@ struct dst_entry *inet6_csk_route_req(struct sock *sk,
|
||||||
fl6.flowi6_mark = inet_rsk(req)->ir_mark;
|
fl6.flowi6_mark = inet_rsk(req)->ir_mark;
|
||||||
fl6.fl6_dport = inet_rsk(req)->rmt_port;
|
fl6.fl6_dport = inet_rsk(req)->rmt_port;
|
||||||
fl6.fl6_sport = inet_rsk(req)->loc_port;
|
fl6.fl6_sport = inet_rsk(req)->loc_port;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
security_req_classify_flow(req, flowi6_to_flowi(&fl6));
|
security_req_classify_flow(req, flowi6_to_flowi(&fl6));
|
||||||
|
|
||||||
dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false);
|
dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false);
|
||||||
|
@ -225,6 +226,7 @@ int inet6_csk_xmit(struct sk_buff *skb, struct flowi *fl_unused)
|
||||||
fl6.flowi6_mark = sk->sk_mark;
|
fl6.flowi6_mark = sk->sk_mark;
|
||||||
fl6.fl6_sport = inet->inet_sport;
|
fl6.fl6_sport = inet->inet_sport;
|
||||||
fl6.fl6_dport = inet->inet_dport;
|
fl6.fl6_dport = inet->inet_dport;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
security_sk_classify_flow(sk, flowi6_to_flowi(&fl6));
|
security_sk_classify_flow(sk, flowi6_to_flowi(&fl6));
|
||||||
|
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
|
|
|
@ -1314,11 +1314,8 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to,
|
||||||
}
|
}
|
||||||
|
|
||||||
/* For UDP, check if TX timestamp is enabled */
|
/* For UDP, check if TX timestamp is enabled */
|
||||||
if (sk->sk_type == SOCK_DGRAM) {
|
if (sk->sk_type == SOCK_DGRAM)
|
||||||
err = sock_tx_timestamp(sk, &tx_flags);
|
sock_tx_timestamp(sk, &tx_flags);
|
||||||
if (err)
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Let's try using as much space as possible.
|
* Let's try using as much space as possible.
|
||||||
|
|
|
@ -1038,6 +1038,8 @@ ip4ip6_tnl_xmit(struct sk_buff *skb, struct net_device *dev)
|
||||||
memcpy(&fl6, &t->fl.u.ip6, sizeof (fl6));
|
memcpy(&fl6, &t->fl.u.ip6, sizeof (fl6));
|
||||||
fl6.flowi6_proto = IPPROTO_IPIP;
|
fl6.flowi6_proto = IPPROTO_IPIP;
|
||||||
|
|
||||||
|
fl6.flowi6_uid = sock_net_uid(dev_net(dev), NULL);
|
||||||
|
|
||||||
dsfield = ipv4_get_dsfield(iph);
|
dsfield = ipv4_get_dsfield(iph);
|
||||||
|
|
||||||
if (t->parms.flags & IP6_TNL_F_USE_ORIG_TCLASS)
|
if (t->parms.flags & IP6_TNL_F_USE_ORIG_TCLASS)
|
||||||
|
@ -1089,6 +1091,7 @@ ip6ip6_tnl_xmit(struct sk_buff *skb, struct net_device *dev)
|
||||||
|
|
||||||
memcpy(&fl6, &t->fl.u.ip6, sizeof (fl6));
|
memcpy(&fl6, &t->fl.u.ip6, sizeof (fl6));
|
||||||
fl6.flowi6_proto = IPPROTO_IPV6;
|
fl6.flowi6_proto = IPPROTO_IPV6;
|
||||||
|
fl6.flowi6_uid = sock_net_uid(dev_net(dev), NULL);
|
||||||
|
|
||||||
dsfield = ipv6_get_dsfield(ipv6h);
|
dsfield = ipv6_get_dsfield(ipv6h);
|
||||||
if (t->parms.flags & IP6_TNL_F_USE_ORIG_TCLASS)
|
if (t->parms.flags & IP6_TNL_F_USE_ORIG_TCLASS)
|
||||||
|
|
|
@ -1588,6 +1588,10 @@ int ip6_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, uns
|
||||||
struct net *net = sock_net(sk);
|
struct net *net = sock_net(sk);
|
||||||
struct mr6_table *mrt;
|
struct mr6_table *mrt;
|
||||||
|
|
||||||
|
if (sk->sk_type != SOCK_RAW ||
|
||||||
|
inet_sk(sk)->inet_num != IPPROTO_ICMPV6)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
mrt = ip6mr_get_table(net, raw6_sk(sk)->ip6mr_table ? : RT6_TABLE_DFLT);
|
mrt = ip6mr_get_table(net, raw6_sk(sk)->ip6mr_table ? : RT6_TABLE_DFLT);
|
||||||
if (mrt == NULL)
|
if (mrt == NULL)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
@ -1599,9 +1603,6 @@ int ip6_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, uns
|
||||||
|
|
||||||
switch (optname) {
|
switch (optname) {
|
||||||
case MRT6_INIT:
|
case MRT6_INIT:
|
||||||
if (sk->sk_type != SOCK_RAW ||
|
|
||||||
inet_sk(sk)->inet_num != IPPROTO_ICMPV6)
|
|
||||||
return -EOPNOTSUPP;
|
|
||||||
if (optlen < sizeof(int))
|
if (optlen < sizeof(int))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
|
@ -1722,6 +1723,10 @@ int ip6_mroute_getsockopt(struct sock *sk, int optname, char __user *optval,
|
||||||
struct net *net = sock_net(sk);
|
struct net *net = sock_net(sk);
|
||||||
struct mr6_table *mrt;
|
struct mr6_table *mrt;
|
||||||
|
|
||||||
|
if (sk->sk_type != SOCK_RAW ||
|
||||||
|
inet_sk(sk)->inet_num != IPPROTO_ICMPV6)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
mrt = ip6mr_get_table(net, raw6_sk(sk)->ip6mr_table ? : RT6_TABLE_DFLT);
|
mrt = ip6mr_get_table(net, raw6_sk(sk)->ip6mr_table ? : RT6_TABLE_DFLT);
|
||||||
if (mrt == NULL)
|
if (mrt == NULL)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
|
|
@ -72,7 +72,7 @@ static void ipcomp6_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
|
||||||
|
|
||||||
printk(KERN_DEBUG "pmtu discovery on SA IPCOMP/%08x/%pI6\n",
|
printk(KERN_DEBUG "pmtu discovery on SA IPCOMP/%08x/%pI6\n",
|
||||||
spi, &iph->daddr);
|
spi, &iph->daddr);
|
||||||
ip6_update_pmtu(skb, net, info, 0, 0);
|
ip6_update_pmtu(skb, net, info, 0, 0, sock_net_uid(net, NULL));
|
||||||
xfrm_state_put(x);
|
xfrm_state_put(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -19,6 +19,7 @@ int ip6_route_me_harder(struct sk_buff *skb)
|
||||||
struct flowi6 fl6 = {
|
struct flowi6 fl6 = {
|
||||||
.flowi6_oif = skb->sk ? skb->sk->sk_bound_dev_if : 0,
|
.flowi6_oif = skb->sk ? skb->sk->sk_bound_dev_if : 0,
|
||||||
.flowi6_mark = skb->mark,
|
.flowi6_mark = skb->mark,
|
||||||
|
.flowi6_uid = sock_net_uid(net, skb->sk),
|
||||||
.daddr = iph->daddr,
|
.daddr = iph->daddr,
|
||||||
.saddr = iph->saddr,
|
.saddr = iph->saddr,
|
||||||
};
|
};
|
||||||
|
|
|
@ -162,6 +162,7 @@ int ping_v6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
||||||
fl6.saddr = np->saddr;
|
fl6.saddr = np->saddr;
|
||||||
fl6.daddr = *daddr;
|
fl6.daddr = *daddr;
|
||||||
fl6.flowi6_mark = sk->sk_mark;
|
fl6.flowi6_mark = sk->sk_mark;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
fl6.fl6_icmp_type = user_icmph.icmp6_type;
|
fl6.fl6_icmp_type = user_icmph.icmp6_type;
|
||||||
fl6.fl6_icmp_code = user_icmph.icmp6_code;
|
fl6.fl6_icmp_code = user_icmph.icmp6_code;
|
||||||
security_sk_classify_flow(sk, flowi6_to_flowi(&fl6));
|
security_sk_classify_flow(sk, flowi6_to_flowi(&fl6));
|
||||||
|
|
|
@ -760,6 +760,7 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk,
|
||||||
memset(&fl6, 0, sizeof(fl6));
|
memset(&fl6, 0, sizeof(fl6));
|
||||||
|
|
||||||
fl6.flowi6_mark = sk->sk_mark;
|
fl6.flowi6_mark = sk->sk_mark;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
|
|
||||||
if (sin6) {
|
if (sin6) {
|
||||||
if (addr_len < SIN6_LEN_RFC2133)
|
if (addr_len < SIN6_LEN_RFC2133)
|
||||||
|
|
|
@ -1094,7 +1094,7 @@ static void ip6_rt_update_pmtu(struct dst_entry *dst, u32 mtu)
|
||||||
}
|
}
|
||||||
|
|
||||||
void ip6_update_pmtu(struct sk_buff *skb, struct net *net, __be32 mtu,
|
void ip6_update_pmtu(struct sk_buff *skb, struct net *net, __be32 mtu,
|
||||||
int oif, u32 mark)
|
int oif, u32 mark, kuid_t uid)
|
||||||
{
|
{
|
||||||
const struct ipv6hdr *iph = (struct ipv6hdr *) skb->data;
|
const struct ipv6hdr *iph = (struct ipv6hdr *) skb->data;
|
||||||
struct dst_entry *dst;
|
struct dst_entry *dst;
|
||||||
|
@ -1107,6 +1107,7 @@ void ip6_update_pmtu(struct sk_buff *skb, struct net *net, __be32 mtu,
|
||||||
fl6.daddr = iph->daddr;
|
fl6.daddr = iph->daddr;
|
||||||
fl6.saddr = iph->saddr;
|
fl6.saddr = iph->saddr;
|
||||||
fl6.flowlabel = (*(__be32 *) iph) & IPV6_FLOWINFO_MASK;
|
fl6.flowlabel = (*(__be32 *) iph) & IPV6_FLOWINFO_MASK;
|
||||||
|
fl6.flowi6_uid = uid;
|
||||||
|
|
||||||
dst = ip6_route_output(net, NULL, &fl6);
|
dst = ip6_route_output(net, NULL, &fl6);
|
||||||
if (!dst->error)
|
if (!dst->error)
|
||||||
|
@ -1118,7 +1119,7 @@ EXPORT_SYMBOL_GPL(ip6_update_pmtu);
|
||||||
void ip6_sk_update_pmtu(struct sk_buff *skb, struct sock *sk, __be32 mtu)
|
void ip6_sk_update_pmtu(struct sk_buff *skb, struct sock *sk, __be32 mtu)
|
||||||
{
|
{
|
||||||
ip6_update_pmtu(skb, sock_net(sk), mtu,
|
ip6_update_pmtu(skb, sock_net(sk), mtu,
|
||||||
sk->sk_bound_dev_if, sk->sk_mark);
|
sk->sk_bound_dev_if, sk->sk_mark, sk->sk_uid);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(ip6_sk_update_pmtu);
|
EXPORT_SYMBOL_GPL(ip6_sk_update_pmtu);
|
||||||
|
|
||||||
|
|
|
@ -246,6 +246,7 @@ struct sock *cookie_v6_check(struct sock *sk, struct sk_buff *skb)
|
||||||
fl6.flowi6_mark = inet_rsk(req)->ir_mark;
|
fl6.flowi6_mark = inet_rsk(req)->ir_mark;
|
||||||
fl6.fl6_dport = inet_rsk(req)->rmt_port;
|
fl6.fl6_dport = inet_rsk(req)->rmt_port;
|
||||||
fl6.fl6_sport = inet_sk(sk)->inet_sport;
|
fl6.fl6_sport = inet_sk(sk)->inet_sport;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
security_req_classify_flow(req, flowi6_to_flowi(&fl6));
|
security_req_classify_flow(req, flowi6_to_flowi(&fl6));
|
||||||
|
|
||||||
dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false);
|
dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false);
|
||||||
|
|
|
@ -252,6 +252,7 @@ static int tcp_v6_connect(struct sock *sk, struct sockaddr *uaddr,
|
||||||
fl6.flowi6_mark = sk->sk_mark;
|
fl6.flowi6_mark = sk->sk_mark;
|
||||||
fl6.fl6_dport = usin->sin6_port;
|
fl6.fl6_dport = usin->sin6_port;
|
||||||
fl6.fl6_sport = inet->inet_sport;
|
fl6.fl6_sport = inet->inet_sport;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
|
|
||||||
opt = rcu_dereference_protected(np->opt, sock_owned_by_user(sk));
|
opt = rcu_dereference_protected(np->opt, sock_owned_by_user(sk));
|
||||||
final_p = fl6_update_dst(&fl6, opt, &final);
|
final_p = fl6_update_dst(&fl6, opt, &final);
|
||||||
|
@ -406,6 +407,7 @@ static void tcp_v6_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
|
||||||
fl6.flowi6_mark = sk->sk_mark;
|
fl6.flowi6_mark = sk->sk_mark;
|
||||||
fl6.fl6_dport = inet->inet_dport;
|
fl6.fl6_dport = inet->inet_dport;
|
||||||
fl6.fl6_sport = inet->inet_sport;
|
fl6.fl6_sport = inet->inet_sport;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
security_skb_classify_flow(skb, flowi6_to_flowi(&fl6));
|
security_skb_classify_flow(skb, flowi6_to_flowi(&fl6));
|
||||||
|
|
||||||
dst = ip6_dst_lookup_flow(sk, &fl6, NULL, false);
|
dst = ip6_dst_lookup_flow(sk, &fl6, NULL, false);
|
||||||
|
@ -907,6 +909,7 @@ static void tcp_v6_send_response(struct sock *sk, struct sk_buff *skb, u32 seq,
|
||||||
fl6.flowi6_mark = IP6_REPLY_MARK(net, skb->mark);
|
fl6.flowi6_mark = IP6_REPLY_MARK(net, skb->mark);
|
||||||
fl6.fl6_dport = t1->dest;
|
fl6.fl6_dport = t1->dest;
|
||||||
fl6.fl6_sport = t1->source;
|
fl6.fl6_sport = t1->source;
|
||||||
|
fl6.flowi6_uid = sock_net_uid(net, sk && sk_fullsock(sk) ? sk : NULL);
|
||||||
security_skb_classify_flow(skb, flowi6_to_flowi(&fl6));
|
security_skb_classify_flow(skb, flowi6_to_flowi(&fl6));
|
||||||
|
|
||||||
/* Pass a socket to ip6_dst_lookup either it is for RST
|
/* Pass a socket to ip6_dst_lookup either it is for RST
|
||||||
|
@ -1464,7 +1467,7 @@ static __sum16 tcp_v6_checksum_init(struct sk_buff *skb)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* The socket must have it's spinlock held when we get
|
/* The socket must have it's spinlock held when we get
|
||||||
* here, unless it is a TCP_LISTEN socket.
|
* here.
|
||||||
*
|
*
|
||||||
* We have a potential double-lock case here, so even when
|
* We have a potential double-lock case here, so even when
|
||||||
* doing backlog processing we use the BH locking scheme.
|
* doing backlog processing we use the BH locking scheme.
|
||||||
|
@ -1658,11 +1661,6 @@ process:
|
||||||
|
|
||||||
skb->dev = NULL;
|
skb->dev = NULL;
|
||||||
|
|
||||||
if (sk->sk_state == TCP_LISTEN) {
|
|
||||||
ret = tcp_v6_do_rcv(sk, skb);
|
|
||||||
goto put_and_return;
|
|
||||||
}
|
|
||||||
|
|
||||||
bh_lock_sock_nested(sk);
|
bh_lock_sock_nested(sk);
|
||||||
tcp_sk(sk)->segs_in += max_t(u16, 1, skb_shinfo(skb)->gso_segs);
|
tcp_sk(sk)->segs_in += max_t(u16, 1, skb_shinfo(skb)->gso_segs);
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
@ -1686,7 +1684,6 @@ process:
|
||||||
}
|
}
|
||||||
bh_unlock_sock(sk);
|
bh_unlock_sock(sk);
|
||||||
|
|
||||||
put_and_return:
|
|
||||||
sock_put(sk);
|
sock_put(sk);
|
||||||
return ret ? -1 : 0;
|
return ret ? -1 : 0;
|
||||||
|
|
||||||
|
|
|
@ -1094,6 +1094,7 @@ do_udp_sendmsg:
|
||||||
fl6.flowi6_oif = np->sticky_pktinfo.ipi6_ifindex;
|
fl6.flowi6_oif = np->sticky_pktinfo.ipi6_ifindex;
|
||||||
|
|
||||||
fl6.flowi6_mark = sk->sk_mark;
|
fl6.flowi6_mark = sk->sk_mark;
|
||||||
|
fl6.flowi6_uid = sk->sk_uid;
|
||||||
|
|
||||||
if (msg->msg_controllen) {
|
if (msg->msg_controllen) {
|
||||||
opt = &opt_space;
|
opt = &opt_space;
|
||||||
|
|
|
@ -614,7 +614,7 @@ static int llc_wait_data(struct sock *sk, long timeo)
|
||||||
if (signal_pending(current))
|
if (signal_pending(current))
|
||||||
break;
|
break;
|
||||||
rc = 0;
|
rc = 0;
|
||||||
if (sk_wait_data(sk, &timeo))
|
if (sk_wait_data(sk, &timeo, NULL))
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
return rc;
|
return rc;
|
||||||
|
@ -803,7 +803,7 @@ static int llc_ui_recvmsg(struct kiocb *iocb, struct socket *sock,
|
||||||
release_sock(sk);
|
release_sock(sk);
|
||||||
lock_sock(sk);
|
lock_sock(sk);
|
||||||
} else
|
} else
|
||||||
sk_wait_data(sk, &timeo);
|
sk_wait_data(sk, &timeo, NULL);
|
||||||
|
|
||||||
if ((flags & MSG_PEEK) && peek_seq != llc->copied_seq) {
|
if ((flags & MSG_PEEK) && peek_seq != llc->copied_seq) {
|
||||||
if (net_ratelimit())
|
if (net_ratelimit())
|
||||||
|
@ -841,7 +841,7 @@ static int llc_ui_recvmsg(struct kiocb *iocb, struct socket *sock,
|
||||||
|
|
||||||
if (!(flags & MSG_PEEK)) {
|
if (!(flags & MSG_PEEK)) {
|
||||||
spin_lock_irqsave(&sk->sk_receive_queue.lock, cpu_flags);
|
spin_lock_irqsave(&sk->sk_receive_queue.lock, cpu_flags);
|
||||||
sk_eat_skb(sk, skb, 0);
|
sk_eat_skb(sk, skb, false);
|
||||||
spin_unlock_irqrestore(&sk->sk_receive_queue.lock, cpu_flags);
|
spin_unlock_irqrestore(&sk->sk_receive_queue.lock, cpu_flags);
|
||||||
*seq = 0;
|
*seq = 0;
|
||||||
}
|
}
|
||||||
|
@ -864,7 +864,7 @@ copy_uaddr:
|
||||||
|
|
||||||
if (!(flags & MSG_PEEK)) {
|
if (!(flags & MSG_PEEK)) {
|
||||||
spin_lock_irqsave(&sk->sk_receive_queue.lock, cpu_flags);
|
spin_lock_irqsave(&sk->sk_receive_queue.lock, cpu_flags);
|
||||||
sk_eat_skb(sk, skb, 0);
|
sk_eat_skb(sk, skb, false);
|
||||||
spin_unlock_irqrestore(&sk->sk_receive_queue.lock, cpu_flags);
|
spin_unlock_irqrestore(&sk->sk_receive_queue.lock, cpu_flags);
|
||||||
*seq = 0;
|
*seq = 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1543,9 +1543,8 @@ retry:
|
||||||
skb->dev = dev;
|
skb->dev = dev;
|
||||||
skb->priority = sk->sk_priority;
|
skb->priority = sk->sk_priority;
|
||||||
skb->mark = sk->sk_mark;
|
skb->mark = sk->sk_mark;
|
||||||
err = sock_tx_timestamp(sk, &skb_shinfo(skb)->tx_flags);
|
|
||||||
if (err < 0)
|
sock_tx_timestamp(sk, &skb_shinfo(skb)->tx_flags);
|
||||||
goto out_unlock;
|
|
||||||
|
|
||||||
if (unlikely(extra_len == 4))
|
if (unlikely(extra_len == 4))
|
||||||
skb->no_fcs = 1;
|
skb->no_fcs = 1;
|
||||||
|
@ -2327,9 +2326,8 @@ static int packet_snd(struct socket *sock,
|
||||||
err = skb_copy_datagram_from_iovec(skb, offset, msg->msg_iov, 0, len);
|
err = skb_copy_datagram_from_iovec(skb, offset, msg->msg_iov, 0, len);
|
||||||
if (err)
|
if (err)
|
||||||
goto out_free;
|
goto out_free;
|
||||||
err = sock_tx_timestamp(sk, &skb_shinfo(skb)->tx_flags);
|
|
||||||
if (err < 0)
|
sock_tx_timestamp(sk, &skb_shinfo(skb)->tx_flags);
|
||||||
goto out_free;
|
|
||||||
|
|
||||||
if (!gso_type && (len > dev->mtu + reserve + extra_len)) {
|
if (!gso_type && (len > dev->mtu + reserve + extra_len)) {
|
||||||
/* Earlier code assumed this would be a VLAN pkt,
|
/* Earlier code assumed this would be a VLAN pkt,
|
||||||
|
|
100
net/socket.c
100
net/socket.c
|
@ -88,6 +88,7 @@
|
||||||
#include <linux/nsproxy.h>
|
#include <linux/nsproxy.h>
|
||||||
#include <linux/magic.h>
|
#include <linux/magic.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
|
#include <linux/xattr.h>
|
||||||
|
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
#include <asm/unistd.h>
|
#include <asm/unistd.h>
|
||||||
|
@ -347,7 +348,8 @@ static struct file_system_type sock_fs_type = {
|
||||||
* but we take care of internal coherence yet.
|
* but we take care of internal coherence yet.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static int sock_alloc_file(struct socket *sock, struct file **f, int flags)
|
static int sock_alloc_file(struct socket *sock, struct file **f, int flags,
|
||||||
|
const char *dname)
|
||||||
{
|
{
|
||||||
struct qstr name = { .name = "" };
|
struct qstr name = { .name = "" };
|
||||||
struct path path;
|
struct path path;
|
||||||
|
@ -358,6 +360,13 @@ static int sock_alloc_file(struct socket *sock, struct file **f, int flags)
|
||||||
if (unlikely(fd < 0))
|
if (unlikely(fd < 0))
|
||||||
return fd;
|
return fd;
|
||||||
|
|
||||||
|
if (dname) {
|
||||||
|
name.name = dname;
|
||||||
|
name.len = strlen(name.name);
|
||||||
|
} else if (sock->sk) {
|
||||||
|
name.name = sock->sk->sk_prot_creator->name;
|
||||||
|
name.len = strlen(name.name);
|
||||||
|
}
|
||||||
path.dentry = d_alloc_pseudo(sock_mnt->mnt_sb, &name);
|
path.dentry = d_alloc_pseudo(sock_mnt->mnt_sb, &name);
|
||||||
if (unlikely(!path.dentry)) {
|
if (unlikely(!path.dentry)) {
|
||||||
put_unused_fd(fd);
|
put_unused_fd(fd);
|
||||||
|
@ -390,7 +399,7 @@ static int sock_alloc_file(struct socket *sock, struct file **f, int flags)
|
||||||
int sock_map_fd(struct socket *sock, int flags)
|
int sock_map_fd(struct socket *sock, int flags)
|
||||||
{
|
{
|
||||||
struct file *newfile;
|
struct file *newfile;
|
||||||
int fd = sock_alloc_file(sock, &newfile, flags);
|
int fd = sock_alloc_file(sock, &newfile, flags, NULL);
|
||||||
|
|
||||||
if (likely(fd >= 0))
|
if (likely(fd >= 0))
|
||||||
fd_install(fd, newfile);
|
fd_install(fd, newfile);
|
||||||
|
@ -455,6 +464,82 @@ static struct socket *sockfd_lookup_light(int fd, int *err, int *fput_needed)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define XATTR_SOCKPROTONAME_SUFFIX "sockprotoname"
|
||||||
|
#define XATTR_NAME_SOCKPROTONAME (XATTR_SYSTEM_PREFIX XATTR_SOCKPROTONAME_SUFFIX)
|
||||||
|
#define XATTR_NAME_SOCKPROTONAME_LEN (sizeof(XATTR_NAME_SOCKPROTONAME)-1)
|
||||||
|
static ssize_t sockfs_getxattr(struct dentry *dentry,
|
||||||
|
const char *name, void *value, size_t size)
|
||||||
|
{
|
||||||
|
const char *proto_name;
|
||||||
|
size_t proto_size;
|
||||||
|
int error;
|
||||||
|
|
||||||
|
error = -ENODATA;
|
||||||
|
if (!strncmp(name, XATTR_NAME_SOCKPROTONAME, XATTR_NAME_SOCKPROTONAME_LEN)) {
|
||||||
|
proto_name = dentry->d_name.name;
|
||||||
|
proto_size = strlen(proto_name);
|
||||||
|
|
||||||
|
if (value) {
|
||||||
|
error = -ERANGE;
|
||||||
|
if (proto_size + 1 > size)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
strncpy(value, proto_name, proto_size + 1);
|
||||||
|
}
|
||||||
|
error = proto_size + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t sockfs_listxattr(struct dentry *dentry, char *buffer,
|
||||||
|
size_t size)
|
||||||
|
{
|
||||||
|
ssize_t len;
|
||||||
|
ssize_t used = 0;
|
||||||
|
|
||||||
|
len = security_inode_listsecurity(dentry->d_inode, buffer, size);
|
||||||
|
if (len < 0)
|
||||||
|
return len;
|
||||||
|
used += len;
|
||||||
|
if (buffer) {
|
||||||
|
if (size < used)
|
||||||
|
return -ERANGE;
|
||||||
|
buffer += len;
|
||||||
|
}
|
||||||
|
|
||||||
|
len = (XATTR_NAME_SOCKPROTONAME_LEN + 1);
|
||||||
|
used += len;
|
||||||
|
if (buffer) {
|
||||||
|
if (size < used)
|
||||||
|
return -ERANGE;
|
||||||
|
memcpy(buffer, XATTR_NAME_SOCKPROTONAME, len);
|
||||||
|
buffer += len;
|
||||||
|
}
|
||||||
|
|
||||||
|
return used;
|
||||||
|
}
|
||||||
|
|
||||||
|
int sockfs_setattr(struct dentry *dentry, struct iattr *iattr)
|
||||||
|
{
|
||||||
|
int err = simple_setattr(dentry, iattr);
|
||||||
|
|
||||||
|
if (!err && (iattr->ia_valid & ATTR_UID)) {
|
||||||
|
struct socket *sock = SOCKET_I(dentry->d_inode);
|
||||||
|
|
||||||
|
sock->sk->sk_uid = iattr->ia_uid;
|
||||||
|
}
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct inode_operations sockfs_inode_ops = {
|
||||||
|
.getxattr = sockfs_getxattr,
|
||||||
|
.listxattr = sockfs_listxattr,
|
||||||
|
.setattr = sockfs_setattr,
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* sock_alloc - allocate a socket
|
* sock_alloc - allocate a socket
|
||||||
*
|
*
|
||||||
|
@ -479,6 +564,7 @@ static struct socket *sock_alloc(void)
|
||||||
inode->i_mode = S_IFSOCK | S_IRWXUGO;
|
inode->i_mode = S_IFSOCK | S_IRWXUGO;
|
||||||
inode->i_uid = current_fsuid();
|
inode->i_uid = current_fsuid();
|
||||||
inode->i_gid = current_fsgid();
|
inode->i_gid = current_fsgid();
|
||||||
|
inode->i_op = &sockfs_inode_ops;
|
||||||
|
|
||||||
percpu_add(sockets_in_use, 1);
|
percpu_add(sockets_in_use, 1);
|
||||||
return sock;
|
return sock;
|
||||||
|
@ -535,7 +621,7 @@ void sock_release(struct socket *sock)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(sock_release);
|
EXPORT_SYMBOL(sock_release);
|
||||||
|
|
||||||
int sock_tx_timestamp(struct sock *sk, __u8 *tx_flags)
|
void sock_tx_timestamp(struct sock *sk, __u8 *tx_flags)
|
||||||
{
|
{
|
||||||
*tx_flags = 0;
|
*tx_flags = 0;
|
||||||
if (sock_flag(sk, SOCK_TIMESTAMPING_TX_HARDWARE))
|
if (sock_flag(sk, SOCK_TIMESTAMPING_TX_HARDWARE))
|
||||||
|
@ -544,7 +630,6 @@ int sock_tx_timestamp(struct sock *sk, __u8 *tx_flags)
|
||||||
*tx_flags |= SKBTX_SW_TSTAMP;
|
*tx_flags |= SKBTX_SW_TSTAMP;
|
||||||
if (sock_flag(sk, SOCK_WIFI_STATUS))
|
if (sock_flag(sk, SOCK_WIFI_STATUS))
|
||||||
*tx_flags |= SKBTX_WIFI_STATUS;
|
*tx_flags |= SKBTX_WIFI_STATUS;
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(sock_tx_timestamp);
|
EXPORT_SYMBOL(sock_tx_timestamp);
|
||||||
|
|
||||||
|
@ -1397,13 +1482,13 @@ SYSCALL_DEFINE4(socketpair, int, family, int, type, int, protocol,
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
goto out_release_both;
|
goto out_release_both;
|
||||||
|
|
||||||
fd1 = sock_alloc_file(sock1, &newfile1, flags);
|
fd1 = sock_alloc_file(sock1, &newfile1, flags, NULL);
|
||||||
if (unlikely(fd1 < 0)) {
|
if (unlikely(fd1 < 0)) {
|
||||||
err = fd1;
|
err = fd1;
|
||||||
goto out_release_both;
|
goto out_release_both;
|
||||||
}
|
}
|
||||||
|
|
||||||
fd2 = sock_alloc_file(sock2, &newfile2, flags);
|
fd2 = sock_alloc_file(sock2, &newfile2, flags, NULL);
|
||||||
if (unlikely(fd2 < 0)) {
|
if (unlikely(fd2 < 0)) {
|
||||||
err = fd2;
|
err = fd2;
|
||||||
fput(newfile1);
|
fput(newfile1);
|
||||||
|
@ -1539,7 +1624,8 @@ SYSCALL_DEFINE4(accept4, int, fd, struct sockaddr __user *, upeer_sockaddr,
|
||||||
*/
|
*/
|
||||||
__module_get(newsock->ops->owner);
|
__module_get(newsock->ops->owner);
|
||||||
|
|
||||||
newfd = sock_alloc_file(newsock, &newfile, flags);
|
newfd = sock_alloc_file(newsock, &newfile, flags,
|
||||||
|
sock->sk->sk_prot_creator->name);
|
||||||
if (unlikely(newfd < 0)) {
|
if (unlikely(newfd < 0)) {
|
||||||
err = newfd;
|
err = newfd;
|
||||||
sock_release(newsock);
|
sock_release(newsock);
|
||||||
|
|
Loading…
Reference in a new issue