olga / rpms / glibc

Forked from rpms/glibc 5 years ago
Clone

Blame SOURCES/glibc-rh677316-alloc_buffer.patch

51f0aa
Backport of the following upstream commits:
51f0aa
51f0aa
commit d54bb9b1d3fd25779fba2c403003c5097ba9af73
51f0aa
Author: Tulio Magno Quites Machado Filho <tuliom@linux.vnet.ibm.com>
51f0aa
Date:   Mon Jun 26 09:55:41 2017 -0300
51f0aa
51f0aa
    Prevent an implicit int promotion in malloc/tst-alloc_buffer.c
51f0aa
51f0aa
commit 4dd8e7c0ce5ecc7f65e33e60ad2f717b31de32ec
51f0aa
Author: Florian Weimer <fweimer@redhat.com>
51f0aa
Date:   Wed Jun 21 22:43:57 2017 +0200
51f0aa
51f0aa
    Implement allocation buffers for internal use
51f0aa
51f0aa
diff --git a/include/alloc_buffer.h b/include/alloc_buffer.h
51f0aa
new file mode 100644
51f0aa
index 0000000000000000..d668a60d66c5b076
51f0aa
--- /dev/null
51f0aa
+++ b/include/alloc_buffer.h
51f0aa
@@ -0,0 +1,367 @@
51f0aa
+/* Allocation from a fixed-size buffer.
51f0aa
+   Copyright (C) 2017 Free Software Foundation, Inc.
51f0aa
+   This file is part of the GNU C Library.
51f0aa
+
51f0aa
+   The GNU C Library is free software; you can redistribute it and/or
51f0aa
+   modify it under the terms of the GNU Lesser General Public
51f0aa
+   License as published by the Free Software Foundation; either
51f0aa
+   version 2.1 of the License, or (at your option) any later version.
51f0aa
+
51f0aa
+   The GNU C Library is distributed in the hope that it will be useful,
51f0aa
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
51f0aa
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
51f0aa
+   Lesser General Public License for more details.
51f0aa
+
51f0aa
+   You should have received a copy of the GNU Lesser General Public
51f0aa
+   License along with the GNU C Library; if not, see
51f0aa
+   <http://www.gnu.org/licenses/>.  */
51f0aa
+
51f0aa
+/* Allocation buffers are used to carve out sub-allocations from a
51f0aa
+   larger allocation.  Their primary application is in writing NSS
51f0aa
+   modules, which receive a caller-allocated buffer in which they are
51f0aa
+   expected to store variable-length results:
51f0aa
+
51f0aa
+     void *buffer = ...;
51f0aa
+     size_t buffer_size = ...;
51f0aa
+
51f0aa
+     struct alloc_buffer buf = alloc_buffer_create (buffer, buffer_size);
51f0aa
+     result->gr_name = alloc_buffer_copy_string (&buf, name);
51f0aa
+
51f0aa
+     // Allocate a list of group_count groups and copy strings into it.
51f0aa
+     char **group_list = alloc_buffer_alloc_array
51f0aa
+       (&buf, char *, group_count  + 1);
51f0aa
+     if (group_list == NULL)
51f0aa
+       return ...; // Request a larger buffer.
51f0aa
+     for (int i = 0; i < group_count; ++i)
51f0aa
+       group_list[i] = alloc_buffer_copy_string (&buf, group_list_src[i]);
51f0aa
+     group_list[group_count] = NULL;
51f0aa
+     ...
51f0aa
+
51f0aa
+     if (alloc_buffer_has_failed (&buf))
51f0aa
+       return ...; // Request a larger buffer.
51f0aa
+     result->gr_mem = group_list;
51f0aa
+     ...
51f0aa
+
51f0aa
+   Note that it is not necessary to check the results of individual
51f0aa
+   allocation operations if the returned pointer is not dereferenced.
51f0aa
+   Allocation failure is sticky, so one check using
51f0aa
+   alloc_buffer_has_failed at the end covers all previous failures.
51f0aa
+
51f0aa
+   A different use case involves combining multiple heap allocations
51f0aa
+   into a single, large one.  In the following example, an array of
51f0aa
+   doubles and an array of ints is allocated:
51f0aa
+
51f0aa
+     size_t double_array_size = ...;
51f0aa
+     size_t int_array_size = ...;
51f0aa
+
51f0aa
+     void *heap_ptr;
51f0aa
+     struct alloc_buffer buf = alloc_buffer_allocate
51f0aa
+       (double_array_size * sizeof (double) + int_array_size * sizeof (int),
51f0aa
+        &heap_ptr);
51f0aa
+     _Static_assert (__alignof__ (double) >= __alignof__ (int),
51f0aa
+                     "no padding after double array");
51f0aa
+     double *double_array = alloc_buffer_alloc_array
51f0aa
+       (&buf, double, double_array_size);
51f0aa
+     int *int_array = alloc_buffer_alloc_array (&buf, int, int_array_size);
51f0aa
+     if (alloc_buffer_has_failed (&buf))
51f0aa
+       return ...; // Report error.
51f0aa
+     ...
51f0aa
+     free (heap_ptr);
51f0aa
+
51f0aa
+   The advantage over manual coding is that the computation of the
51f0aa
+   allocation size does not need an overflow check.  In case of an
51f0aa
+   overflow, one of the subsequent allocations from the buffer will
51f0aa
+   fail.  The initial size computation is checked for consistency at
51f0aa
+   run time, too.  */
51f0aa
+
51f0aa
+#ifndef _ALLOC_BUFFER_H
51f0aa
+#define _ALLOC_BUFFER_H
51f0aa
+
51f0aa
+#include <inttypes.h>
51f0aa
+#include <stdbool.h>
51f0aa
+#include <stddef.h>
51f0aa
+#include <stdlib.h>
51f0aa
+#include <sys/param.h>
51f0aa
+
51f0aa
+/* struct alloc_buffer objects refer to a region of bytes in memory of a
51f0aa
+   fixed size.  The functions below can be used to allocate single
51f0aa
+   objects and arrays from this memory region, or write to its end.
51f0aa
+   On allocation failure (or if an attempt to write beyond the end of
51f0aa
+   the buffer with one of the copy functions), the buffer enters a
51f0aa
+   failed state.
51f0aa
+
51f0aa
+   struct alloc_buffer objects can be copied.  The backing buffer will
51f0aa
+   be shared, but the current write position will be independent.
51f0aa
+
51f0aa
+   Conceptually, the memory region consists of a current write pointer
51f0aa
+   and a limit, beyond which the write pointer cannot move.  */
51f0aa
+struct alloc_buffer
51f0aa
+{
51f0aa
+  /* uintptr_t is used here to simplify the alignment code, and to
51f0aa
+     avoid issues undefined subtractions if the buffer covers more
51f0aa
+     than half of the address space (which would result in differences
51f0aa
+     which could not be represented as a ptrdiff_t value).  */
51f0aa
+  uintptr_t __alloc_buffer_current;
51f0aa
+  uintptr_t __alloc_buffer_end;
51f0aa
+};
51f0aa
+
51f0aa
+enum
51f0aa
+  {
51f0aa
+    /* The value for the __alloc_buffer_current member which marks the
51f0aa
+       buffer as invalid (together with a zero-length buffer).  */
51f0aa
+    __ALLOC_BUFFER_INVALID_POINTER = 0,
51f0aa
+  };
51f0aa
+
51f0aa
+/* Internal function.  Terminate the process using __libc_fatal.  */
51f0aa
+void __libc_alloc_buffer_create_failure (void *start, size_t size);
51f0aa
+
51f0aa
+/* Create a new allocation buffer.  The byte range from START to START
51f0aa
+   + SIZE - 1 must be valid, and the allocation buffer allocates
51f0aa
+   objects from that range.  If START is NULL (so that SIZE must be
51f0aa
+   0), the buffer is marked as failed immediately.  */
51f0aa
+static inline struct alloc_buffer
51f0aa
+alloc_buffer_create (void *start, size_t size)
51f0aa
+{
51f0aa
+  uintptr_t current = (uintptr_t) start;
51f0aa
+  uintptr_t end = (uintptr_t) start + size;
51f0aa
+  if (end < current)
51f0aa
+    __libc_alloc_buffer_create_failure (start, size);
51f0aa
+  return (struct alloc_buffer) { current, end };
51f0aa
+}
51f0aa
+
51f0aa
+/* Internal function.  See alloc_buffer_allocate below.  */
51f0aa
+struct alloc_buffer __libc_alloc_buffer_allocate (size_t size, void **pptr)
51f0aa
+  __attribute__ ((nonnull (2)));
51f0aa
+
51f0aa
+/* Allocate a buffer of SIZE bytes using malloc.  The returned buffer
51f0aa
+   is in a failed state if malloc fails.  *PPTR points to the start of
51f0aa
+   the buffer and can be used to free it later, after the returned
51f0aa
+   buffer has been freed.  */
51f0aa
+static __always_inline __attribute__ ((nonnull (2)))
51f0aa
+struct alloc_buffer alloc_buffer_allocate (size_t size, void **pptr)
51f0aa
+{
51f0aa
+  return __libc_alloc_buffer_allocate (size, pptr);
51f0aa
+}
51f0aa
+
51f0aa
+/* Mark the buffer as failed.  */
51f0aa
+static inline void __attribute__ ((nonnull (1)))
51f0aa
+alloc_buffer_mark_failed (struct alloc_buffer *buf)
51f0aa
+{
51f0aa
+  buf->__alloc_buffer_current = __ALLOC_BUFFER_INVALID_POINTER;
51f0aa
+  buf->__alloc_buffer_end = __ALLOC_BUFFER_INVALID_POINTER;
51f0aa
+}
51f0aa
+
51f0aa
+/* Return the remaining number of bytes in the buffer.  */
51f0aa
+static __always_inline __attribute__ ((nonnull (1))) size_t
51f0aa
+alloc_buffer_size (const struct alloc_buffer *buf)
51f0aa
+{
51f0aa
+  return buf->__alloc_buffer_end - buf->__alloc_buffer_current;
51f0aa
+}
51f0aa
+
51f0aa
+/* Return true if the buffer has been marked as failed.  */
51f0aa
+static inline bool __attribute__ ((nonnull (1)))
51f0aa
+alloc_buffer_has_failed (const struct alloc_buffer *buf)
51f0aa
+{
51f0aa
+  return buf->__alloc_buffer_current == __ALLOC_BUFFER_INVALID_POINTER;
51f0aa
+}
51f0aa
+
51f0aa
+/* Add a single byte to the buffer (consuming the space for this
51f0aa
+   byte).  Mark the buffer as failed if there is not enough room.  */
51f0aa
+static inline void __attribute__ ((nonnull (1)))
51f0aa
+alloc_buffer_add_byte (struct alloc_buffer *buf, unsigned char b)
51f0aa
+{
51f0aa
+  if (__glibc_likely (buf->__alloc_buffer_current < buf->__alloc_buffer_end))
51f0aa
+    {
51f0aa
+      *(unsigned char *) buf->__alloc_buffer_current = b;
51f0aa
+      ++buf->__alloc_buffer_current;
51f0aa
+    }
51f0aa
+  else
51f0aa
+    alloc_buffer_mark_failed (buf);
51f0aa
+}
51f0aa
+
51f0aa
+/* Obtain a pointer to LENGTH bytes in BUF, and consume these bytes.
51f0aa
+   NULL is returned if there is not enough room, and the buffer is
51f0aa
+   marked as failed, or if the buffer has already failed.
51f0aa
+   (Zero-length allocations from an empty buffer which has not yet
51f0aa
+   failed succeed.)  */
51f0aa
+static inline __attribute__ ((nonnull (1))) void *
51f0aa
+alloc_buffer_alloc_bytes (struct alloc_buffer *buf, size_t length)
51f0aa
+{
51f0aa
+  if (length <= alloc_buffer_size (buf))
51f0aa
+    {
51f0aa
+      void *result = (void *) buf->__alloc_buffer_current;
51f0aa
+      buf->__alloc_buffer_current += length;
51f0aa
+      return result;
51f0aa
+    }
51f0aa
+  else
51f0aa
+    {
51f0aa
+      alloc_buffer_mark_failed (buf);
51f0aa
+      return NULL;
51f0aa
+    }
51f0aa
+}
51f0aa
+
51f0aa
+/* Internal function.  Statically assert that the type size is
51f0aa
+   constant and valid.  */
51f0aa
+static __always_inline size_t
51f0aa
+__alloc_buffer_assert_size (size_t size)
51f0aa
+{
51f0aa
+  if (!__builtin_constant_p (size))
51f0aa
+    {
51f0aa
+      __errordecl (error, "type size is not constant");
51f0aa
+      error ();
51f0aa
+    }
51f0aa
+  else if (size == 0)
51f0aa
+    {
51f0aa
+      __errordecl (error, "type size is zero");
51f0aa
+      error ();
51f0aa
+    }
51f0aa
+  return size;
51f0aa
+}
51f0aa
+
51f0aa
+/* Internal function.  Statically assert that the type alignment is
51f0aa
+   constant and valid.  */
51f0aa
+static __always_inline size_t
51f0aa
+__alloc_buffer_assert_align (size_t align)
51f0aa
+{
51f0aa
+  if (!__builtin_constant_p (align))
51f0aa
+    {
51f0aa
+      __errordecl (error, "type alignment is not constant");
51f0aa
+      error ();
51f0aa
+    }
51f0aa
+  else if (align == 0)
51f0aa
+    {
51f0aa
+      __errordecl (error, "type alignment is zero");
51f0aa
+      error ();
51f0aa
+    }
51f0aa
+  else if (!powerof2 (align))
51f0aa
+    {
51f0aa
+      __errordecl (error, "type alignment is not a power of two");
51f0aa
+      error ();
51f0aa
+    }
51f0aa
+  return align;
51f0aa
+}
51f0aa
+
51f0aa
+/* Internal function.  Obtain a pointer to an object.  */
51f0aa
+static inline __attribute__ ((nonnull (1))) void *
51f0aa
+__alloc_buffer_alloc (struct alloc_buffer *buf, size_t size, size_t align)
51f0aa
+{
51f0aa
+  if (size == 1 && align == 1)
51f0aa
+    return alloc_buffer_alloc_bytes (buf, size);
51f0aa
+
51f0aa
+  size_t current = buf->__alloc_buffer_current;
51f0aa
+  size_t aligned = roundup (current, align);
51f0aa
+  size_t new_current = aligned + size;
51f0aa
+  if (aligned >= current        /* No overflow in align step.  */
51f0aa
+      && new_current >= size    /* No overflow in size computation.  */
51f0aa
+      && new_current <= buf->__alloc_buffer_end) /* Room in buffer.  */
51f0aa
+    {
51f0aa
+      buf->__alloc_buffer_current = new_current;
51f0aa
+      return (void *) aligned;
51f0aa
+    }
51f0aa
+  else
51f0aa
+    {
51f0aa
+      alloc_buffer_mark_failed (buf);
51f0aa
+      return NULL;
51f0aa
+    }
51f0aa
+}
51f0aa
+
51f0aa
+/* Obtain a TYPE * pointer to an object in BUF of TYPE.  Consume these
51f0aa
+   bytes from the buffer.  Return NULL and mark the buffer as failed
51f0aa
+   if if there is not enough room in the buffer, or if the buffer has
51f0aa
+   failed before.  */
51f0aa
+#define alloc_buffer_alloc(buf, type)				\
51f0aa
+  ((type *) __alloc_buffer_alloc				\
51f0aa
+   (buf, __alloc_buffer_assert_size (sizeof (type)),		\
51f0aa
+    __alloc_buffer_assert_align (__alignof__ (type))))
51f0aa
+
51f0aa
+/* Internal function.  Obtain a pointer to an object which is
51f0aa
+   subsequently added.  */
51f0aa
+static inline const __attribute__ ((nonnull (1))) void *
51f0aa
+__alloc_buffer_next (struct alloc_buffer *buf, size_t align)
51f0aa
+{
51f0aa
+  if (align == 1)
51f0aa
+    return (const void *) buf->__alloc_buffer_current;
51f0aa
+
51f0aa
+  size_t current = buf->__alloc_buffer_current;
51f0aa
+  size_t aligned = roundup (current, align);
51f0aa
+  if (aligned >= current        /* No overflow in align step.  */
51f0aa
+      && aligned <= buf->__alloc_buffer_end) /* Room in buffer.  */
51f0aa
+    {
51f0aa
+      buf->__alloc_buffer_current = aligned;
51f0aa
+      return (const void *) aligned;
51f0aa
+    }
51f0aa
+  else
51f0aa
+    {
51f0aa
+      alloc_buffer_mark_failed (buf);
51f0aa
+      return NULL;
51f0aa
+    }
51f0aa
+}
51f0aa
+
51f0aa
+/* Like alloc_buffer_alloc, but do not advance the pointer beyond the
51f0aa
+   object (so a subseqent call to alloc_buffer_next or
51f0aa
+   alloc_buffer_alloc returns the same pointer).  Note that the buffer
51f0aa
+   is still aligned according to the requirements of TYPE.  The effect
51f0aa
+   of this function is similar to allocating a zero-length array from
51f0aa
+   the buffer.  */
51f0aa
+#define alloc_buffer_next(buf, type)				\
51f0aa
+  ((const type *) __alloc_buffer_next				\
51f0aa
+   (buf, __alloc_buffer_assert_align (__alignof__ (type))))
51f0aa
+
51f0aa
+/* Internal function.  Allocate an array.  */
51f0aa
+void * __libc_alloc_buffer_alloc_array (struct alloc_buffer *buf,
51f0aa
+					size_t size, size_t align,
51f0aa
+					size_t count)
51f0aa
+  __attribute__ ((nonnull (1)));
51f0aa
+
51f0aa
+/* Obtain a TYPE * pointer to an array of COUNT objects in BUF of
51f0aa
+   TYPE.  Consume these bytes from the buffer.  Return NULL and mark
51f0aa
+   the buffer as failed if if there is not enough room in the buffer,
51f0aa
+   or if the buffer has failed before.  (Zero-length allocations from
51f0aa
+   an empty buffer which has not yet failed succeed.)  */
51f0aa
+#define alloc_buffer_alloc_array(buf, type, count)       \
51f0aa
+  ((type *) __libc_alloc_buffer_alloc_array		 \
51f0aa
+   (buf, __alloc_buffer_assert_size (sizeof (type)),	 \
51f0aa
+    __alloc_buffer_assert_align (__alignof__ (type)),	 \
51f0aa
+    count))
51f0aa
+
51f0aa
+/* Internal function.  See alloc_buffer_copy_bytes below.  */
51f0aa
+struct alloc_buffer __libc_alloc_buffer_copy_bytes (struct alloc_buffer,
51f0aa
+						    const void *, size_t)
51f0aa
+  __attribute__ ((nonnull (2)));
51f0aa
+
51f0aa
+/* Copy SIZE bytes starting at SRC into the buffer.  If there is not
51f0aa
+   enough room in the buffer, the buffer is marked as failed.  No
51f0aa
+   alignment of the buffer is performed.  */
51f0aa
+static inline __attribute__ ((nonnull (1, 2))) void
51f0aa
+alloc_buffer_copy_bytes (struct alloc_buffer *buf, const void *src, size_t size)
51f0aa
+{
51f0aa
+  *buf = __libc_alloc_buffer_copy_bytes (*buf, src, size);
51f0aa
+}
51f0aa
+
51f0aa
+/* Internal function.  See alloc_buffer_copy_string below.  */
51f0aa
+struct alloc_buffer __libc_alloc_buffer_copy_string (struct alloc_buffer,
51f0aa
+						     const char *)
51f0aa
+  __attribute__ ((nonnull (2)));
51f0aa
+
51f0aa
+/* Copy the string at SRC into the buffer, including its null
51f0aa
+   terminator.  If there is not enough room in the buffer, the buffer
51f0aa
+   is marked as failed.  Return a pointer to the string.  */
51f0aa
+static inline __attribute__ ((nonnull (1, 2))) char *
51f0aa
+alloc_buffer_copy_string (struct alloc_buffer *buf, const char *src)
51f0aa
+{
51f0aa
+  char *result = (char *) buf->__alloc_buffer_current;
51f0aa
+  *buf = __libc_alloc_buffer_copy_string (*buf, src);
51f0aa
+  if (alloc_buffer_has_failed (buf))
51f0aa
+    result = NULL;
51f0aa
+  return result;
51f0aa
+}
51f0aa
+
51f0aa
+#ifndef _ISOMAC
51f0aa
+libc_hidden_proto (__libc_alloc_buffer_alloc_array)
51f0aa
+libc_hidden_proto (__libc_alloc_buffer_allocate)
51f0aa
+libc_hidden_proto (__libc_alloc_buffer_copy_bytes)
51f0aa
+libc_hidden_proto (__libc_alloc_buffer_copy_string)
51f0aa
+libc_hidden_proto (__libc_alloc_buffer_create_failure)
51f0aa
+#endif
51f0aa
+
51f0aa
+#endif /* _ALLOC_BUFFER_H */
51f0aa
diff --git a/malloc/Makefile b/malloc/Makefile
51f0aa
index 63fc3291dcc4077a..db4b1f921d56da32 100644
51f0aa
--- a/malloc/Makefile
51f0aa
+++ b/malloc/Makefile
51f0aa
@@ -37,6 +37,7 @@ tests := mallocbug tst-malloc tst-valloc tst-calloc tst-obstack \
51f0aa
 	 tst-dynarray \
