x86-64 strncat: Properly handle the length parameter [BZ# 24097]
[glibc.git] / misc / sbrk.c
bloba5bd025a80949ea6c4a1372011628bbf6b297466
1 /* Copyright (C) 1991-2022 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, see
16 <https://www.gnu.org/licenses/>. */
18 /* Mark symbols hidden in static PIE for early self relocation to work. */
19 #if BUILD_PIE_DEFAULT
20 # pragma GCC visibility push(hidden)
21 #endif
22 #include <errno.h>
23 #include <libc-internal.h>
24 #include <stdbool.h>
25 #include <stdint.h>
26 #include <unistd.h>
28 /* Defined in brk.c. */
29 extern void *__curbrk;
30 extern int __brk (void *addr);
32 /* Extend the process's data space by INCREMENT.
33 If INCREMENT is negative, shrink data space by - INCREMENT.
34 Return start of new space allocated, or -1 for errors. */
35 void *
36 __sbrk (intptr_t increment)
38 /* Controls whether __brk (0) is called to read the brk value from
39 the kernel. */
40 bool update_brk = __curbrk == NULL;
42 #if defined (SHARED) && ! IS_IN (rtld)
43 if (!__libc_initial)
45 if (increment != 0)
47 /* Do not allow changing the brk from an inner libc because
48 it cannot be synchronized with the outer libc's brk. */
49 __set_errno (ENOMEM);
50 return (void *) -1;
52 /* Querying the kernel's brk value from an inner namespace is
53 fine. */
54 update_brk = true;
56 #endif
58 if (update_brk)
59 if (__brk (0) < 0) /* Initialize the break. */
60 return (void *) -1;
62 if (increment == 0)
63 return __curbrk;
65 void *oldbrk = __curbrk;
66 if (increment > 0
67 ? ((uintptr_t) oldbrk + (uintptr_t) increment < (uintptr_t) oldbrk)
68 : ((uintptr_t) oldbrk < (uintptr_t) -increment))
70 __set_errno (ENOMEM);
71 return (void *) -1;
74 if (__brk (oldbrk + increment) < 0)
75 return (void *) -1;
77 return oldbrk;
79 libc_hidden_def (__sbrk)
80 weak_alias (__sbrk, sbrk)