Blame test/utils.h

Packit 030a23
#ifdef HAVE_CONFIG_H
Packit 030a23
#include <config.h>
Packit 030a23
#endif
Packit 030a23
Packit 030a23
#include <assert.h>
Packit 030a23
#include "pixman-private.h" /* For 'inline' definition */
Packit 030a23
#include "utils-prng.h"
Packit 030a23
Packit 030a23
#if defined(_MSC_VER)
Packit 030a23
#define snprintf _snprintf
Packit 030a23
#define strcasecmp _stricmp
Packit 030a23
#endif
Packit 030a23
Packit 030a23
#define ARRAY_LENGTH(A) ((int) (sizeof (A) / sizeof ((A) [0])))
Packit 030a23
Packit 030a23
/* A primitive pseudorandom number generator,
Packit 030a23
 * taken from POSIX.1-2001 example
Packit 030a23
 */
Packit 030a23
Packit 030a23
extern prng_t prng_state_data;
Packit 030a23
extern prng_t *prng_state;
Packit 030a23
#ifdef USE_OPENMP
Packit 030a23
#pragma omp threadprivate(prng_state_data)
Packit 030a23
#pragma omp threadprivate(prng_state)
Packit 030a23
#endif
Packit 030a23
Packit 030a23
static inline uint32_t
Packit 030a23
prng_rand (void)
Packit 030a23
{
Packit 030a23
    return prng_rand_r (prng_state);
Packit 030a23
}
Packit 030a23
Packit 030a23
static inline void
Packit 030a23
prng_srand (uint32_t seed)
Packit 030a23
{
Packit 030a23
    if (!prng_state)
Packit 030a23
    {
Packit 030a23
        /* Without setting a seed, PRNG does not work properly (is just
Packit 030a23
         * returning zeros). So we only initialize the pointer here to
Packit 030a23
         * make sure that 'prng_srand' is always called before any
Packit 030a23
         * other 'prng_*' function. The wrongdoers violating this order
Packit 030a23
         * will get a segfault. */
Packit 030a23
        prng_state = &prng_state_data;
Packit 030a23
    }
Packit 030a23
    prng_srand_r (prng_state, seed);
Packit 030a23
}
Packit 030a23
Packit 030a23
static inline uint32_t
Packit 030a23
prng_rand_n (int max)
Packit 030a23
{
Packit 030a23
    return prng_rand () % max;
Packit 030a23
}
Packit 030a23
Packit 030a23
static inline void
Packit 030a23
prng_randmemset (void *buffer, size_t size, prng_randmemset_flags_t flags)
Packit 030a23
{
Packit 030a23
    prng_randmemset_r (prng_state, buffer, size, flags);
Packit 030a23
}
Packit 030a23
Packit 030a23
/* CRC 32 computation
Packit 030a23
 */
Packit 030a23
uint32_t
Packit 030a23
compute_crc32 (uint32_t    in_crc32,
Packit 030a23
	       const void *buf,
Packit 030a23
	       size_t      buf_len);
Packit 030a23
Packit 030a23
uint32_t
Packit 030a23
compute_crc32_for_image (uint32_t        in_crc32,
Packit 030a23
			 pixman_image_t *image);
Packit 030a23
Packit 030a23
/* Print the image in hexadecimal */
Packit 030a23
void
Packit 030a23
print_image (pixman_image_t *image);
Packit 030a23
Packit 030a23
/* Returns TRUE if running on a little endian system
Packit 030a23
 */
Packit 030a23
static force_inline pixman_bool_t
Packit 030a23
is_little_endian (void)
Packit 030a23
{
Packit 030a23
    unsigned long endian_check_var = 1;
Packit 030a23
    return *(unsigned char *)&endian_check_var == 1;
Packit 030a23
}
Packit 030a23
Packit 030a23
/* perform endian conversion of pixel data
Packit 030a23
 */
Packit 030a23
void
Packit 030a23
image_endian_swap (pixman_image_t *img);
Packit 030a23
Packit 030a23
#if defined (HAVE_MPROTECT) && defined (HAVE_GETPAGESIZE) && \
Packit 030a23
    defined (HAVE_SYS_MMAN_H) && defined (HAVE_MMAP)
Packit 030a23
/* fence_malloc and friends have working fence implementation.
Packit 030a23
 * Without this, fence_malloc still allocs but does not catch
Packit 030a23
 * out-of-bounds accesses.
Packit 030a23
 */
Packit 030a23
#define FENCE_MALLOC_ACTIVE 1
Packit 030a23
#else
Packit 030a23
#define FENCE_MALLOC_ACTIVE 0
Packit 030a23
#endif
Packit 030a23
Packit 030a23
/* Allocate memory that is bounded by protected pages,
Packit 030a23
 * so that out-of-bounds access will cause segfaults
Packit 030a23
 */
