185 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			185 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C
		
	
	
	
| /* A minor test-program for memmove.
 | |
|    Copyright (C) 2005 Axis Communications.
 | |
|    All rights reserved.
 | |
| 
 | |
|    Redistribution and use in source and binary forms, with or without
 | |
|    modification, are permitted provided that the following conditions
 | |
|    are met:
 | |
| 
 | |
|    1. Redistributions of source code must retain the above copyright
 | |
|       notice, this list of conditions and the following disclaimer.
 | |
| 
 | |
|    2. Neither the name of Axis Communications nor the names of its
 | |
|       contributors may be used to endorse or promote products derived
 | |
|       from this software without specific prior written permission.
 | |
| 
 | |
|    THIS SOFTWARE IS PROVIDED BY AXIS COMMUNICATIONS AND ITS CONTRIBUTORS
 | |
|    ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 | |
|    LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 | |
|    A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL AXIS
 | |
|    COMMUNICATIONS OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
 | |
|    INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 | |
|    (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 | |
|    SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 | |
|    HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 | |
|    STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
 | |
|    IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 | |
|    POSSIBILITY OF SUCH DAMAGE.  */
 | |
| 
 | |
| /* Test moves of 0..MAX bytes; overlapping-src-higher,
 | |
|    overlapping-src-lower and non-overlapping.  The overlap varies with
 | |
|    1..N where N is the size moved.  This means an order of MAX**2
 | |
|    iterations.  The size of an octet may seem appropriate for MAX and
 | |
|    makes an upper limit for simple testing.  For the CRIS simulator,
 | |
|    making this 256 added 90s to the test-run (2GHz P4) while 64 (4s) was
 | |
|    enough to spot the bugs that had crept in, hence the number chosen.  */
 | |
| #define MAX 64
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| 
 | |
| #define TOO_MANY_ERRORS 11
 | |
| int errors = 0;
 | |
| 
 | |
| #define DEBUGP					\
 | |
|  if (errors == TOO_MANY_ERRORS)			\
 | |
|    printf ("Further errors omitted\n");		\
 | |
|  else if (errors < TOO_MANY_ERRORS)		\
 | |
|    printf
 | |
| 
 | |
| /* A safe target-independent memmove.  */
 | |
| 
 | |
| void
 | |
| mymemmove (unsigned char *dest, unsigned char *src, size_t n)
 | |
| {
 | |
|   size_t i;
 | |
| 
 | |
|   if ((src <= dest && src + n <= dest)
 | |
|       || src >= dest)
 | |
|     while (n-- > 0)
 | |
|       *dest++ = *src++;
 | |
|   else
 | |
|     {
 | |
|       dest += n;
 | |
|       src += n;
 | |
|       while (n-- > 0)
 | |
| 	*--dest = *--src;
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* It's either the noinline attribute or forcing the test framework to
 | |
|    pass -fno-builtin-memmove.  */
 | |
| void
 | |
| xmemmove (unsigned char *dest, unsigned char *src, size_t n)
 | |
|      __attribute__ ((__noinline__));
 | |
| 
 | |
| void
 | |
| xmemmove (unsigned char *dest, unsigned char *src, size_t n)
 | |
| {
 | |
|   void *retp;
 | |
|   retp = memmove (dest, src, n);
 | |
| 
 | |
|   if (retp != dest)
 | |
|     {
 | |
|       errors++;
 | |
|       DEBUGP ("memmove of n bytes returned %p instead of dest=%p\n",
 | |
| 	      retp, dest);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| /* Fill the array with something we can associate with a position, but
 | |
|    not exactly the same as the position index.  */
 | |
| 
 | |
| void
 | |
| fill (unsigned char dest[MAX*3])
 | |
| {
 | |
|   size_t i;
 | |
|   for (i = 0; i < MAX*3; i++)
 | |
|     dest[i] = (10 + i) % MAX;
 | |
| }
 | |
| 
 | |
| int
 | |
| main (void)
 | |
| {
 | |
|   size_t i;
 | |
|   int errors = 0;
 | |
| 
 | |
|   /* Leave some room before and after the area tested, so we can detect
 | |
|      overwrites of up to N bytes, N being the amount tested.  If you
 | |
|      want to test using valgrind, make these malloced instead.  */
 | |
|   unsigned char from_test[MAX*3];
 | |
|   unsigned char to_test[MAX*3];
 | |
|   unsigned char from_known[MAX*3];
 | |
|   unsigned char to_known[MAX*3];
 | |
| 
 | |
|   /* Non-overlap.  */
 | |
|   for (i = 0; i < MAX; i++)
 | |
|     {
 | |
|       /* Do the memmove first before setting the known array, so we know
 | |
|          it didn't change any of the known array.  */
 | |
|       fill (from_test);
 | |
|       fill (to_test);
 | |
|       xmemmove (to_test + MAX, 1 + from_test + MAX, i);
 | |
| 
 | |
|       fill (from_known);
 | |
|       fill (to_known);
 | |
|       mymemmove (to_known + MAX, 1 + from_known + MAX, i);
 | |
| 
 | |
|       if (memcmp (to_known, to_test, sizeof (to_known)) != 0)
 | |
| 	{
 | |
| 	  errors++;
 | |
| 	  DEBUGP ("memmove failed non-overlap test for %d bytes\n", i);
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|   /* Overlap-from-before.  */
 | |
|   for (i = 0; i < MAX; i++)
 | |
|     {
 | |
|       size_t j;
 | |
|       for (j = 0; j < i; j++)
 | |
| 	{
 | |
| 	  fill (to_test);
 | |
| 	  xmemmove (to_test + MAX * 2 - i, to_test + MAX * 2 - i - j, i);
 | |
| 
 | |
| 	  fill (to_known);
 | |
| 	  mymemmove (to_known + MAX * 2 - i, to_known + MAX * 2 - i - j, i);
 | |
| 
 | |
| 	  if (memcmp (to_known, to_test, sizeof (to_known)) != 0)
 | |
| 	    {
 | |
| 	      errors++;
 | |
| 	      DEBUGP ("memmove failed for %d bytes,"
 | |
| 		      " with src %d bytes before dest\n",
 | |
| 		      i, j);
 | |
| 	    }
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|   /* Overlap-from-after.  */
 | |
|   for (i = 0; i < MAX; i++)
 | |
|     {
 | |
|       size_t j;
 | |
|       for (j = 0; j < i; j++)
 | |
| 	{
 | |
| 	  fill (to_test);
 | |
| 	  xmemmove (to_test + MAX, to_test + MAX + j, i);
 | |
| 
 | |
| 	  fill (to_known);
 | |
| 	  mymemmove (to_known + MAX, to_known + MAX + j, i);
 | |
| 
 | |
| 	  if (memcmp (to_known, to_test, sizeof (to_known)) != 0)
 | |
| 	    {
 | |
| 	      errors++;
 | |
| 	      DEBUGP ("memmove failed when moving %d bytes,"
 | |
| 		      " with src %d bytes after dest\n",
 | |
| 		      i, j);
 | |
| 	    }
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|   if (errors != 0)
 | |
|     abort ();
 | |
|   exit (0);
 | |
| }
 |