51f0aa
 	 tst-dynarray-fail \
51f0aa
 	 tst-dynarray-at-fail \
51f0aa
+	 tst-alloc_buffer \
51f0aa
 
51f0aa
 tests-static := \
51f0aa
 	 tst-interpose-static-nothread \
51f0aa
@@ -55,6 +56,11 @@ routines = malloc morecore mcheck mtrace obstack \
51f0aa
   dynarray_finalize \
51f0aa
   dynarray_resize \
51f0aa
   dynarray_resize_clear \
51f0aa
+  alloc_buffer_alloc_array \
51f0aa
+  alloc_buffer_allocate \
51f0aa
+  alloc_buffer_copy_bytes  \
51f0aa
+  alloc_buffer_copy_string \
51f0aa
+  alloc_buffer_create_failure \
51f0aa
 
51f0aa
 
51f0aa
 install-lib := libmcheck.a
51f0aa
diff --git a/malloc/Versions b/malloc/Versions
51f0aa
index 16f9dab418a4e3f6..0b7b690bbbdd1c2c 100644
51f0aa
--- a/malloc/Versions
51f0aa
+++ b/malloc/Versions
51f0aa
@@ -79,5 +79,12 @@ libc {
51f0aa
     __libc_dynarray_finalize;
51f0aa
     __libc_dynarray_resize;
51f0aa
     __libc_dynarray_resize_clear;
51f0aa
+
51f0aa
+    # struct alloc_buffer support
51f0aa
+    __libc_alloc_buffer_alloc_array;
51f0aa
+    __libc_alloc_buffer_allocate;
51f0aa
+    __libc_alloc_buffer_copy_bytes;
51f0aa
+    __libc_alloc_buffer_copy_string;
51f0aa
+    __libc_alloc_buffer_create_failure;
51f0aa
   }