Packit 030a23
void *
Packit 030a23
fence_malloc (int64_t len);
Packit 030a23
Packit 030a23
void
Packit 030a23
fence_free (void *data);
Packit 030a23
Packit 030a23
pixman_image_t *
Packit 030a23
fence_image_create_bits (pixman_format_code_t format,
Packit 030a23
                         int min_width,
Packit 030a23
                         int height,
Packit 030a23
                         pixman_bool_t stride_fence);
Packit 030a23
Packit 030a23
/* Return the page size if FENCE_MALLOC_ACTIVE, or zero otherwise */
Packit 030a23
unsigned long
Packit 030a23
fence_get_page_size ();
Packit 030a23
Packit 030a23
/* Generate n_bytes random bytes in fence_malloced memory */
Packit 030a23
uint8_t *
Packit 030a23
make_random_bytes (int n_bytes);
Packit 030a23
Packit 030a23
/* Return current time in seconds */
Packit 030a23
double
Packit 030a23
gettime (void);
Packit 030a23
Packit 030a23
uint32_t
Packit 030a23
get_random_seed (void);
Packit 030a23
Packit 030a23
/* main body of the fuzzer test */
Packit 030a23
int
Packit 030a23
fuzzer_test_main (const char *test_name,
Packit 030a23
		  int         default_number_of_iterations,
Packit 030a23
		  uint32_t    expected_checksum,
Packit 030a23
		  uint32_t    (*test_function)(int testnum, int verbose),
Packit 030a23
		  int         argc,
Packit 030a23
		  const char *argv[]);
Packit 030a23
Packit 030a23
void
Packit 030a23
fail_after (int seconds, const char *msg);
Packit 030a23
Packit 030a23
/* If possible, enable traps for floating point exceptions */
Packit 030a23
void enable_divbyzero_exceptions(void);
Packit 030a23
void enable_invalid_exceptions(void);
Packit 030a23
Packit 030a23
/* Converts a8r8g8b8 pixels to pixels that
Packit 030a23
 *  - are not premultiplied,
Packit 030a23
 *  - are stored in this order in memory: R, G, B, A, regardless of
Packit 030a23
 *    the endianness of the computer.
Packit 030a23
 * It is allowed for @src and @dst to point to the same memory buffer.
Packit 030a23
 */
Packit 030a23
void
Packit 030a23
a8r8g8b8_to_rgba_np (uint32_t *dst, uint32_t *src, int n_pixels);
Packit 030a23
Packit 030a23
pixman_bool_t
Packit 030a23
write_png (pixman_image_t *image, const char *filename);
Packit 030a23
Packit 030a23
void
Packit 030a23
draw_checkerboard (pixman_image_t *image,
Packit 030a23
		   int check_size,
Packit 030a23
		   uint32_t color1, uint32_t color2);
Packit 030a23
Packit 030a23
/* A pair of macros which can help to detect corruption of
Packit 030a23
 * floating point registers after a function call. This may
Packit 030a23
 * happen if _mm_empty() call is forgotten in MMX/SSE2 fast
Packit 030a23
 * path code, or ARM NEON assembly optimized function forgets
Packit 030a23
 * to save/restore d8-d15 registers before use.
Packit 030a23
 */
Packit 030a23
Packit 030a23
#define FLOAT_REGS_CORRUPTION_DETECTOR_START()                 \
Packit 030a23
    static volatile double frcd_volatile_constant1 = 123451;   \
Packit 030a23
    static volatile double frcd_volatile_constant2 = 123452;   \
Packit 030a23
    static volatile double frcd_volatile_constant3 = 123453;   \
Packit 030a23
    static volatile double frcd_volatile_constant4 = 123454;   \
Packit 030a23
    static volatile double frcd_volatile_constant5 = 123455;   \
Packit 030a23
    static volatile double frcd_volatile_constant6 = 123456;   \
Packit 030a23
    static volatile double frcd_volatile_constant7 = 123457;   \
Packit 030a23
    static volatile double frcd_volatile_constant8 = 123458;   \
Packit 030a23
    double frcd_canary_variable1 = frcd_volatile_constant1;    \
Packit 030a23
    double frcd_canary_variable2 = frcd_volatile_constant2;    \
Packit 030a23
    double frcd_canary_variable3 = frcd_volatile_constant3;    \
Packit 030a23
    double frcd_canary_variable4 = frcd_volatile_constant4;    \
Packit 030a23
    double frcd_canary_variable5 = frcd_volatile_constant5;    \
Packit 030a23
    double frcd_canary_variable6 = frcd_volatile_constant6;    \
Packit 030a23
    double frcd_canary_variable7 = frcd_volatile_constant7;    \
