240 lines
5.8 KiB
C
240 lines
5.8 KiB
C
/*
|
|
* Copyright © 2015 Raspberry Pi Foundation
|
|
*
|
|
* Permission to use, copy, modify, distribute, and sell this software and its
|
|
* documentation for any purpose is hereby granted without fee, provided that
|
|
* the above copyright notice appear in all copies and that both that
|
|
* copyright notice and this permission notice appear in supporting
|
|
* documentation, and that the name of the copyright holders not be used in
|
|
* advertising or publicity pertaining to distribution of the software without
|
|
* specific, written prior permission. The copyright holders make no
|
|
* representations about the suitability of this software for any purpose. It
|
|
* is provided "as is" without express or implied warranty.
|
|
*
|
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
|
|
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
* FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
* SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN
|
|
* AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
|
|
* OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
|
* SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include "utils.h"
|
|
|
|
|
|
#if FENCE_MALLOC_ACTIVE && defined (HAVE_SIGACTION)
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <stdarg.h>
|
|
#include <errno.h>
|
|
#include <signal.h>
|
|
#include <unistd.h>
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
|
|
pixman_bool_t verbose;
|
|
|
|
static void
|
|
segv_handler (int sig, siginfo_t *si, void *unused)
|
|
{
|
|
_exit (EXIT_SUCCESS);
|
|
}
|
|
|
|
static void
|
|
die (const char *msg, int err)
|
|
{
|
|
if (err)
|
|
perror (msg);
|
|
else
|
|
fprintf (stderr, "%s\n", msg);
|
|
|
|
abort ();
|
|
}
|
|
|
|
static void
|
|
prinfo (const char *fmt, ...)
|
|
{
|
|
va_list ap;
|
|
|
|
if (!verbose)
|
|
return;
|
|
|
|
va_start (ap, fmt);
|
|
vfprintf (stderr, fmt, ap);
|
|
va_end (ap);
|
|
}
|
|
|
|
static void
|
|
do_expect_signal (void (*fn)(void *), void *data)
|
|
{
|
|
struct sigaction sa;
|
|
|
|
sa.sa_flags = SA_SIGINFO;
|
|
sigemptyset (&sa.sa_mask);
|
|
sa.sa_sigaction = segv_handler;
|
|
if (sigaction (SIGSEGV, &sa, NULL) == -1)
|
|
die ("sigaction failed", errno);
|
|
if (sigaction (SIGBUS, &sa, NULL) == -1)
|
|
die ("sigaction failed", errno);
|
|
|
|
(*fn)(data);
|
|
|
|
_exit (EXIT_FAILURE);
|
|
}
|
|
|
|
/* Check that calling fn(data) causes a segmentation fault.
|
|
*
|
|
* You cannot portably return from a SIGSEGV handler in any way,
|
|
* so we fork, and do the test in the child process. Child's
|
|
* exit status will reflect the result. Its SEGV handler causes it
|
|
* to exit with success, and return failure otherwise.
|
|
*/
|
|
static pixman_bool_t
|
|
expect_signal (void (*fn)(void *), void *data)
|
|
{
|
|
pid_t pid, wp;
|
|
int status;
|
|
|
|
pid = fork ();
|
|
if (pid == -1)
|
|
die ("fork failed", errno);
|
|
|
|
if (pid == 0)
|
|
do_expect_signal (fn, data); /* never returns */
|
|
|
|
wp = waitpid (pid, &status, 0);
|
|
if (wp != pid)
|
|
die ("waitpid did not work", wp == -1 ? errno : 0);
|
|
|
|
if (WIFEXITED (status) && WEXITSTATUS (status) == EXIT_SUCCESS)
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
read_u8 (void *data)
|
|
{
|
|
volatile uint8_t *p = data;
|
|
|
|
*p;
|
|
}
|
|
|
|
static pixman_bool_t
|
|
test_read_fault (uint8_t *p, int offset)
|
|
{
|
|
prinfo ("*(uint8_t *)(%p + %d)", p, offset);
|
|
|
|
if (expect_signal (read_u8, p + offset))
|
|
{
|
|
prinfo ("\tsignal OK\n");
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
prinfo ("\tFAILED\n");
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
test_read_ok (uint8_t *p, int offset)
|
|
{
|
|
prinfo ("*(uint8_t *)(%p + %d)", p, offset);
|
|
|
|
/* If fails, SEGV. */
|
|
read_u8 (p + offset);
|
|
|
|
prinfo ("\tOK\n");
|
|
}
|
|
|
|
static pixman_bool_t
|
|
test_read_faults (pixman_image_t *image)
|
|
{
|
|
pixman_bool_t ok = TRUE;
|
|
pixman_format_code_t format = pixman_image_get_format (image);
|
|
int width = pixman_image_get_width (image);
|
|
int height = pixman_image_get_height (image);
|
|
int stride = pixman_image_get_stride (image);
|
|
uint8_t *p = (void *)pixman_image_get_data (image);
|
|
int row_bytes = width * PIXMAN_FORMAT_BPP (format) / 8;
|
|
|
|
prinfo ("%s %dx%d, row %d B, stride %d B:\n",
|
|
format_name (format), width, height, row_bytes, stride);
|
|
|
|
assert (height > 3);
|
|
|
|
test_read_ok (p, 0);
|
|
test_read_ok (p, row_bytes - 1);
|
|
test_read_ok (p, stride);
|
|
test_read_ok (p, stride + row_bytes - 1);
|
|
test_read_ok (p, 2 * stride);
|
|
test_read_ok (p, 2 * stride + row_bytes - 1);
|
|
test_read_ok (p, 3 * stride);
|
|
test_read_ok (p, (height - 1) * stride + row_bytes - 1);
|
|
|
|
ok &= test_read_fault (p, -1);
|
|
ok &= test_read_fault (p, row_bytes);
|
|
ok &= test_read_fault (p, stride - 1);
|
|
ok &= test_read_fault (p, stride + row_bytes);
|
|
ok &= test_read_fault (p, 2 * stride - 1);
|
|
ok &= test_read_fault (p, 2 * stride + row_bytes);
|
|
ok &= test_read_fault (p, 3 * stride - 1);
|
|
ok &= test_read_fault (p, height * stride);
|
|
|
|
return ok;
|
|
}
|
|
|
|
static pixman_bool_t
|
|
test_image_faults (pixman_format_code_t format, int min_width, int height)
|
|
{
|
|
pixman_bool_t ok;
|
|
pixman_image_t *image;
|
|
|
|
image = fence_image_create_bits (format, min_width, height, TRUE);
|
|
ok = test_read_faults (image);
|
|
pixman_image_unref (image);
|
|
|
|
return ok;
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
pixman_bool_t ok = TRUE;
|
|
|
|
if (getenv ("VERBOSE") != NULL)
|
|
verbose = TRUE;
|
|
|
|
ok &= test_image_faults (PIXMAN_a8r8g8b8, 7, 5);
|
|
ok &= test_image_faults (PIXMAN_r8g8b8, 7, 5);
|
|
ok &= test_image_faults (PIXMAN_r5g6b5, 7, 5);
|
|
ok &= test_image_faults (PIXMAN_a8, 7, 5);
|
|
ok &= test_image_faults (PIXMAN_a4, 7, 5);
|
|
ok &= test_image_faults (PIXMAN_a1, 7, 5);
|
|
|
|
if (ok)
|
|
return EXIT_SUCCESS;
|
|
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
#else /* FENCE_MALLOC_ACTIVE */
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
/* Automake return code for test SKIP. */
|
|
return 77;
|
|
}
|
|
|
|
#endif /* FENCE_MALLOC_ACTIVE */
|