mirror of
https://sourceware.org/git/glibc.git
synced 2024-11-21 12:30:06 +00:00
aligned_alloc: conform to C17
This patch adds the strict checking for power-of-two alignments in aligned_alloc(), and updates the manual accordingly. Reviewed-by: Carlos O'Donell <carlos@redhat.com>
This commit is contained in:
parent
cea74a4a24
commit
d1417176a3
@ -44,11 +44,13 @@ tests := mallocbug tst-malloc tst-valloc tst-calloc tst-obstack \
|
|||||||
tst-safe-linking \
|
tst-safe-linking \
|
||||||
tst-mallocalign1 \
|
tst-mallocalign1 \
|
||||||
tst-memalign-2 \
|
tst-memalign-2 \
|
||||||
tst-memalign-3
|
tst-memalign-3 \
|
||||||
|
tst-aligned-alloc
|
||||||
|
|
||||||
tests-static := \
|
tests-static := \
|
||||||
tst-interpose-static-nothread \
|
tst-interpose-static-nothread \
|
||||||
tst-interpose-static-thread
|
tst-interpose-static-thread \
|
||||||
|
tst-aligned-alloc-static
|
||||||
|
|
||||||
# Test for the malloc_set_state symbol removed in glibc 2.25.
|
# Test for the malloc_set_state symbol removed in glibc 2.25.
|
||||||
ifeq ($(have-GLIBC_2.23)$(build-shared),yesyes)
|
ifeq ($(have-GLIBC_2.23)$(build-shared),yesyes)
|
||||||
|
@ -299,7 +299,14 @@ __debug_memalign (size_t alignment, size_t bytes)
|
|||||||
return _debug_mid_memalign (alignment, bytes, RETURN_ADDRESS (0));
|
return _debug_mid_memalign (alignment, bytes, RETURN_ADDRESS (0));
|
||||||
}
|
}
|
||||||
strong_alias (__debug_memalign, memalign)
|
strong_alias (__debug_memalign, memalign)
|
||||||
strong_alias (__debug_memalign, aligned_alloc)
|
static void *
|
||||||
|
__debug_aligned_alloc (size_t alignment, size_t bytes)
|
||||||
|
{
|
||||||
|
if (!powerof2 (alignment) || alignment == 0)
|
||||||
|
return NULL;
|
||||||
|
return _debug_mid_memalign (alignment, bytes, RETURN_ADDRESS (0));
|
||||||
|
}
|
||||||
|
strong_alias (__debug_aligned_alloc, aligned_alloc)
|
||||||
|
|
||||||
static void *
|
static void *
|
||||||
__debug_pvalloc (size_t bytes)
|
__debug_pvalloc (size_t bytes)
|
||||||
|
@ -3528,6 +3528,29 @@ __libc_memalign (size_t alignment, size_t bytes)
|
|||||||
void *address = RETURN_ADDRESS (0);
|
void *address = RETURN_ADDRESS (0);
|
||||||
return _mid_memalign (alignment, bytes, address);
|
return _mid_memalign (alignment, bytes, address);
|
||||||
}
|
}
|
||||||
|
libc_hidden_def (__libc_memalign)
|
||||||
|
|
||||||
|
/* For ISO C17. */
|
||||||
|
void *
|
||||||
|
weak_function
|
||||||
|
aligned_alloc (size_t alignment, size_t bytes)
|
||||||
|
{
|
||||||
|
if (!__malloc_initialized)
|
||||||
|
ptmalloc_init ();
|
||||||
|
|
||||||
|
/* Similar to memalign, but starting with ISO C17 the standard
|
||||||
|
requires an error for alignments that are not supported by the
|
||||||
|
implementation. Valid alignments for the current implementation
|
||||||
|
are non-negative powers of two. */
|
||||||
|
if (!powerof2 (alignment) || alignment == 0)
|
||||||
|
{
|
||||||
|
__set_errno (EINVAL);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void *address = RETURN_ADDRESS (0);
|
||||||
|
return _mid_memalign (alignment, bytes, address);
|
||||||
|
}
|
||||||
|
|
||||||
static void *
|
static void *
|
||||||
_mid_memalign (size_t alignment, size_t bytes, void *address)
|
_mid_memalign (size_t alignment, size_t bytes, void *address)
|
||||||
@ -3618,9 +3641,6 @@ _mid_memalign (size_t alignment, size_t bytes, void *address)
|
|||||||
ar_ptr == arena_for_chunk (mem2chunk (p)));
|
ar_ptr == arena_for_chunk (mem2chunk (p)));
|
||||||
return tag_new_usable (p);
|
return tag_new_usable (p);
|
||||||
}
|
}
|
||||||
/* For ISO C11. */
|
|
||||||
weak_alias (__libc_memalign, aligned_alloc)
|
|
||||||
libc_hidden_def (__libc_memalign)
|
|
||||||
|
|
||||||
void *
|
void *
|
||||||
__libc_valloc (size_t bytes)
|
__libc_valloc (size_t bytes)
|
||||||
|
1
malloc/tst-aligned-alloc-static.c
Normal file
1
malloc/tst-aligned-alloc-static.c
Normal file
@ -0,0 +1 @@
|
|||||||
|
#include "tst-aligned-alloc.c"
|
80
malloc/tst-aligned-alloc.c
Normal file
80
malloc/tst-aligned-alloc.c
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
/* Test for C17 alignment requirements.
|
||||||
|
Copyright (C) 2023 Free Software Foundation, Inc.
|
||||||
|
This file is part of the GNU C Library.
|
||||||
|
|
||||||
|
The GNU C Library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2.1 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
The GNU C Library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with the GNU C Library; if not, see
|
||||||
|
<https://www.gnu.org/licenses/>. */
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <libc-diag.h>
|
||||||
|
#include <support/check.h>
|
||||||
|
|
||||||
|
static int
|
||||||
|
do_test (void)
|
||||||
|
{
|
||||||
|
void *p1;
|
||||||
|
void *p2;
|
||||||
|
void *p3;
|
||||||
|
void *p4;
|
||||||
|
void *p5;
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
|
||||||
|
/* The implementation supports alignments that are non-negative powers of 2.
|
||||||
|
We test 5 distinct conditions here:
|
||||||
|
- A non-negative power of 2 alignment e.g. 64.
|
||||||
|
- A degenerate zero power of 2 alignment e.g. 1.
|
||||||
|
- A non-power-of-2 alignment e.g. 65.
|
||||||
|
- A zero alignment.
|
||||||
|
- A corner case SIZE_MAX / 2 + 1 alignment.
|
||||||
|
*/
|
||||||
|
|
||||||
|
p1 = aligned_alloc (64, 64);
|
||||||
|
|
||||||
|
if (p1 == NULL)
|
||||||
|
FAIL_EXIT1 ("aligned_alloc(64, 64) failed");
|
||||||
|
|
||||||
|
p2 = aligned_alloc (1, 64);
|
||||||
|
|
||||||
|
if (p2 == NULL)
|
||||||
|
FAIL_EXIT1 ("aligned_alloc(1, 64) failed");
|
||||||
|
|
||||||
|
p3 = aligned_alloc (65, 64);
|
||||||
|
|
||||||
|
if (p3 != NULL)
|
||||||
|
FAIL_EXIT1 ("aligned_alloc(65, 64) did not fail");
|
||||||
|
|
||||||
|
p4 = aligned_alloc (0, 64);
|
||||||
|
|
||||||
|
if (p4 != NULL)
|
||||||
|
FAIL_EXIT1 ("aligned_alloc(0, 64) did not fail");
|
||||||
|
|
||||||
|
/* This is an alignment like 0x80000000...UL */
|
||||||
|
p5 = aligned_alloc (SIZE_MAX / 2 + 1, 64);
|
||||||
|
|
||||||
|
if (p5 != NULL)
|
||||||
|
FAIL_EXIT1 ("aligned_alloc(SIZE_MAX/2+1, 64) did not fail");
|
||||||
|
|
||||||
|
free (p1);
|
||||||
|
free (p2);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define TEST_FUNCTION do_test ()
|
||||||
|
#include "../test-skeleton.c"
|
@ -995,7 +995,7 @@ power of two than that, use @code{aligned_alloc} or @code{posix_memalign}.
|
|||||||
@c Alias to memalign.
|
@c Alias to memalign.
|
||||||
The @code{aligned_alloc} function allocates a block of @var{size} bytes whose
|
The @code{aligned_alloc} function allocates a block of @var{size} bytes whose
|
||||||
address is a multiple of @var{alignment}. The @var{alignment} must be a
|
address is a multiple of @var{alignment}. The @var{alignment} must be a
|
||||||
power of two and @var{size} must be a multiple of @var{alignment}.
|
power of two.
|
||||||
|
|
||||||
The @code{aligned_alloc} function returns a null pointer on error and sets
|
The @code{aligned_alloc} function returns a null pointer on error and sets
|
||||||
@code{errno} to one of the following values:
|
@code{errno} to one of the following values:
|
||||||
|
Loading…
Reference in New Issue
Block a user