forked from mirrors/linux
		
	xfrm: configure policy hash table thresholds by netlink
Enable to specify local and remote prefix length thresholds for the
policy hash table via a netlink XFRM_MSG_NEWSPDINFO message.
prefix length thresholds are specified by XFRMA_SPD_IPV4_HTHRESH and
XFRMA_SPD_IPV6_HTHRESH optional attributes (struct xfrmu_spdhthresh).
example:
    struct xfrmu_spdhthresh thresh4 = {
        .lbits = 0;
        .rbits = 24;
    };
    struct xfrmu_spdhthresh thresh6 = {
        .lbits = 0;
        .rbits = 56;
    };
    struct nlmsghdr *hdr;
    struct nl_msg *msg;
    msg = nlmsg_alloc();
    hdr = nlmsg_put(msg, NL_AUTO_PORT, NL_AUTO_SEQ, XFRMA_SPD_IPV4_HTHRESH, sizeof(__u32), NLM_F_REQUEST);
    nla_put(msg, XFRMA_SPD_IPV4_HTHRESH, sizeof(thresh4), &thresh4);
    nla_put(msg, XFRMA_SPD_IPV6_HTHRESH, sizeof(thresh6), &thresh6);
    nla_send_auto(sk, msg);
The numbers are the policy selector minimum prefix lengths to put a
policy in the hash table.
- lbits is the local threshold (source address for out policies,
  destination address for in and fwd policies).
- rbits is the remote threshold (destination address for out
  policies, source address for in and fwd policies).