Packit 030a23
    double frcd_canary_variable8 = frcd_volatile_constant8;
Packit 030a23
Packit 030a23
#define FLOAT_REGS_CORRUPTION_DETECTOR_FINISH()                \
Packit 030a23
    assert (frcd_canary_variable1 == frcd_volatile_constant1); \
Packit 030a23
    assert (frcd_canary_variable2 == frcd_volatile_constant2); \
Packit 030a23
    assert (frcd_canary_variable3 == frcd_volatile_constant3); \
Packit 030a23
    assert (frcd_canary_variable4 == frcd_volatile_constant4); \
Packit 030a23
    assert (frcd_canary_variable5 == frcd_volatile_constant5); \
Packit 030a23
    assert (frcd_canary_variable6 == frcd_volatile_constant6); \
Packit 030a23
    assert (frcd_canary_variable7 == frcd_volatile_constant7); \
Packit 030a23
    assert (frcd_canary_variable8 == frcd_volatile_constant8);
Packit 030a23
Packit 030a23
/* Try to get an aligned memory chunk */
Packit 030a23
void *
Packit 030a23
aligned_malloc (size_t align, size_t size);
Packit 030a23
Packit 030a23
double
Packit 030a23
convert_srgb_to_linear (double component);
Packit 030a23
Packit 030a23
double
Packit 030a23
convert_linear_to_srgb (double component);
Packit 030a23
Packit 030a23
void
Packit 030a23
initialize_palette (pixman_indexed_t *palette, uint32_t depth, int is_rgb);
Packit 030a23
Packit 030a23
pixman_format_code_t
Packit 030a23
format_from_string (const char *s);
Packit 030a23
Packit 030a23
void
Packit 030a23
list_formats (void);
Packit 030a23
Packit 030a23
void
Packit 030a23
list_operators (void);
Packit 030a23
Packit 030a23
pixman_op_t
Packit 030a23
operator_from_string (const char *s);
Packit 030a23
Packit 030a23
const char *
Packit 030a23
operator_name (pixman_op_t op);
Packit 030a23
Packit 030a23
const char *
Packit 030a23
format_name (pixman_format_code_t format);
Packit 030a23
Packit 030a23
typedef struct
Packit 030a23
{
Packit 030a23
    double r, g, b, a;
Packit 030a23
} color_t;
Packit 030a23
Packit 030a23
void
Packit 030a23
do_composite (pixman_op_t op,
Packit 030a23
	      const color_t *src,
Packit 030a23
	      const color_t *mask,
Packit 030a23
	      const color_t *dst,
Packit 030a23
	      color_t *result,
Packit 030a23
	      pixman_bool_t component_alpha);
Packit 030a23
Packit 030a23
void
Packit 030a23
round_color (pixman_format_code_t format, color_t *color);
Packit 030a23
Packit 030a23
typedef struct
Packit 030a23
{
Packit 030a23
    pixman_format_code_t format;
Packit 030a23
    uint32_t am, rm, gm, bm;
Packit 030a23
    uint32_t as, rs, gs, bs;
Packit 030a23
    uint32_t aw, rw, gw, bw;
Packit 030a23
} pixel_checker_t;
Packit 030a23
Packit 030a23
void
Packit 030a23
pixel_checker_init (pixel_checker_t *checker, pixman_format_code_t format);
Packit 030a23
Packit 030a23
void
Packit 030a23
pixel_checker_split_pixel (const pixel_checker_t *checker, uint32_t pixel,
Packit 030a23
			   int *a, int *r, int *g, int *b);
Packit 030a23
Packit 030a23
void
Packit 030a23
pixel_checker_get_max (const pixel_checker_t *checker, color_t *color,
Packit 030a23
		       int *a, int *r, int *g, int *b);
Packit 030a23
Packit 030a23
void
Packit 030a23
pixel_checker_get_min (const pixel_checker_t *checker, color_t *color,
Packit 030a23
		       int *a, int *r, int *g, int *b);
Packit 030a23
Packit 030a23
pixman_bool_t
Packit 030a23
pixel_checker_check (const pixel_checker_t *checker,
Packit 030a23
		     uint32_t pixel, color_t *color);
Packit 030a23
Packit 030a23
void
Packit 030a23
pixel_checker_convert_pixel_to_color (const pixel_checker_t *checker,
Packit 030a23
                                      uint32_t pixel, color_t *color);
Packit 030a23
Packit 030a23
void
Packit 030a23
pixel_checker_get_masks (const pixel_checker_t *checker,
Packit 030a23
                         uint32_t              *am,
Packit 030a23
                         uint32_t              *rm,
Packit 030a23
                         uint32_t              *gm,
Packit 030a23
                         uint32_t              *bm);