51f0aa
 }
51f0aa
diff --git a/malloc/alloc_buffer_alloc_array.c b/malloc/alloc_buffer_alloc_array.c
51f0aa
new file mode 100644
51f0aa
index 0000000000000000..68e14da8dd4e545c
51f0aa
--- /dev/null
51f0aa
+++ b/malloc/alloc_buffer_alloc_array.c
51f0aa
@@ -0,0 +1,47 @@
51f0aa
+/* Array allocation from a fixed-size buffer.
51f0aa
+   Copyright (C) 2017 Free Software Foundation, Inc.
51f0aa
+   This file is part of the GNU C Library.
51f0aa
+
51f0aa
+   The GNU C Library is free software; you can redistribute it and/or
51f0aa
+   modify it under the terms of the GNU Lesser General Public
51f0aa
+   License as published by the Free Software Foundation; either
51f0aa
+   version 2.1 of the License, or (at your option) any later version.
51f0aa
+
51f0aa
+   The GNU C Library is distributed in the hope that it will be useful,
51f0aa
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
51f0aa
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
51f0aa
+   Lesser General Public License for more details.
51f0aa
+
51f0aa
+   You should have received a copy of the GNU Lesser General Public
51f0aa
+   License along with the GNU C Library; if not, see
51f0aa
+   <http://www.gnu.org/licenses/>.  */
51f0aa
+
51f0aa
+#include <alloc_buffer.h>
51f0aa
+#include <malloc-internal.h>
51f0aa
+#include <libc-pointer-arith.h>
51f0aa
+
51f0aa
+void *
51f0aa
+__libc_alloc_buffer_alloc_array (struct alloc_buffer *buf, size_t element_size,
51f0aa
+                                 size_t align, size_t count)
51f0aa
+{
51f0aa
+  size_t current = buf->__alloc_buffer_current;
51f0aa
+  /* The caller asserts that align is a power of two.  */
51f0aa
+  size_t aligned = ALIGN_UP (current, align);
51f0aa
+  size_t size;
51f0aa
+  bool overflow = check_mul_overflow_size_t (element_size, count, &size);
51f0aa
+  size_t new_current = aligned + size;
51f0aa
+  if (!overflow                /* Multiplication did not overflow.  */
51f0aa
+      && aligned >= current    /* No overflow in align step.  */
51f0aa
+      && new_current >= size   /* No overflow in size computation.  */
51f0aa
+      && new_current <= buf->__alloc_buffer_end) /* Room in buffer.  */
51f0aa
+    {
51f0aa
+      buf->__alloc_buffer_current = new_current;
51f0aa
+      return (void *) aligned;
51f0aa
+    }
51f0aa
+  else
51f0aa
+    {
51f0aa
+      alloc_buffer_mark_failed (buf);
51f0aa
+      return NULL;
51f0aa
+    }
51f0aa
+}
51f0aa
+libc_hidden_def (__libc_alloc_buffer_alloc_array)
51f0aa
diff --git a/malloc/alloc_buffer_allocate.c b/malloc/alloc_buffer_allocate.c
51f0aa
new file mode 100644
51f0aa
index 0000000000000000..cbde72b842f0b5d1
51f0aa
--- /dev/null
51f0aa
+++ b/malloc/alloc_buffer_allocate.c
51f0aa
@@ -0,0 +1,36 @@
51f0aa
+/* Allocate a fixed-size allocation buffer using malloc.
51f0aa
+   Copyright (C) 2017 Free Software Foundation, Inc.
51f0aa
+   This file is part of the GNU C Library.
51f0aa
+
51f0aa
+   The GNU C Library is free software; you can redistribute it and/or
51f0aa
+   modify it under the terms of the GNU Lesser General Public
51f0aa
+   License as published by the Free Software Foundation; either
51f0aa
+   version 2.1 of the License, or (at your option) any later version.
51f0aa
+
51f0aa
+   The GNU C Library is distributed in the hope that it will be useful,
51f0aa
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
51f0aa
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
51f0aa
+   Lesser General Public License for more details.
51f0aa
+
51f0aa
+   You should have received a copy of the GNU Lesser General Public
51f0aa
+   License along with the GNU C Library; if not, see
51f0aa
+   <http://www.gnu.org/licenses/>.  */
51f0aa
+
51f0aa
+#include <alloc_buffer.h>
51f0aa
+
51f0aa
+#include <stdlib.h>
51f0aa
+
51f0aa
+struct alloc_buffer
51f0aa
+__libc_alloc_buffer_allocate (size_t size, void **pptr)
51f0aa
+{
51f0aa
+  *pptr = malloc (size);
51f0aa
+  if (*pptr == NULL)
51f0aa
+    return (struct alloc_buffer)
51f0aa
+      {
51f0aa
+        .__alloc_buffer_current = __ALLOC_BUFFER_INVALID_POINTER,
51f0aa
+        .__alloc_buffer_end = __ALLOC_BUFFER_INVALID_POINTER
51f0aa
+      };
51f0aa
+  else
51f0aa
+    return alloc_buffer_create (*pptr, size);
51f0aa
+}
51f0aa
+libc_hidden_def (__libc_alloc_buffer_allocate)
51f0aa
diff --git a/malloc/alloc_buffer_copy_bytes.c b/malloc/alloc_buffer_copy_bytes.c
51f0aa
new file mode 100644
51f0aa
index 0000000000000000..66196f1520c6fc84
51f0aa
--- /dev/null
51f0aa
+++ b/malloc/alloc_buffer_copy_bytes.c
51f0aa
@@ -0,0 +1,34 @@
51f0aa
+/* Copy an array of bytes into the buffer.
51f0aa
+   Copyright (C) 2017 Free Software Foundation, Inc.
51f0aa
+   This file is part of the GNU C Library.
51f0aa
+
51f0aa
+   The GNU C Library is free software; you can redistribute it and/or
51f0aa
+   modify it under the terms of the GNU Lesser General Public
51f0aa
+   License as published by the Free Software Foundation; either
51f0aa
+   version 2.1 of the License, or (at your option) any later version.
51f0aa
+
51f0aa
+   The GNU C Library is distributed in the hope that it will be useful,
51f0aa
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
51f0aa
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
51f0aa
+   Lesser General Public License for more details.
51f0aa
+
51f0aa
+   You should have received a copy of the GNU Lesser General Public
51f0aa
+   License along with the GNU C Library; if not, see
51f0aa
+   <http://www.gnu.org/licenses/>.  */
51f0aa
+
51f0aa
+#include <alloc_buffer.h>
51f0aa
+
51f0aa
+#include <string.h>
51f0aa
+
51f0aa
+/* This function works on a copy of the buffer object, so that it can
51f0aa
+   remain non-addressable in the caller.  */
51f0aa
+struct alloc_buffer
51f0aa
+__libc_alloc_buffer_copy_bytes (struct alloc_buffer buf,
51f0aa
+                                const void *src, size_t len)
51f0aa
+{
51f0aa
+  void *ptr = alloc_buffer_alloc_bytes (&buf, len);
51f0aa
+  if (ptr != NULL)
51f0aa
+    memcpy (ptr, src, len);
51f0aa
+  return buf;
51f0aa
+}
51f0aa
+libc_hidden_def (__libc_alloc_buffer_copy_bytes)
51f0aa
diff --git a/malloc/alloc_buffer_copy_string.c b/malloc/alloc_buffer_copy_string.c
51f0aa
new file mode 100644
51f0aa
index 0000000000000000..77c0023d510b43be
51f0aa
--- /dev/null
51f0aa
+++ b/malloc/alloc_buffer_copy_string.c
51f0aa
@@ -0,0 +1,30 @@
51f0aa
+/* Copy a string into the allocation buffer.
51f0aa
+   Copyright (C) 2017 Free Software Foundation, Inc.
51f0aa
+   This file is part of the GNU C Library.
51f0aa
+
51f0aa
+   The GNU C Library is free software; you can redistribute it and/or
51f0aa
+   modify it under the terms of the GNU Lesser General Public
51f0aa
+   License as published by the Free Software Foundation; either
51f0aa
+   version 2.1 of the License, or (at your option) any later version.
51f0aa
+
51f0aa
+   The GNU C Library is distributed in the hope that it will be useful,
51f0aa
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
51f0aa
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
51f0aa
+   Lesser General Public License for more details.
51f0aa
+
51f0aa
+   You should have received a copy of the GNU Lesser General Public
51f0aa
+   License along with the GNU C Library; if not, see
51f0aa
+   <http://www.gnu.org/licenses/>.  */
51f0aa
+
51f0aa
+#include <alloc_buffer.h>
51f0aa
+
51f0aa
+#include <string.h>
51f0aa
+
51f0aa
+/* This function works on a copy of the buffer object, so that it can
51f0aa
+   remain non-addressable in the caller.  */
51f0aa
+struct alloc_buffer
51f0aa
+__libc_alloc_buffer_copy_string (struct alloc_buffer buf, const char *src)
51f0aa
+{
51f0aa
+  return __libc_alloc_buffer_copy_bytes (buf, src, strlen (src) + 1);
51f0aa
+}
51f0aa
+libc_hidden_def (__libc_alloc_buffer_copy_string)
51f0aa
diff --git a/malloc/alloc_buffer_create_failure.c b/malloc/alloc_buffer_create_failure.c
51f0aa
new file mode 100644
51f0aa
index 0000000000000000..5ffba22f7b1e0dc5
51f0aa
--- /dev/null
51f0aa
+++ b/malloc/alloc_buffer_create_failure.c
51f0aa
@@ -0,0 +1,31 @@
51f0aa
+/* Terminate the process as the result of an invalid allocation buffer.
51f0aa
+   Copyright (C) 2017 Free Software Foundation, Inc.
51f0aa
+   This file is part of the GNU C Library.
51f0aa
+
51f0aa
+   The GNU C Library is free software; you can redistribute it and/or
51f0aa
+   modify it under the terms of the GNU Lesser General Public
51f0aa
+   License as published by the Free Software Foundation; either
51f0aa
+   version 2.1 of the License, or (at your option) any later version.
51f0aa
+
51f0aa
+   The GNU C Library is distributed in the hope that it will be useful,
51f0aa
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
51f0aa
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
51f0aa
+   Lesser General Public License for more details.
51f0aa
+
51f0aa
+   You should have received a copy of the GNU Lesser General Public
51f0aa
+   License along with the GNU C Library; if not, see
51f0aa
+   <http://www.gnu.org/licenses/>.  */
51f0aa
+
51f0aa
+#include <alloc_buffer.h>
51f0aa
+#include <stdio.h>
51f0aa
+
51f0aa
+void
51f0aa
+__libc_alloc_buffer_create_failure (void *start, size_t size)
51f0aa
+{
51f0aa
+  char buf[200];
51f0aa
+  __snprintf (buf, sizeof (buf), "Fatal glibc error: "
51f0aa
+              "invalid allocation buffer of size %zu\n",
51f0aa
+              size);
51f0aa
+  __libc_fatal (buf);
51f0aa
+}
51f0aa
+libc_hidden_def (__libc_alloc_buffer_create_failure)
51f0aa
diff --git a/malloc/tst-alloc_buffer.c b/malloc/tst-alloc_buffer.c
51f0aa
new file mode 100644
51f0aa
index 0000000000000000..9b2bd2046a12c0f2
51f0aa
--- /dev/null
51f0aa
+++ b/malloc/tst-alloc_buffer.c
51f0aa
@@ -0,0 +1,665 @@
51f0aa
+/* Tests for struct alloc_buffer.
51f0aa
+   Copyright (C) 2017 Free Software Foundation, Inc.
51f0aa
+   This file is part of the GNU C Library.
51f0aa
+
51f0aa
+   The GNU C Library is free software; you can redistribute it and/or
51f0aa
+   modify it under the terms of the GNU Lesser General Public
51f0aa
+   License as published by the Free Software Foundation; either
51f0aa
+   version 2.1 of the License, or (at your option) any later version.
51f0aa
+
51f0aa
+   The GNU C Library is distributed in the hope that it will be useful,
51f0aa
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
51f0aa
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
51f0aa
+   Lesser General Public License for more details.
51f0aa
+
51f0aa
+   You should have received a copy of the GNU Lesser General Public
51f0aa
+   License along with the GNU C Library; if not, see
51f0aa
+   <http://www.gnu.org/licenses/>.  */
51f0aa
+
51f0aa
+#include <arpa/inet.h>
51f0aa
+#include <alloc_buffer.h>
51f0aa
+#include <stdio.h>
51f0aa
+#include <stdlib.h>
51f0aa
+#include <string.h>
51f0aa
+#include <support/check.h>
51f0aa
+#include <support/support.h>
51f0aa
+#include <support/test-driver.h>
51f0aa
+
51f0aa
+/* Return true if PTR is sufficiently aligned for TYPE.  */
51f0aa
+#define IS_ALIGNED(ptr, type) \
51f0aa
+  ((((uintptr_t) ptr) & (__alloc_buffer_assert_align (__alignof (type)) - 1)) \
51f0aa
+   == 0)
51f0aa
+
51f0aa
+/* Structure with non-power-of-two size.  */
51f0aa
+struct twelve
51f0aa
+{
51f0aa
+  uint32_t buffer[3] __attribute__ ((aligned (4)));
51f0aa
+};
51f0aa
+_Static_assert (sizeof (struct twelve) == 12, "struct twelve");
51f0aa
+_Static_assert (__alignof__ (struct twelve) == 4, "struct twelve");
51f0aa
+
51f0aa
+/* Check for success obtaining empty arrays.  Does not assume the
51f0aa
+   buffer is empty.  */
51f0aa
+static void
51f0aa
+test_empty_array (struct alloc_buffer refbuf)
51f0aa
+{
51f0aa
+  bool refbuf_failed = alloc_buffer_has_failed (&refbuf);
51f0aa
+  if (test_verbose)
51f0aa
+    printf ("info: %s: current=0x%llx end=0x%llx refbuf_failed=%d\n",
51f0aa
+            __func__, (unsigned long long) refbuf.__alloc_buffer_current,
51f0aa
+            (unsigned long long) refbuf.__alloc_buffer_end, refbuf_failed);
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY ((alloc_buffer_alloc_bytes (&buf, 0) == NULL)
51f0aa
+                 == refbuf_failed);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf) == refbuf_failed);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY ((alloc_buffer_alloc_array (&buf, char, 0) == NULL)
51f0aa
+                 == refbuf_failed);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf) == refbuf_failed);
51f0aa
+  }
51f0aa
+  /* The following tests can fail due to the need for aligning the
51f0aa
+     returned pointer.  */
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    bool expect_failure = refbuf_failed
51f0aa
+      || !IS_ALIGNED (alloc_buffer_next (&buf, void), double);
51f0aa
+    double *ptr = alloc_buffer_alloc_array (&buf, double, 0);
51f0aa
+    TEST_VERIFY (IS_ALIGNED (ptr, double));
51f0aa
+    TEST_VERIFY ((ptr == NULL) == expect_failure);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf) == expect_failure);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    bool expect_failure = refbuf_failed
51f0aa
+      || !IS_ALIGNED (alloc_buffer_next (&buf, void), struct twelve);
51f0aa
+    struct twelve *ptr = alloc_buffer_alloc_array (&buf, struct twelve, 0);
51f0aa
+    TEST_VERIFY (IS_ALIGNED (ptr, struct twelve));
51f0aa
+    TEST_VERIFY ((ptr == NULL) == expect_failure);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf) == expect_failure);
51f0aa
+  }
51f0aa
+}
51f0aa
+
51f0aa
+/* Test allocation of impossibly large arrays.  */
51f0aa
+static void
51f0aa
+test_impossible_array (struct alloc_buffer refbuf)
51f0aa
+{
51f0aa
+  if (test_verbose)
51f0aa
+    printf ("info: %s: current=0x%llx end=0x%llx\n",
51f0aa
+            __func__, (unsigned long long) refbuf.__alloc_buffer_current,
51f0aa
+            (unsigned long long) refbuf.__alloc_buffer_end);
51f0aa
+  static const size_t counts[] =
51f0aa
+    { SIZE_MAX, SIZE_MAX - 1, SIZE_MAX - 2, SIZE_MAX - 3, SIZE_MAX - 4,
51f0aa
+      SIZE_MAX / 2, SIZE_MAX / 2 + 1, SIZE_MAX / 2 - 1, 0};
51f0aa
+
51f0aa
+  for (int i = 0; counts[i] != 0; ++i)
51f0aa
+    {
51f0aa
+      size_t count = counts[i];
51f0aa
+      if (test_verbose)
51f0aa
+        printf ("info: %s: count=%zu\n", __func__, count);
51f0aa
+      {
51f0aa
+        struct alloc_buffer buf = refbuf;
51f0aa
+        TEST_VERIFY (alloc_buffer_alloc_bytes (&buf, count) == NULL);
51f0aa
+        TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+      }
51f0aa
+      {
51f0aa
+        struct alloc_buffer buf = refbuf;
51f0aa
+        TEST_VERIFY (alloc_buffer_alloc_array (&buf, char, count) == NULL);
51f0aa
+        TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+      }
51f0aa
+      {
51f0aa
+        struct alloc_buffer buf = refbuf;
51f0aa
+        TEST_VERIFY (alloc_buffer_alloc_array (&buf, short, count) == NULL);
51f0aa
+        TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+      }
51f0aa
+      {
51f0aa
+        struct alloc_buffer buf = refbuf;
51f0aa
+        TEST_VERIFY (alloc_buffer_alloc_array (&buf, double, count) == NULL);
51f0aa
+        TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+      }
51f0aa
+      {
51f0aa
+        struct alloc_buffer buf = refbuf;
51f0aa
+        TEST_VERIFY (alloc_buffer_alloc_array (&buf, struct twelve, count)
51f0aa
+                     == NULL);
51f0aa
+        TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+      }
51f0aa
+    }
51f0aa
+}
51f0aa
+
51f0aa
+/* Check for failure to obtain anything from a failed buffer.  */
51f0aa
+static void
51f0aa
+test_after_failure (struct alloc_buffer refbuf)
51f0aa
+{
51f0aa
+  if (test_verbose)
51f0aa
+    printf ("info: %s: current=0x%llx end=0x%llx\n",
51f0aa
+            __func__, (unsigned long long) refbuf.__alloc_buffer_current,
51f0aa
+            (unsigned long long) refbuf.__alloc_buffer_end);
51f0aa
+  TEST_VERIFY (alloc_buffer_has_failed (&refbuf));
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    alloc_buffer_add_byte (&buf, 17);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, char) == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, double) == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, struct twelve) == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+  }
51f0aa
+
51f0aa
+  test_impossible_array (refbuf);
51f0aa
+  for (int count = 0; count <= 4; ++count)
51f0aa
+    {
51f0aa
+      {
51f0aa
+        struct alloc_buffer buf = refbuf;
51f0aa
+        TEST_VERIFY (alloc_buffer_alloc_bytes (&buf, count) == NULL);
51f0aa
+        TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+      }
51f0aa
+      {
51f0aa
+        struct alloc_buffer buf = refbuf;
51f0aa
+        TEST_VERIFY (alloc_buffer_alloc_array (&buf, char, count) == NULL);
51f0aa
+        TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+      }
51f0aa
+      {
51f0aa
+        struct alloc_buffer buf = refbuf;
51f0aa
+        TEST_VERIFY (alloc_buffer_alloc_array (&buf, double, count) == NULL);
51f0aa
+        TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+      }
51f0aa
+      {
51f0aa
+        struct alloc_buffer buf = refbuf;
51f0aa
+        TEST_VERIFY (alloc_buffer_alloc_array (&buf, struct twelve, count)
51f0aa
+                     == NULL);
51f0aa
+        TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+      }
51f0aa
+    }
51f0aa
+}
51f0aa
+
51f0aa
+static void
51f0aa
+test_empty (struct alloc_buffer refbuf)
51f0aa
+{
51f0aa
+  TEST_VERIFY (alloc_buffer_size (&refbuf) == 0);
51f0aa
+  if (alloc_buffer_next (&refbuf, void) != NULL)
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&refbuf));
51f0aa
+  test_empty_array (refbuf);
51f0aa
+  test_impossible_array (refbuf);
51f0aa
+
51f0aa
+  /* Failure to obtain non-empty objects.  */
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    alloc_buffer_add_byte (&buf, 17);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, char) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, double) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, struct twelve) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, char, 1) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, double, 1) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, struct twelve, 1) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+}
51f0aa
+
51f0aa
+static void
51f0aa
+test_size_1 (struct alloc_buffer refbuf)
51f0aa
+{
51f0aa
+  TEST_VERIFY (!alloc_buffer_has_failed (&refbuf));
51f0aa
+  TEST_VERIFY (alloc_buffer_size (&refbuf) == 1);
51f0aa
+  test_empty_array (refbuf);
51f0aa
+  test_impossible_array (refbuf);
51f0aa
+
51f0aa
+  /* Success adding a single byte.  */
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    alloc_buffer_add_byte (&buf, 17);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    test_empty (buf);
51f0aa
+  }
51f0aa
+  TEST_VERIFY (memcmp (alloc_buffer_next (&refbuf, void), "\x11", 1) == 0);
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    signed char *ptr = alloc_buffer_alloc (&buf, signed char);
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    *ptr = 126;
51f0aa
+    test_empty (buf);
51f0aa
+  }
51f0aa
+  TEST_VERIFY (memcmp (alloc_buffer_next (&refbuf, void), "\176", 1) == 0);
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    char *ptr = alloc_buffer_alloc_array (&buf, char, 1);
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    *ptr = (char) 253;
51f0aa
+    test_empty (buf);
51f0aa
+  }
51f0aa
+  TEST_VERIFY (memcmp (alloc_buffer_next (&refbuf, void), "\xfd", 1) == 0);
51f0aa
+
51f0aa
+  /* Failure with larger objects.  */
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, short) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, double) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, struct twelve) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, short, 1) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, double, 1) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, struct twelve, 1) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+}
51f0aa
+
51f0aa
+static void
51f0aa
+test_size_2 (struct alloc_buffer refbuf)
51f0aa
+{
51f0aa
+  TEST_VERIFY (!alloc_buffer_has_failed (&refbuf));
51f0aa
+  TEST_VERIFY (alloc_buffer_size (&refbuf) == 2);
51f0aa
+  TEST_VERIFY (IS_ALIGNED (alloc_buffer_next (&refbuf, void), short));
51f0aa
+  test_empty_array (refbuf);
51f0aa
+  test_impossible_array (refbuf);
51f0aa
+
51f0aa
+  /* Success adding two bytes.  */
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    alloc_buffer_add_byte (&buf, '@');
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    test_size_1 (buf);
51f0aa
+  }
51f0aa
+  TEST_VERIFY (memcmp (alloc_buffer_next (&refbuf, void), "@\xfd", 2) == 0);
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    signed char *ptr = alloc_buffer_alloc (&buf, signed char);
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    *ptr = 'A';
51f0aa
+    test_size_1 (buf);
51f0aa
+  }
51f0aa
+  TEST_VERIFY (memcmp (alloc_buffer_next (&refbuf, void), "A\xfd", 2) == 0);
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    char *ptr = alloc_buffer_alloc_array (&buf, char, 1);
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    *ptr = 'B';
51f0aa
+    test_size_1 (buf);
51f0aa
+  }
51f0aa
+  TEST_VERIFY (memcmp (alloc_buffer_next (&refbuf, void), "B\xfd", 2) == 0);
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    unsigned short *ptr = alloc_buffer_alloc (&buf, unsigned short);
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (IS_ALIGNED (ptr, unsigned short));
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    *ptr = htons (0x12f4);
51f0aa
+    test_empty (buf);
51f0aa
+  }
51f0aa
+  TEST_VERIFY (memcmp (alloc_buffer_next (&refbuf, void), "\x12\xf4", 2) == 0);
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    unsigned short *ptr = alloc_buffer_alloc_array (&buf, unsigned short, 1);
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (IS_ALIGNED (ptr, unsigned short));
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    *ptr = htons (0x13f5);
51f0aa
+    test_empty (buf);
51f0aa
+  }
51f0aa
+  TEST_VERIFY (memcmp (alloc_buffer_next (&refbuf, void), "\x13\xf5", 2) == 0);
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    char *ptr = alloc_buffer_alloc_array (&buf, char, 2);
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    memcpy (ptr, "12", 2);
51f0aa
+    test_empty (buf);
51f0aa
+  }
51f0aa
+  TEST_VERIFY (memcmp (alloc_buffer_next (&refbuf, void), "12", 2) == 0);
51f0aa
+}
51f0aa
+
51f0aa
+static void
51f0aa
+test_misaligned (char pad)
51f0aa
+{
51f0aa
+  enum { SIZE = 23 };
51f0aa
+  char *backing = xmalloc (SIZE + 2);
51f0aa
+  backing[0] = ~pad;
51f0aa
+  backing[SIZE + 1] = pad;
51f0aa
+  struct alloc_buffer refbuf = alloc_buffer_create (backing + 1, SIZE);
51f0aa
+
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    short *ptr = alloc_buffer_alloc_array (&buf, short, SIZE / sizeof (short));
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (IS_ALIGNED (ptr, short));
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    for (int i = 0; i < SIZE / sizeof (short); ++i)
51f0aa
+      ptr[i] = htons (0xff01 + i);
51f0aa
+    TEST_VERIFY (memcmp (ptr,
51f0aa
+                         "\xff\x01\xff\x02\xff\x03\xff\x04"
51f0aa
+                         "\xff\x05\xff\x06\xff\x07\xff\x08"
51f0aa
+                         "\xff\x09\xff\x0a\xff\x0b", 22) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    uint32_t *ptr = alloc_buffer_alloc_array
51f0aa
+      (&buf, uint32_t, SIZE / sizeof (uint32_t));
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (IS_ALIGNED (ptr, uint32_t));
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    for (int i = 0; i < SIZE / sizeof (uint32_t); ++i)
51f0aa
+      ptr[i] = htonl (0xf1e2d301 + i);
51f0aa
+    TEST_VERIFY (memcmp (ptr,
51f0aa
+                         "\xf1\xe2\xd3\x01\xf1\xe2\xd3\x02"
51f0aa
+                         "\xf1\xe2\xd3\x03\xf1\xe2\xd3\x04"
51f0aa
+                         "\xf1\xe2\xd3\x05", 20) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    struct twelve *ptr = alloc_buffer_alloc (&buf, struct twelve);
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (IS_ALIGNED (ptr, struct twelve));
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    ptr->buffer[0] = htonl (0x11223344);
51f0aa
+    ptr->buffer[1] = htonl (0x55667788);
51f0aa
+    ptr->buffer[2] = htonl (0x99aabbcc);
51f0aa
+    TEST_VERIFY (memcmp (ptr,
51f0aa
+                         "\x11\x22\x33\x44"
51f0aa
+                         "\x55\x66\x77\x88"
51f0aa
+                         "\x99\xaa\xbb\xcc", 12) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    static const double nums[] = { 1, 2 };
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    double *ptr = alloc_buffer_alloc_array (&buf, double, 2);
51f0aa
+    TEST_VERIFY_EXIT (ptr != NULL);
51f0aa
+    TEST_VERIFY (IS_ALIGNED (ptr, double));
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    ptr[0] = nums[0];
51f0aa
+    ptr[1] = nums[1];
51f0aa
+    TEST_VERIFY (memcmp (ptr, nums, sizeof (nums)) == 0);
51f0aa
+  }
51f0aa
+
51f0aa
+  /* Verify that padding was not overwritten.  */
51f0aa
+  TEST_VERIFY (backing[0] == (char) ~pad);
51f0aa
+  TEST_VERIFY (backing[SIZE + 1] == pad);
51f0aa
+  free (backing);
51f0aa
+}
51f0aa
+
51f0aa
+/* Check that overflow during alignment is handled properly.  */
51f0aa
+static void
51f0aa
+test_large_misaligned (void)
51f0aa
+{
51f0aa
+  uintptr_t minus1 = -1;
51f0aa
+  uintptr_t start = minus1 & ~0xfe;
51f0aa
+  struct alloc_buffer refbuf = alloc_buffer_create ((void *) start, 16);
51f0aa
+  TEST_VERIFY (!alloc_buffer_has_failed (&refbuf));
51f0aa
+
51f0aa
+  struct __attribute__ ((aligned (256))) align256
51f0aa
+  {
51f0aa
+    int dymmy;
51f0aa
+  };
51f0aa
+
51f0aa
+  {
51f0aa
+    struct alloc_buffer buf = refbuf;
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc (&buf, struct align256) == NULL);
51f0aa
+    test_after_failure (buf);
51f0aa
+  }
51f0aa
+  for (int count = 0; count < 3; ++count)
51f0aa
+    {
51f0aa
+      struct alloc_buffer buf = refbuf;
51f0aa
+      TEST_VERIFY (alloc_buffer_alloc_array (&buf, struct align256, count)
51f0aa
+                   == NULL);
51f0aa
+      test_after_failure (buf);
51f0aa
+    }
51f0aa
+}
51f0aa
+
51f0aa
+/* Check behavior of large allocations.  */
51f0aa
+static void
51f0aa
+test_large (void)
51f0aa
+{
51f0aa
+  {
51f0aa
+    /* Allocation which wraps around.  */
51f0aa
+    struct alloc_buffer buf = { 1, SIZE_MAX };
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, char, SIZE_MAX) == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+  }
51f0aa
+
51f0aa
+  {
51f0aa
+    /* Successful very large allocation.  */
51f0aa
+    struct alloc_buffer buf = { 1, SIZE_MAX };
51f0aa
+    uintptr_t val = (uintptr_t) alloc_buffer_alloc_array
51f0aa
+      (&buf, char, SIZE_MAX - 1);
51f0aa
+    TEST_VERIFY (val == 1);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    test_empty (buf);
51f0aa
+  }
51f0aa
+
51f0aa
+  {
51f0aa
+    typedef char __attribute__ ((aligned (2))) char2;
51f0aa
+
51f0aa
+    /* Overflow in array size computation.   */
51f0aa
+    struct alloc_buffer buf = { 1, SIZE_MAX };
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, char2, SIZE_MAX - 1) == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+
51f0aa
+    /* Successful allocation after alignment.  */
51f0aa
+    buf = (struct alloc_buffer) { 1, SIZE_MAX };
51f0aa
+    uintptr_t val = (uintptr_t) alloc_buffer_alloc_array
51f0aa
+      (&buf, char2, SIZE_MAX - 2);
51f0aa
+    TEST_VERIFY (val == 2);
51f0aa
+    test_empty (buf);
51f0aa
+
51f0aa
+    /* Alignment behavior near the top of the address space.  */
51f0aa
+    buf = (struct alloc_buffer) { SIZE_MAX, SIZE_MAX };
51f0aa
+    TEST_VERIFY (alloc_buffer_next (&buf, char2) == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+    buf = (struct alloc_buffer) { SIZE_MAX, SIZE_MAX };
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, char2, 0) == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+  }
51f0aa
+
51f0aa
+  {
51f0aa
+    typedef short __attribute__ ((aligned (2))) short2;
51f0aa
+
51f0aa
+    /* Test overflow in size computation.  */
51f0aa
+    struct alloc_buffer buf = { 1, SIZE_MAX };
51f0aa
+    TEST_VERIFY (alloc_buffer_alloc_array (&buf, short2, SIZE_MAX / 2)
51f0aa
+                 == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+
51f0aa
+    /* A slightly smaller array fits within the allocation.  */
51f0aa
+    buf = (struct alloc_buffer) { 2, SIZE_MAX - 1 };
51f0aa
+    uintptr_t val = (uintptr_t) alloc_buffer_alloc_array
51f0aa
+      (&buf, short2, SIZE_MAX / 2 - 1);
51f0aa
+    TEST_VERIFY (val == 2);
51f0aa
+    test_empty (buf);
51f0aa
+  }
51f0aa
+}
51f0aa
+
51f0aa
+static void
51f0aa
+test_copy_bytes (void)
51f0aa
+{
51f0aa
+  char backing[4];
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    alloc_buffer_copy_bytes (&buf, "1", 1);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (alloc_buffer_size (&buf) == 3);
51f0aa
+    TEST_VERIFY (memcmp (backing, "1@@@", 4) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    alloc_buffer_copy_bytes (&buf, "12", 3);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (alloc_buffer_size (&buf) == 1);
51f0aa
+    TEST_VERIFY (memcmp (backing, "12\0@", 4) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    alloc_buffer_copy_bytes (&buf, "1234", 4);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (alloc_buffer_size (&buf) == 0);
51f0aa
+    TEST_VERIFY (memcmp (backing, "1234", 4) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    alloc_buffer_copy_bytes (&buf, "1234", 5);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (memcmp (backing, "@@@@", 4) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    alloc_buffer_copy_bytes (&buf, "1234", -1);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (memcmp (backing, "@@@@", 4) == 0);
51f0aa
+  }
51f0aa
+}
51f0aa
+
51f0aa
+static void
51f0aa
+test_copy_string (void)
51f0aa
+{
51f0aa
+  char backing[4];
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    const char *p = alloc_buffer_copy_string (&buf, "");
51f0aa
+    TEST_VERIFY (p == backing);
51f0aa
+    TEST_VERIFY (strcmp (p, "") == 0);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (alloc_buffer_size (&buf) == 3);
51f0aa
+    TEST_VERIFY (memcmp (backing, "\0@@@", 4) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    const char *p = alloc_buffer_copy_string (&buf, "1");
51f0aa
+    TEST_VERIFY (p == backing);
51f0aa
+    TEST_VERIFY (strcmp (p, "1") == 0);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (alloc_buffer_size (&buf) == 2);
51f0aa
+    TEST_VERIFY (memcmp (backing, "1\0@@", 4) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    const char *p = alloc_buffer_copy_string (&buf, "12");
51f0aa
+    TEST_VERIFY (p == backing);
51f0aa
+    TEST_VERIFY (strcmp (p, "12") == 0);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (alloc_buffer_size (&buf) == 1);
51f0aa
+    TEST_VERIFY (memcmp (backing, "12\0@", 4) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    const char *p = alloc_buffer_copy_string (&buf, "123");
51f0aa
+    TEST_VERIFY (p == backing);
51f0aa
+    TEST_VERIFY (strcmp (p, "123") == 0);
51f0aa
+    TEST_VERIFY (!alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (alloc_buffer_size (&buf) == 0);
51f0aa
+    TEST_VERIFY (memcmp (backing, "123", 4) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    TEST_VERIFY (alloc_buffer_copy_string (&buf, "1234") == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (memcmp (backing, "@@@@", 4) == 0);
51f0aa
+  }
51f0aa
+  {
51f0aa
+    memset (backing, '@', sizeof (backing));
51f0aa
+    struct alloc_buffer buf = alloc_buffer_create (backing, sizeof (backing));
51f0aa
+    TEST_VERIFY (alloc_buffer_copy_string (&buf, "12345") == NULL);
51f0aa
+    TEST_VERIFY (alloc_buffer_has_failed (&buf));
51f0aa
+    TEST_VERIFY (memcmp (backing, "@@@@", 4) == 0);
51f0aa
+  }
51f0aa
+}
51f0aa
+
51f0aa
+static int
51f0aa
+do_test (void)
51f0aa
+{
51f0aa
+  test_empty (alloc_buffer_create (NULL, 0));
51f0aa
+  test_empty (alloc_buffer_create ((char *) "", 0));
51f0aa
+  test_empty (alloc_buffer_create ((void *) 1, 0));
51f0aa
+
51f0aa
+  {
51f0aa
+    void *ptr = (void *) "";    /* Cannot be freed. */
51f0aa
+    struct alloc_buffer buf = alloc_buffer_allocate (1, &ptr);
51f0aa
+    test_size_1 (buf);
51f0aa
+    free (ptr);                 /* Should have been overwritten.  */
51f0aa
+  }
51f0aa
+
51f0aa
+  {
51f0aa
+    void *ptr= (void *) "";     /* Cannot be freed.  */
51f0aa
+    struct alloc_buffer buf = alloc_buffer_allocate (2, &ptr);
51f0aa
+    test_size_2 (buf);
51f0aa
+    free (ptr);                 /* Should have been overwritten.  */
51f0aa
+  }
51f0aa
+
51f0aa
+  test_misaligned (0);
51f0aa
+  test_misaligned (0xc7);
51f0aa
+  test_misaligned (0xff);
51f0aa
+
51f0aa
+  test_large_misaligned ();
51f0aa
+  test_large ();
51f0aa
+  test_copy_bytes ();
51f0aa
+  test_copy_string ();
51f0aa
+
51f0aa
+  return 0;
51f0aa
+}
51f0aa
+
51f0aa
+#include <support/test-driver.c>