diff options
author | Pavel Emelyanov <xemul@openvz.org> | 2008-01-22 09:07:25 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2008-01-28 18:10:36 -0500 |
commit | 6ddc082223ef0f73717b4133fa7e648842bbfd02 (patch) | |
tree | 62315ac04e396bc6e68543502fb6aeec50be6662 /net/ipv4 | |
parent | e5a2bb842cd9681d00d4ca963e63e4d3647e66f8 (diff) |
[NETNS][FRAGS]: Make the mem counter per-namespace.
This is also simple, but introduces more changes, since
then mem counter is altered in more places.
Signed-off-by: Pavel Emelyanov <xemul@openvz.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv4')
-rw-r--r-- | net/ipv4/inet_fragment.c | 21 | ||||
-rw-r--r-- | net/ipv4/ip_fragment.c | 29 | ||||
-rw-r--r-- | net/ipv4/proc.c | 2 |
3 files changed, 27 insertions, 25 deletions
diff --git a/net/ipv4/inet_fragment.c b/net/ipv4/inet_fragment.c index 4fec0b911f85..ad79ae0c0264 100644 --- a/net/ipv4/inet_fragment.c +++ b/net/ipv4/inet_fragment.c | |||
@@ -63,8 +63,6 @@ void inet_frags_init(struct inet_frags *f) | |||
63 | f->rnd = (u32) ((num_physpages ^ (num_physpages>>7)) ^ | 63 | f->rnd = (u32) ((num_physpages ^ (num_physpages>>7)) ^ |
64 | (jiffies ^ (jiffies >> 6))); | 64 | (jiffies ^ (jiffies >> 6))); |
65 | 65 | ||
66 | atomic_set(&f->mem, 0); | ||
67 | |||
68 | setup_timer(&f->secret_timer, inet_frag_secret_rebuild, | 66 | setup_timer(&f->secret_timer, inet_frag_secret_rebuild, |
69 | (unsigned long)f); | 67 | (unsigned long)f); |
70 | f->secret_timer.expires = jiffies + f->ctl->secret_interval; | 68 | f->secret_timer.expires = jiffies + f->ctl->secret_interval; |
@@ -75,6 +73,7 @@ EXPORT_SYMBOL(inet_frags_init); | |||
75 | void inet_frags_init_net(struct netns_frags *nf) | 73 | void inet_frags_init_net(struct netns_frags *nf) |
76 | { | 74 | { |
77 | nf->nqueues = 0; | 75 | nf->nqueues = 0; |
76 | atomic_set(&nf->mem, 0); | ||
78 | } | 77 | } |
79 | EXPORT_SYMBOL(inet_frags_init_net); | 78 | EXPORT_SYMBOL(inet_frags_init_net); |
80 | 79 | ||
@@ -107,13 +106,13 @@ void inet_frag_kill(struct inet_frag_queue *fq, struct inet_frags *f) | |||
107 | 106 | ||
108 | EXPORT_SYMBOL(inet_frag_kill); | 107 | EXPORT_SYMBOL(inet_frag_kill); |
109 | 108 | ||
110 | static inline void frag_kfree_skb(struct inet_frags *f, struct sk_buff *skb, | 109 | static inline void frag_kfree_skb(struct netns_frags *nf, struct inet_frags *f, |
111 | int *work) | 110 | struct sk_buff *skb, int *work) |
112 | { | 111 | { |
113 | if (work) | 112 | if (work) |
114 | *work -= skb->truesize; | 113 | *work -= skb->truesize; |
115 | 114 | ||
116 | atomic_sub(skb->truesize, &f->mem); | 115 | atomic_sub(skb->truesize, &nf->mem); |
117 | if (f->skb_free) | 116 | if (f->skb_free) |
118 | f->skb_free(skb); | 117 | f->skb_free(skb); |
119 | kfree_skb(skb); | 118 | kfree_skb(skb); |
@@ -123,22 +122,24 @@ void inet_frag_destroy(struct inet_frag_queue *q, struct inet_frags *f, | |||
123 | int *work) | 122 | int *work) |
124 | { | 123 | { |
125 | struct sk_buff *fp; | 124 | struct sk_buff *fp; |
125 | struct netns_frags *nf; | ||
126 | 126 | ||
127 | BUG_TRAP(q->last_in & COMPLETE); | 127 | BUG_TRAP(q->last_in & COMPLETE); |
128 | BUG_TRAP(del_timer(&q->timer) == 0); | 128 | BUG_TRAP(del_timer(&q->timer) == 0); |
129 | 129 | ||
130 | /* Release all fragment data. */ | 130 | /* Release all fragment data. */ |
131 | fp = q->fragments; | 131 | fp = q->fragments; |
132 | nf = q->net; | ||
132 | while (fp) { | 133 | while (fp) { |
133 | struct sk_buff *xp = fp->next; | 134 | struct sk_buff *xp = fp->next; |
134 | 135 | ||
135 | frag_kfree_skb(f, fp, work); | 136 | frag_kfree_skb(nf, f, fp, work); |
136 | fp = xp; | 137 | fp = xp; |
137 | } | 138 | } |
138 | 139 | ||
139 | if (work) | 140 | if (work) |
140 | *work -= f->qsize; | 141 | *work -= f->qsize; |
141 | atomic_sub(f->qsize, &f->mem); | 142 | atomic_sub(f->qsize, &nf->mem); |
142 | 143 | ||
143 | if (f->destructor) | 144 | if (f->destructor) |
144 | f->destructor(q); | 145 | f->destructor(q); |
@@ -147,12 +148,12 @@ void inet_frag_destroy(struct inet_frag_queue *q, struct inet_frags *f, | |||
147 | } | 148 | } |
148 | EXPORT_SYMBOL(inet_frag_destroy); | 149 | EXPORT_SYMBOL(inet_frag_destroy); |
149 | 150 | ||
150 | int inet_frag_evictor(struct inet_frags *f) | 151 | int inet_frag_evictor(struct netns_frags *nf, struct inet_frags *f) |
151 | { | 152 | { |
152 | struct inet_frag_queue *q; | 153 | struct inet_frag_queue *q; |
153 | int work, evicted = 0; | 154 | int work, evicted = 0; |
154 | 155 | ||
155 | work = atomic_read(&f->mem) - f->ctl->low_thresh; | 156 | work = atomic_read(&nf->mem) - f->ctl->low_thresh; |
156 | while (work > 0) { | 157 | while (work > 0) { |
157 | read_lock(&f->lock); | 158 | read_lock(&f->lock); |
158 | if (list_empty(&f->lru_list)) { | 159 | if (list_empty(&f->lru_list)) { |
@@ -226,7 +227,7 @@ static struct inet_frag_queue *inet_frag_alloc(struct netns_frags *nf, | |||
226 | return NULL; | 227 | return NULL; |
227 | 228 | ||
228 | f->constructor(q, arg); | 229 | f->constructor(q, arg); |
229 | atomic_add(f->qsize, &f->mem); | 230 | atomic_add(f->qsize, &nf->mem); |
230 | setup_timer(&q->timer, f->frag_expire, (unsigned long)q); | 231 | setup_timer(&q->timer, f->frag_expire, (unsigned long)q); |
231 | spin_lock_init(&q->lock); | 232 | spin_lock_init(&q->lock); |
232 | atomic_set(&q->refcnt, 1); | 233 | atomic_set(&q->refcnt, 1); |
diff --git a/net/ipv4/ip_fragment.c b/net/ipv4/ip_fragment.c index cd8c83025b48..4f013343cef7 100644 --- a/net/ipv4/ip_fragment.c +++ b/net/ipv4/ip_fragment.c | |||
@@ -100,9 +100,9 @@ int ip_frag_nqueues(struct net *net) | |||
100 | return net->ipv4.frags.nqueues; | 100 | return net->ipv4.frags.nqueues; |
101 | } | 101 | } |
102 | 102 | ||
103 | int ip_frag_mem(void) | 103 | int ip_frag_mem(struct net *net) |
104 | { | 104 | { |
105 | return atomic_read(&ip4_frags.mem); | 105 | return atomic_read(&net->ipv4.frags.mem); |
106 | } | 106 | } |
107 | 107 | ||
108 | static int ip_frag_reasm(struct ipq *qp, struct sk_buff *prev, | 108 | static int ip_frag_reasm(struct ipq *qp, struct sk_buff *prev, |
@@ -142,11 +142,12 @@ static int ip4_frag_match(struct inet_frag_queue *q, void *a) | |||
142 | } | 142 | } |
143 | 143 | ||
144 | /* Memory Tracking Functions. */ | 144 | /* Memory Tracking Functions. */ |
145 | static __inline__ void frag_kfree_skb(struct sk_buff *skb, int *work) | 145 | static __inline__ void frag_kfree_skb(struct netns_frags *nf, |
146 | struct sk_buff *skb, int *work) | ||
146 | { | 147 | { |
147 | if (work) | 148 | if (work) |
148 | *work -= skb->truesize; | 149 | *work -= skb->truesize; |
149 | atomic_sub(skb->truesize, &ip4_frags.mem); | 150 | atomic_sub(skb->truesize, &nf->mem); |
150 | kfree_skb(skb); | 151 | kfree_skb(skb); |
151 | } | 152 | } |
152 | 153 | ||
@@ -192,11 +193,11 @@ static void ipq_kill(struct ipq *ipq) | |||
192 | /* Memory limiting on fragments. Evictor trashes the oldest | 193 | /* Memory limiting on fragments. Evictor trashes the oldest |
193 | * fragment queue until we are back under the threshold. | 194 | * fragment queue until we are back under the threshold. |
194 | */ | 195 | */ |
195 | static void ip_evictor(void) | 196 | static void ip_evictor(struct net *net) |
196 | { | 197 | { |
197 | int evicted; | 198 | int evicted; |
198 | 199 | ||
199 | evicted = inet_frag_evictor(&ip4_frags); | 200 | evicted = inet_frag_evictor(&net->ipv4.frags, &ip4_frags); |
200 | if (evicted) | 201 | if (evicted) |
201 | IP_ADD_STATS_BH(IPSTATS_MIB_REASMFAILS, evicted); | 202 | IP_ADD_STATS_BH(IPSTATS_MIB_REASMFAILS, evicted); |
202 | } | 203 | } |
@@ -294,7 +295,7 @@ static int ip_frag_reinit(struct ipq *qp) | |||
294 | fp = qp->q.fragments; | 295 | fp = qp->q.fragments; |
295 | do { | 296 | do { |
296 | struct sk_buff *xp = fp->next; | 297 | struct sk_buff *xp = fp->next; |
297 | frag_kfree_skb(fp, NULL); | 298 | frag_kfree_skb(qp->q.net, fp, NULL); |
298 | fp = xp; | 299 | fp = xp; |
299 | } while (fp); | 300 | } while (fp); |
300 | 301 | ||
@@ -431,7 +432,7 @@ static int ip_frag_queue(struct ipq *qp, struct sk_buff *skb) | |||
431 | qp->q.fragments = next; | 432 | qp->q.fragments = next; |
432 | 433 | ||
433 | qp->q.meat -= free_it->len; | 434 | qp->q.meat -= free_it->len; |
434 | frag_kfree_skb(free_it, NULL); | 435 | frag_kfree_skb(qp->q.net, free_it, NULL); |
435 | } | 436 | } |
436 | } | 437 | } |
437 | 438 | ||
@@ -451,7 +452,7 @@ static int ip_frag_queue(struct ipq *qp, struct sk_buff *skb) | |||
451 | } | 452 | } |
452 | qp->q.stamp = skb->tstamp; | 453 | qp->q.stamp = skb->tstamp; |
453 | qp->q.meat += skb->len; | 454 | qp->q.meat += skb->len; |
454 | atomic_add(skb->truesize, &ip4_frags.mem); | 455 | atomic_add(skb->truesize, &qp->q.net->mem); |
455 | if (offset == 0) | 456 | if (offset == 0) |
456 | qp->q.last_in |= FIRST_IN; | 457 | qp->q.last_in |= FIRST_IN; |
457 | 458 | ||
@@ -534,12 +535,12 @@ static int ip_frag_reasm(struct ipq *qp, struct sk_buff *prev, | |||
534 | head->len -= clone->len; | 535 | head->len -= clone->len; |
535 | clone->csum = 0; | 536 | clone->csum = 0; |
536 | clone->ip_summed = head->ip_summed; | 537 | clone->ip_summed = head->ip_summed; |
537 | atomic_add(clone->truesize, &ip4_frags.mem); | 538 | atomic_add(clone->truesize, &qp->q.net->mem); |
538 | } | 539 | } |
539 | 540 | ||
540 | skb_shinfo(head)->frag_list = head->next; | 541 | skb_shinfo(head)->frag_list = head->next; |
541 | skb_push(head, head->data - skb_network_header(head)); | 542 | skb_push(head, head->data - skb_network_header(head)); |
542 | atomic_sub(head->truesize, &ip4_frags.mem); | 543 | atomic_sub(head->truesize, &qp->q.net->mem); |
543 | 544 | ||
544 | for (fp=head->next; fp; fp = fp->next) { | 545 | for (fp=head->next; fp; fp = fp->next) { |
545 | head->data_len += fp->len; | 546 | head->data_len += fp->len; |
@@ -549,7 +550,7 @@ static int ip_frag_reasm(struct ipq *qp, struct sk_buff *prev, | |||
549 | else if (head->ip_summed == CHECKSUM_COMPLETE) | 550 | else if (head->ip_summed == CHECKSUM_COMPLETE) |
550 | head->csum = csum_add(head->csum, fp->csum); | 551 | head->csum = csum_add(head->csum, fp->csum); |
551 | head->truesize += fp->truesize; | 552 | head->truesize += fp->truesize; |
552 | atomic_sub(fp->truesize, &ip4_frags.mem); | 553 | atomic_sub(fp->truesize, &qp->q.net->mem); |
553 | } | 554 | } |
554 | 555 | ||
555 | head->next = NULL; | 556 | head->next = NULL; |
@@ -588,8 +589,8 @@ int ip_defrag(struct sk_buff *skb, u32 user) | |||
588 | 589 | ||
589 | net = skb->dev->nd_net; | 590 | net = skb->dev->nd_net; |
590 | /* Start by cleaning up the memory. */ | 591 | /* Start by cleaning up the memory. */ |
591 | if (atomic_read(&ip4_frags.mem) > ip4_frags_ctl.high_thresh) | 592 | if (atomic_read(&net->ipv4.frags.mem) > ip4_frags_ctl.high_thresh) |
592 | ip_evictor(); | 593 | ip_evictor(net); |
593 | 594 | ||
594 | /* Lookup (or create) queue header */ | 595 | /* Lookup (or create) queue header */ |
595 | if ((qp = ip_find(net, ip_hdr(skb), user)) != NULL) { | 596 | if ((qp = ip_find(net, ip_hdr(skb), user)) != NULL) { |
diff --git a/net/ipv4/proc.c b/net/ipv4/proc.c index bae32808616e..d63474c6b400 100644 --- a/net/ipv4/proc.c +++ b/net/ipv4/proc.c | |||
@@ -62,7 +62,7 @@ static int sockstat_seq_show(struct seq_file *seq, void *v) | |||
62 | seq_printf(seq, "UDPLITE: inuse %d\n", sock_prot_inuse_get(&udplite_prot)); | 62 | seq_printf(seq, "UDPLITE: inuse %d\n", sock_prot_inuse_get(&udplite_prot)); |
63 | seq_printf(seq, "RAW: inuse %d\n", sock_prot_inuse_get(&raw_prot)); | 63 | seq_printf(seq, "RAW: inuse %d\n", sock_prot_inuse_get(&raw_prot)); |
64 | seq_printf(seq, "FRAG: inuse %d memory %d\n", | 64 | seq_printf(seq, "FRAG: inuse %d memory %d\n", |
65 | ip_frag_nqueues(&init_net), ip_frag_mem()); | 65 | ip_frag_nqueues(&init_net), ip_frag_mem(&init_net)); |
66 | return 0; | 66 | return 0; |
67 | } | 67 | } |
68 | 68 | ||