The default values are:
XFRMA_SPD_IPV4_HTHRESH: 32 32
XFRMA_SPD_IPV6_HTHRESH: 128 128
Dynamic re-building of the SPD is performed when the thresholds values
are changed.
The current thresholds can be read via a XFRM_MSG_GETSPDINFO request:
the kernel replies to XFRM_MSG_GETSPDINFO requests by an
XFRM_MSG_NEWSPDINFO message, with both attributes
XFRMA_SPD_IPV4_HTHRESH and XFRMA_SPD_IPV6_HTHRESH.
Signed-off-by: Christophe Gouault <christophe.gouault@6wind.com>
Signed-off-by: Steffen Klassert <steffen.klassert@secunet.com>
			
			
This commit is contained in:
		
							parent
							
								
									b58555f176
								
							
						
					
					
						commit
						880a6fab8f
					
				
					 5 changed files with 182 additions and 3 deletions
				
			
		| 
						 | 
					@ -19,6 +19,15 @@ struct xfrm_policy_hash {
 | 
				
			||||||
	u8			sbits6;
 | 
						u8			sbits6;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct xfrm_policy_hthresh {
 | 
				
			||||||
 | 
						struct work_struct	work;
 | 
				
			||||||
 | 
						seqlock_t		lock;
 | 
				
			||||||
 | 
						u8			lbits4;
 | 
				
			||||||
 | 
						u8			rbits4;
 | 
				
			||||||
 | 
						u8			lbits6;
 | 
				
			||||||
 | 
						u8			rbits6;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct netns_xfrm {
 | 
					struct netns_xfrm {
 | 
				
			||||||
	struct list_head	state_all;
 | 
						struct list_head	state_all;
 | 
				
			||||||
	/*
 | 
						/*
 | 
				
			||||||
| 
						 | 
					@ -45,6 +54,7 @@ struct netns_xfrm {
 | 
				
			||||||
	struct xfrm_policy_hash	policy_bydst[XFRM_POLICY_MAX * 2];
 | 
						struct xfrm_policy_hash	policy_bydst[XFRM_POLICY_MAX * 2];
 | 
				
			||||||
	unsigned int		policy_count[XFRM_POLICY_MAX * 2];
 | 
						unsigned int		policy_count[XFRM_POLICY_MAX * 2];
 | 
				
			||||||
	struct work_struct	policy_hash_work;
 | 
						struct work_struct	policy_hash_work;
 | 
				
			||||||
 | 
						struct xfrm_policy_hthresh policy_hthresh;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	struct sock		*nlsk;
 | 
						struct sock		*nlsk;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1591,6 +1591,7 @@ struct xfrm_policy *xfrm_policy_bysel_ctx(struct net *net, u32 mark,
 | 
				
			||||||
struct xfrm_policy *xfrm_policy_byid(struct net *net, u32 mark, u8, int dir,
 | 
					struct xfrm_policy *xfrm_policy_byid(struct net *net, u32 mark, u8, int dir,
 | 
				
			||||||
				     u32 id, int delete, int *err);
 | 
									     u32 id, int delete, int *err);
 | 
				
			||||||
int xfrm_policy_flush(struct net *net, u8 type, bool task_valid);
 | 
					int xfrm_policy_flush(struct net *net, u8 type, bool task_valid);
 | 
				
			||||||
 | 
					void xfrm_policy_hash_rebuild(struct net *net);
 | 
				
			||||||
u32 xfrm_get_acqseq(void);
 | 
					u32 xfrm_get_acqseq(void);
 | 
				
			||||||
int verify_spi_info(u8 proto, u32 min, u32 max);
 | 
					int verify_spi_info(u8 proto, u32 min, u32 max);
 | 
				
			||||||
int xfrm_alloc_spi(struct xfrm_state *x, u32 minspi, u32 maxspi);
 | 
					int xfrm_alloc_spi(struct xfrm_state *x, u32 minspi, u32 maxspi);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -328,6 +328,8 @@ enum xfrm_spdattr_type_t {
 | 
				
			||||||
	XFRMA_SPD_UNSPEC,
 | 
						XFRMA_SPD_UNSPEC,
 | 
				
			||||||
	XFRMA_SPD_INFO,
 | 
						XFRMA_SPD_INFO,
 | 
				
			||||||
	XFRMA_SPD_HINFO,
 | 
						XFRMA_SPD_HINFO,
 | 
				
			||||||
 | 
						XFRMA_SPD_IPV4_HTHRESH,
 | 
				
			||||||
 | 
						XFRMA_SPD_IPV6_HTHRESH,
 | 
				
			||||||
	__XFRMA_SPD_MAX
 | 
						__XFRMA_SPD_MAX
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define XFRMA_SPD_MAX (__XFRMA_SPD_MAX - 1)
 | 
					#define XFRMA_SPD_MAX (__XFRMA_SPD_MAX - 1)
 | 
				
			||||||
| 
						 | 
					@ -347,6 +349,11 @@ struct xfrmu_spdhinfo {
 | 
				
			||||||
	__u32 spdhmcnt;
 | 
						__u32 spdhmcnt;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct xfrmu_spdhthresh {
 | 
				
			||||||
 | 
						__u8 lbits;
 | 
				
			||||||
 | 
						__u8 rbits;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct xfrm_usersa_info {
 | 
					struct xfrm_usersa_info {
 | 
				
			||||||
	struct xfrm_selector		sel;
 | 
						struct xfrm_selector		sel;
 | 
				
			||||||
	struct xfrm_id			id;
 | 
						struct xfrm_id			id;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -566,6 +566,86 @@ static void xfrm_hash_resize(struct work_struct *work)
 | 
				
			||||||
	mutex_unlock(&hash_resize_mutex);
 | 
						mutex_unlock(&hash_resize_mutex);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void xfrm_hash_rebuild(struct work_struct *work)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct net *net = container_of(work, struct net,
 | 
				
			||||||
 | 
									       xfrm.policy_hthresh.work);
 | 
				
			||||||
 | 
						unsigned int hmask;
 | 
				
			||||||
 | 
						struct xfrm_policy *pol;
 | 
				
			||||||
 | 
						struct xfrm_policy *policy;
 | 
				
			||||||
 | 
						struct hlist_head *chain;
 | 
				
			||||||
 | 
						struct hlist_head *odst;
 | 
				
			||||||
 | 
						struct hlist_node *newpos;
 | 
				
			||||||
 | 
						int i;
 | 
				
			||||||
 | 
						int dir;
 | 
				
			||||||
 | 
						unsigned seq;
 | 
				
			||||||
 | 
						u8 lbits4, rbits4, lbits6, rbits6;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						mutex_lock(&hash_resize_mutex);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* read selector prefixlen thresholds */
 | 
				
			||||||
 | 
						do {
 | 
				
			||||||
 | 
							seq = read_seqbegin(&net->xfrm.policy_hthresh.lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							lbits4 = net->xfrm.policy_hthresh.lbits4;
 | 
				
			||||||
 | 
							rbits4 = net->xfrm.policy_hthresh.rbits4;
 | 
				
			||||||
 | 
							lbits6 = net->xfrm.policy_hthresh.lbits6;
 | 
				
			||||||
 | 
							rbits6 = net->xfrm.policy_hthresh.rbits6;
 | 
				
			||||||
 | 
						} while (read_seqretry(&net->xfrm.policy_hthresh.lock, seq));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						write_lock_bh(&net->xfrm.xfrm_policy_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* reset the bydst and inexact table in all directions */
 | 
				
			||||||
 | 
						for (dir = 0; dir < XFRM_POLICY_MAX * 2; dir++) {
 | 
				
			||||||
 | 
							INIT_HLIST_HEAD(&net->xfrm.policy_inexact[dir]);
 | 
				
			||||||
 | 
							hmask = net->xfrm.policy_bydst[dir].hmask;
 | 
				
			||||||
 | 
							odst = net->xfrm.policy_bydst[dir].table;
 | 
				
			||||||
 | 
							for (i = hmask; i >= 0; i--)
 | 
				
			||||||
 | 
								INIT_HLIST_HEAD(odst + i);
 | 
				
			||||||
 | 
							if ((dir & XFRM_POLICY_MASK) == XFRM_POLICY_OUT) {
 | 
				
			||||||
 | 
								/* dir out => dst = remote, src = local */
 | 
				
			||||||
 | 
								net->xfrm.policy_bydst[dir].dbits4 = rbits4;
 | 
				
			||||||
 | 
								net->xfrm.policy_bydst[dir].sbits4 = lbits4;
 | 
				
			||||||
 | 
								net->xfrm.policy_bydst[dir].dbits6 = rbits6;
 | 
				
			||||||
 | 
								net->xfrm.policy_bydst[dir].sbits6 = lbits6;
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								/* dir in/fwd => dst = local, src = remote */
 | 
				
			||||||
 | 
								net->xfrm.policy_bydst[dir].dbits4 = lbits4;
 | 
				
			||||||
 | 
								net->xfrm.policy_bydst[dir].sbits4 = rbits4;
 | 
				
			||||||
 | 
								net->xfrm.policy_bydst[dir].dbits6 = lbits6;
 | 
				
			||||||
 | 
								net->xfrm.policy_bydst[dir].sbits6 = rbits6;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* re-insert all policies by order of creation */
 | 
				
			||||||
 | 
						list_for_each_entry_reverse(policy, &net->xfrm.policy_all, walk.all) {
 | 
				
			||||||
 | 
							newpos = NULL;
 | 
				
			||||||
 | 
							chain = policy_hash_bysel(net, &policy->selector,
 | 
				
			||||||
 | 
										  policy->family,
 | 
				
			||||||
 | 
										  xfrm_policy_id2dir(policy->index));
 | 
				
			||||||
 | 
							hlist_for_each_entry(pol, chain, bydst) {
 | 
				
			||||||
 | 
								if (policy->priority >= pol->priority)
 | 
				
			||||||
 | 
									newpos = &pol->bydst;
 | 
				
			||||||
 | 
								else
 | 
				
			||||||
 | 
									break;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (newpos)
 | 
				
			||||||
 | 
								hlist_add_behind(&policy->bydst, newpos);
 | 
				
			||||||
 | 
							else
 | 
				
			||||||
 | 
								hlist_add_head(&policy->bydst, chain);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						write_unlock_bh(&net->xfrm.xfrm_policy_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						mutex_unlock(&hash_resize_mutex);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void xfrm_policy_hash_rebuild(struct net *net)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						schedule_work(&net->xfrm.policy_hthresh.work);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EXPORT_SYMBOL(xfrm_policy_hash_rebuild);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Generate new index... KAME seems to generate them ordered by cost
 | 
					/* Generate new index... KAME seems to generate them ordered by cost
 | 
				
			||||||
 * of an absolute inpredictability of ordering of rules. This will not pass. */
 | 
					 * of an absolute inpredictability of ordering of rules. This will not pass. */
 | 
				
			||||||
static u32 xfrm_gen_index(struct net *net, int dir, u32 index)
 | 
					static u32 xfrm_gen_index(struct net *net, int dir, u32 index)
 | 
				
			||||||
| 
						 | 
					@ -2872,9 +2952,16 @@ static int __net_init xfrm_policy_init(struct net *net)
 | 
				
			||||||
		htab->dbits6 = 128;
 | 
							htab->dbits6 = 128;
 | 
				
			||||||
		htab->sbits6 = 128;
 | 
							htab->sbits6 = 128;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						net->xfrm.policy_hthresh.lbits4 = 32;
 | 
				
			||||||
 | 
						net->xfrm.policy_hthresh.rbits4 = 32;
 | 
				
			||||||
 | 
						net->xfrm.policy_hthresh.lbits6 = 128;
 | 
				
			||||||
 | 
						net->xfrm.policy_hthresh.rbits6 = 128;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						seqlock_init(&net->xfrm.policy_hthresh.lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	INIT_LIST_HEAD(&net->xfrm.policy_all);
 | 
						INIT_LIST_HEAD(&net->xfrm.policy_all);
 | 
				
			||||||
	INIT_WORK(&net->xfrm.policy_hash_work, xfrm_hash_resize);
 | 
						INIT_WORK(&net->xfrm.policy_hash_work, xfrm_hash_resize);
 | 
				
			||||||
 | 
						INIT_WORK(&net->xfrm.policy_hthresh.work, xfrm_hash_rebuild);
 | 
				
			||||||
	if (net_eq(net, &init_net))
 | 
						if (net_eq(net, &init_net))
 | 
				
			||||||
		register_netdevice_notifier(&xfrm_dev_notifier);
 | 
							register_netdevice_notifier(&xfrm_dev_notifier);
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -964,7 +964,9 @@ static inline size_t xfrm_spdinfo_msgsize(void)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	return NLMSG_ALIGN(4)
 | 
						return NLMSG_ALIGN(4)
 | 
				
			||||||
	       + nla_total_size(sizeof(struct xfrmu_spdinfo))
 | 
						       + nla_total_size(sizeof(struct xfrmu_spdinfo))
 | 
				
			||||||
	       + nla_total_size(sizeof(struct xfrmu_spdhinfo));
 | 
						       + nla_total_size(sizeof(struct xfrmu_spdhinfo))
 | 
				
			||||||
 | 
						       + nla_total_size(sizeof(struct xfrmu_spdhthresh))
 | 
				
			||||||
 | 
						       + nla_total_size(sizeof(struct xfrmu_spdhthresh));
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int build_spdinfo(struct sk_buff *skb, struct net *net,
 | 
					static int build_spdinfo(struct sk_buff *skb, struct net *net,
 | 
				
			||||||
| 
						 | 
					@ -973,9 +975,11 @@ static int build_spdinfo(struct sk_buff *skb, struct net *net,
 | 
				
			||||||
	struct xfrmk_spdinfo si;
 | 
						struct xfrmk_spdinfo si;
 | 
				
			||||||
	struct xfrmu_spdinfo spc;
 | 
						struct xfrmu_spdinfo spc;
 | 
				
			||||||
	struct xfrmu_spdhinfo sph;
 | 
						struct xfrmu_spdhinfo sph;
 | 
				
			||||||
 | 
						struct xfrmu_spdhthresh spt4, spt6;
 | 
				
			||||||
	struct nlmsghdr *nlh;
 | 
						struct nlmsghdr *nlh;
 | 
				
			||||||
	int err;
 | 
						int err;
 | 
				
			||||||
	u32 *f;
 | 
						u32 *f;
 | 
				
			||||||
 | 
						unsigned lseq;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	nlh = nlmsg_put(skb, portid, seq, XFRM_MSG_NEWSPDINFO, sizeof(u32), 0);
 | 
						nlh = nlmsg_put(skb, portid, seq, XFRM_MSG_NEWSPDINFO, sizeof(u32), 0);
 | 
				
			||||||
	if (nlh == NULL) /* shouldn't really happen ... */
 | 
						if (nlh == NULL) /* shouldn't really happen ... */
 | 
				
			||||||
| 
						 | 
					@ -993,9 +997,22 @@ static int build_spdinfo(struct sk_buff *skb, struct net *net,
 | 
				
			||||||
	sph.spdhcnt = si.spdhcnt;
 | 
						sph.spdhcnt = si.spdhcnt;
 | 
				
			||||||
	sph.spdhmcnt = si.spdhmcnt;
 | 
						sph.spdhmcnt = si.spdhmcnt;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						do {
 | 
				
			||||||
 | 
							lseq = read_seqbegin(&net->xfrm.policy_hthresh.lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							spt4.lbits = net->xfrm.policy_hthresh.lbits4;
 | 
				
			||||||
 | 
							spt4.rbits = net->xfrm.policy_hthresh.rbits4;
 | 
				
			||||||
 | 
							spt6.lbits = net->xfrm.policy_hthresh.lbits6;
 | 
				
			||||||
 | 
							spt6.rbits = net->xfrm.policy_hthresh.rbits6;
 | 
				
			||||||
 | 
						} while (read_seqretry(&net->xfrm.policy_hthresh.lock, lseq));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err = nla_put(skb, XFRMA_SPD_INFO, sizeof(spc), &spc);
 | 
						err = nla_put(skb, XFRMA_SPD_INFO, sizeof(spc), &spc);
 | 
				
			||||||
	if (!err)
 | 
						if (!err)
 | 
				
			||||||
		err = nla_put(skb, XFRMA_SPD_HINFO, sizeof(sph), &sph);
 | 
							err = nla_put(skb, XFRMA_SPD_HINFO, sizeof(sph), &sph);
 | 
				
			||||||
 | 
						if (!err)
 | 
				
			||||||
 | 
							err = nla_put(skb, XFRMA_SPD_IPV4_HTHRESH, sizeof(spt4), &spt4);
 | 
				
			||||||
 | 
						if (!err)
 | 
				
			||||||
 | 
							err = nla_put(skb, XFRMA_SPD_IPV6_HTHRESH, sizeof(spt6), &spt6);
 | 
				
			||||||
	if (err) {
 | 
						if (err) {
 | 
				
			||||||
		nlmsg_cancel(skb, nlh);
 | 
							nlmsg_cancel(skb, nlh);
 | 
				
			||||||
		return err;
 | 
							return err;
 | 
				
			||||||
| 
						 | 
					@ -1004,6 +1021,51 @@ static int build_spdinfo(struct sk_buff *skb, struct net *net,
 | 
				
			||||||
	return nlmsg_end(skb, nlh);
 | 
						return nlmsg_end(skb, nlh);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int xfrm_set_spdinfo(struct sk_buff *skb, struct nlmsghdr *nlh,
 | 
				
			||||||
 | 
								    struct nlattr **attrs)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct net *net = sock_net(skb->sk);
 | 
				
			||||||
 | 
						struct xfrmu_spdhthresh *thresh4 = NULL;
 | 
				
			||||||
 | 
						struct xfrmu_spdhthresh *thresh6 = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* selector prefixlen thresholds to hash policies */
 | 
				
			||||||
 | 
						if (attrs[XFRMA_SPD_IPV4_HTHRESH]) {
 | 
				
			||||||
 | 
							struct nlattr *rta = attrs[XFRMA_SPD_IPV4_HTHRESH];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (nla_len(rta) < sizeof(*thresh4))
 | 
				
			||||||
 | 
								return -EINVAL;
 | 
				
			||||||
 | 
							thresh4 = nla_data(rta);
 | 
				
			||||||
 | 
							if (thresh4->lbits > 32 || thresh4->rbits > 32)
 | 
				
			||||||
 | 
								return -EINVAL;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if (attrs[XFRMA_SPD_IPV6_HTHRESH]) {
 | 
				
			||||||
 | 
							struct nlattr *rta = attrs[XFRMA_SPD_IPV6_HTHRESH];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (nla_len(rta) < sizeof(*thresh6))
 | 
				
			||||||
 | 
								return -EINVAL;
 | 
				
			||||||
 | 
							thresh6 = nla_data(rta);
 | 
				
			||||||
 | 
							if (thresh6->lbits > 128 || thresh6->rbits > 128)
 | 
				
			||||||
 | 
								return -EINVAL;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (thresh4 || thresh6) {
 | 
				
			||||||
 | 
							write_seqlock(&net->xfrm.policy_hthresh.lock);
 | 
				
			||||||
 | 
							if (thresh4) {
 | 
				
			||||||
 | 
								net->xfrm.policy_hthresh.lbits4 = thresh4->lbits;
 | 
				
			||||||
 | 
								net->xfrm.policy_hthresh.rbits4 = thresh4->rbits;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (thresh6) {
 | 
				
			||||||
 | 
								net->xfrm.policy_hthresh.lbits6 = thresh6->lbits;
 | 
				
			||||||
 | 
								net->xfrm.policy_hthresh.rbits6 = thresh6->rbits;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							write_sequnlock(&net->xfrm.policy_hthresh.lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							xfrm_policy_hash_rebuild(net);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int xfrm_get_spdinfo(struct sk_buff *skb, struct nlmsghdr *nlh,
 | 
					static int xfrm_get_spdinfo(struct sk_buff *skb, struct nlmsghdr *nlh,
 | 
				
			||||||
		struct nlattr **attrs)
 | 
							struct nlattr **attrs)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
| 
						 | 
					@ -2274,6 +2336,7 @@ static const int xfrm_msg_min[XFRM_NR_MSGTYPES] = {
 | 
				
			||||||
	[XFRM_MSG_REPORT      - XFRM_MSG_BASE] = XMSGSIZE(xfrm_user_report),
 | 
						[XFRM_MSG_REPORT      - XFRM_MSG_BASE] = XMSGSIZE(xfrm_user_report),
 | 
				
			||||||
	[XFRM_MSG_MIGRATE     - XFRM_MSG_BASE] = XMSGSIZE(xfrm_userpolicy_id),
 | 
						[XFRM_MSG_MIGRATE     - XFRM_MSG_BASE] = XMSGSIZE(xfrm_userpolicy_id),
 | 
				
			||||||
	[XFRM_MSG_GETSADINFO  - XFRM_MSG_BASE] = sizeof(u32),
 | 
						[XFRM_MSG_GETSADINFO  - XFRM_MSG_BASE] = sizeof(u32),
 | 
				
			||||||
 | 
						[XFRM_MSG_NEWSPDINFO  - XFRM_MSG_BASE] = sizeof(u32),
 | 
				
			||||||
	[XFRM_MSG_GETSPDINFO  - XFRM_MSG_BASE] = sizeof(u32),
 | 
						[XFRM_MSG_GETSPDINFO  - XFRM_MSG_BASE] = sizeof(u32),
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2308,10 +2371,17 @@ static const struct nla_policy xfrma_policy[XFRMA_MAX+1] = {
 | 
				
			||||||
	[XFRMA_ADDRESS_FILTER]	= { .len = sizeof(struct xfrm_address_filter) },
 | 
						[XFRMA_ADDRESS_FILTER]	= { .len = sizeof(struct xfrm_address_filter) },
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static const struct nla_policy xfrma_spd_policy[XFRMA_SPD_MAX+1] = {
 | 
				
			||||||
 | 
						[XFRMA_SPD_IPV4_HTHRESH] = { .len = sizeof(struct xfrmu_spdhthresh) },
 | 
				
			||||||
 | 
						[XFRMA_SPD_IPV6_HTHRESH] = { .len = sizeof(struct xfrmu_spdhthresh) },
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static const struct xfrm_link {
 | 
					static const struct xfrm_link {
 | 
				
			||||||
	int (*doit)(struct sk_buff *, struct nlmsghdr *, struct nlattr **);
 | 
						int (*doit)(struct sk_buff *, struct nlmsghdr *, struct nlattr **);
 | 
				
			||||||
	int (*dump)(struct sk_buff *, struct netlink_callback *);
 | 
						int (*dump)(struct sk_buff *, struct netlink_callback *);
 | 
				
			||||||
	int (*done)(struct netlink_callback *);
 | 
						int (*done)(struct netlink_callback *);
 | 
				
			||||||
 | 
						const struct nla_policy *nla_pol;
 | 
				
			||||||
 | 
						int nla_max;
 | 
				
			||||||
} xfrm_dispatch[XFRM_NR_MSGTYPES] = {
 | 
					} xfrm_dispatch[XFRM_NR_MSGTYPES] = {
 | 
				
			||||||
	[XFRM_MSG_NEWSA       - XFRM_MSG_BASE] = { .doit = xfrm_add_sa        },
 | 
						[XFRM_MSG_NEWSA       - XFRM_MSG_BASE] = { .doit = xfrm_add_sa        },
 | 
				
			||||||
	[XFRM_MSG_DELSA       - XFRM_MSG_BASE] = { .doit = xfrm_del_sa        },
 | 
						[XFRM_MSG_DELSA       - XFRM_MSG_BASE] = { .doit = xfrm_del_sa        },
 | 
				
			||||||
| 
						 | 
					@ -2335,6 +2405,9 @@ static const struct xfrm_link {
 | 
				
			||||||
	[XFRM_MSG_GETAE       - XFRM_MSG_BASE] = { .doit = xfrm_get_ae  },
 | 
						[XFRM_MSG_GETAE       - XFRM_MSG_BASE] = { .doit = xfrm_get_ae  },
 | 
				
			||||||
	[XFRM_MSG_MIGRATE     - XFRM_MSG_BASE] = { .doit = xfrm_do_migrate    },
 | 
						[XFRM_MSG_MIGRATE     - XFRM_MSG_BASE] = { .doit = xfrm_do_migrate    },
 | 
				
			||||||
	[XFRM_MSG_GETSADINFO  - XFRM_MSG_BASE] = { .doit = xfrm_get_sadinfo   },
 | 
						[XFRM_MSG_GETSADINFO  - XFRM_MSG_BASE] = { .doit = xfrm_get_sadinfo   },
 | 
				
			||||||
 | 
						[XFRM_MSG_NEWSPDINFO  - XFRM_MSG_BASE] = { .doit = xfrm_set_spdinfo,
 | 
				
			||||||
 | 
											   .nla_pol = xfrma_spd_policy,
 | 
				
			||||||
 | 
											   .nla_max = XFRMA_SPD_MAX },
 | 
				
			||||||
	[XFRM_MSG_GETSPDINFO  - XFRM_MSG_BASE] = { .doit = xfrm_get_spdinfo   },
 | 
						[XFRM_MSG_GETSPDINFO  - XFRM_MSG_BASE] = { .doit = xfrm_get_spdinfo   },
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2371,8 +2444,9 @@ static int xfrm_user_rcv_msg(struct sk_buff *skb, struct nlmsghdr *nlh)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err = nlmsg_parse(nlh, xfrm_msg_min[type], attrs, XFRMA_MAX,
 | 
						err = nlmsg_parse(nlh, xfrm_msg_min[type], attrs,
 | 
				
			||||||
			  xfrma_policy);
 | 
								  link->nla_max ? : XFRMA_MAX,
 | 
				
			||||||
 | 
								  link->nla_pol ? : xfrma_policy);
 | 
				
			||||||
	if (err < 0)
 | 
						if (err < 0)
 | 
				
			||||||
		return err;
 | 
							return err;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue