aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/linux/slab.h3
-rw-r--r--mm/slab.c49
-rw-r--r--mm/slob.c35
3 files changed, 84 insertions, 3 deletions
diff --git a/include/linux/slab.h b/include/linux/slab.h
index 1ef822e31c77..2f8f60ff294a 100644
--- a/include/linux/slab.h
+++ b/include/linux/slab.h
@@ -72,8 +72,9 @@ static inline void *kmem_cache_alloc_node(struct kmem_cache *cachep,
72 */ 72 */
73void *__kmalloc(size_t, gfp_t); 73void *__kmalloc(size_t, gfp_t);
74void *__kzalloc(size_t, gfp_t); 74void *__kzalloc(size_t, gfp_t);
75void * __must_check krealloc(const void *, size_t, gfp_t);
75void kfree(const void *); 76void kfree(const void *);
76unsigned int ksize(const void *); 77size_t ksize(const void *);
77 78
78/** 79/**
79 * kcalloc - allocate memory for an array. The memory is set to zero. 80 * kcalloc - allocate memory for an array. The memory is set to zero.
diff --git a/mm/slab.c b/mm/slab.c
index 168bfe9d8ffe..8b71a9c3daa4 100644
--- a/mm/slab.c
+++ b/mm/slab.c
@@ -3740,6 +3740,53 @@ EXPORT_SYMBOL(__kmalloc);
3740#endif 3740#endif
3741 3741
3742/** 3742/**
3743 * krealloc - reallocate memory. The contents will remain unchanged.
3744 *
3745 * @p: object to reallocate memory for.
3746 * @new_size: how many bytes of memory are required.
3747 * @flags: the type of memory to allocate.
3748 *
3749 * The contents of the object pointed to are preserved up to the
3750 * lesser of the new and old sizes. If @p is %NULL, krealloc()
3751 * behaves exactly like kmalloc(). If @size is 0 and @p is not a
3752 * %NULL pointer, the object pointed to is freed.
3753 */
3754void *krealloc(const void *p, size_t new_size, gfp_t flags)
3755{
3756 struct kmem_cache *cache, *new_cache;
3757 void *ret;
3758
3759 if (unlikely(!p))
3760 return kmalloc_track_caller(new_size, flags);
3761
3762 if (unlikely(!new_size)) {
3763 kfree(p);
3764 return NULL;
3765 }
3766
3767 cache = virt_to_cache(p);
3768 new_cache = __find_general_cachep(new_size, flags);
3769
3770 /*
3771 * If new size fits in the current cache, bail out.
3772 */
3773 if (likely(cache == new_cache))
3774 return (void *)p;
3775
3776 /*
3777 * We are on the slow-path here so do not use __cache_alloc
3778 * because it bloats kernel text.
3779 */
3780 ret = kmalloc_track_caller(new_size, flags);
3781 if (ret) {
3782 memcpy(ret, p, min(new_size, ksize(p)));
3783 kfree(p);
3784 }
3785 return ret;
3786}
3787EXPORT_SYMBOL(krealloc);
3788
3789/**
3743 * kmem_cache_free - Deallocate an object 3790 * kmem_cache_free - Deallocate an object
3744 * @cachep: The cache the allocation was from. 3791 * @cachep: The cache the allocation was from.
3745 * @objp: The previously allocated object. 3792 * @objp: The previously allocated object.
@@ -4481,7 +4528,7 @@ const struct seq_operations slabstats_op = {
4481 * allocated with either kmalloc() or kmem_cache_alloc(). The object 4528 * allocated with either kmalloc() or kmem_cache_alloc(). The object
4482 * must not be freed during the duration of the call. 4529 * must not be freed during the duration of the call.
4483 */ 4530 */
4484unsigned int ksize(const void *objp) 4531size_t ksize(const void *objp)
4485{ 4532{
4486 if (unlikely(objp == NULL)) 4533 if (unlikely(objp == NULL))
4487 return 0; 4534 return 0;
diff --git a/mm/slob.c b/mm/slob.c
index 5adc29cb58dd..03cce3d3d986 100644
--- a/mm/slob.c
+++ b/mm/slob.c
@@ -190,6 +190,39 @@ void *__kmalloc(size_t size, gfp_t gfp)
190} 190}
191EXPORT_SYMBOL(__kmalloc); 191EXPORT_SYMBOL(__kmalloc);
192 192
193/**
194 * krealloc - reallocate memory. The contents will remain unchanged.
195 *
196 * @p: object to reallocate memory for.
197 * @new_size: how many bytes of memory are required.
198 * @flags: the type of memory to allocate.
199 *
200 * The contents of the object pointed to are preserved up to the
201 * lesser of the new and old sizes. If @p is %NULL, krealloc()
202 * behaves exactly like kmalloc(). If @size is 0 and @p is not a
203 * %NULL pointer, the object pointed to is freed.
204 */
205void *krealloc(const void *p, size_t new_size, gfp_t flags)
206{
207 void *ret;
208
209 if (unlikely(!p))
210 return kmalloc_track_caller(new_size, flags);
211
212 if (unlikely(!new_size)) {
213 kfree(p);
214 return NULL;
215 }
216
217 ret = kmalloc_track_caller(new_size, flags);
218 if (ret) {
219 memcpy(ret, p, min(new_size, ksize(p)));
220 kfree(p);
221 }
222 return ret;
223}
224EXPORT_SYMBOL(krealloc);
225
193void kfree(const void *block) 226void kfree(const void *block)
194{ 227{
195 bigblock_t *bb, **last = &bigblocks; 228 bigblock_t *bb, **last = &bigblocks;
@@ -219,7 +252,7 @@ void kfree(const void *block)
219 252
220EXPORT_SYMBOL(kfree); 253EXPORT_SYMBOL(kfree);
221 254
222unsigned int ksize(const void *block) 255size_t ksize(const void *block)
223{ 256{
224 bigblock_t *bb; 257 bigblock_t *bb;
225 unsigned long flags; 258 unsigned long flags;