mirror of
https://github.com/harfbuzz/harfbuzz.git
synced 2025-04-12 16:23:01 +00:00
API change: - hb_face_is_immutable() now takes `hb_face_t *` instead of previous `const hb_face_t *`. This should not pose any problem for any clients in our belief.
371 lines
10 KiB
C
371 lines
10 KiB
C
/*
|
|
* Copyright © 2011 Google, Inc.
|
|
*
|
|
* This is part of HarfBuzz, a text shaping library.
|
|
*
|
|
* Permission is hereby granted, without written agreement and without
|
|
* license or royalty fees, to use, copy, modify, and distribute this
|
|
* software and its documentation for any purpose, provided that the
|
|
* above copyright notice and the following two paragraphs appear in
|
|
* all copies of this software.
|
|
*
|
|
* IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
|
|
* DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
|
|
* ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
|
|
* IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
|
|
* DAMAGE.
|
|
*
|
|
* THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
|
|
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
* FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
|
|
* ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
|
|
* PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
|
|
*
|
|
* Google Author(s): Behdad Esfahbod
|
|
*/
|
|
|
|
#include "hb-test.h"
|
|
|
|
/* Unit tests for hb-object-private.h */
|
|
|
|
|
|
static hb_blob_t *
|
|
create_blob (void)
|
|
{
|
|
static char data[] = "test data";
|
|
return hb_blob_create (data, sizeof (data), HB_MEMORY_MODE_READONLY, NULL, NULL);
|
|
}
|
|
static hb_blob_t *
|
|
create_blob_from_inert (void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
static hb_buffer_t *
|
|
create_buffer (void)
|
|
{
|
|
return hb_buffer_create ();
|
|
}
|
|
static hb_buffer_t *
|
|
create_buffer_from_inert (void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
static hb_map_t *
|
|
create_map (void)
|
|
{
|
|
return hb_map_create ();
|
|
}
|
|
static hb_map_t *
|
|
create_map_from_inert (void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
static hb_set_t *
|
|
create_set (void)
|
|
{
|
|
return hb_set_create ();
|
|
}
|
|
static hb_set_t *
|
|
create_set_from_inert (void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
static hb_face_t *
|
|
create_face (void)
|
|
{
|
|
hb_blob_t *blob = (hb_blob_t *) create_blob ();
|
|
hb_face_t *face = hb_face_create (blob, 0);
|
|
hb_blob_destroy (blob);
|
|
return face;
|
|
}
|
|
static hb_face_t *
|
|
create_face_from_inert (void)
|
|
{
|
|
return hb_face_create (hb_blob_get_empty (), 0);
|
|
}
|
|
|
|
static hb_font_t *
|
|
create_font (void)
|
|
{
|
|
hb_face_t *face = (hb_face_t *) create_face ();
|
|
hb_font_t *font = hb_font_create (face);
|
|
hb_face_destroy (face);
|
|
return font;
|
|
}
|
|
static hb_font_t *
|
|
create_font_from_inert (void)
|
|
{
|
|
return hb_font_create (hb_face_get_empty ());
|
|
}
|
|
|
|
static hb_font_funcs_t *
|
|
create_font_funcs (void)
|
|
{
|
|
return hb_font_funcs_create ();
|
|
}
|
|
static hb_font_funcs_t *
|
|
create_font_funcs_from_inert (void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
static hb_shape_plan_t *
|
|
create_shape_plan (void)
|
|
{
|
|
hb_segment_properties_t props = HB_SEGMENT_PROPERTIES_DEFAULT;
|
|
props.direction = HB_DIRECTION_LTR;
|
|
hb_face_t *face = (hb_face_t *) create_face ();
|
|
hb_shape_plan_t *shape_plan = hb_shape_plan_create (face, &props, NULL, 0, NULL);
|
|
hb_face_destroy (face);
|
|
return shape_plan;
|
|
}
|
|
static hb_shape_plan_t *
|
|
create_shape_plan_from_inert (void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
static hb_unicode_funcs_t *
|
|
create_unicode_funcs (void)
|
|
{
|
|
return hb_unicode_funcs_create (NULL);
|
|
}
|
|
static hb_unicode_funcs_t *
|
|
create_unicode_funcs_from_inert (void)
|
|
{
|
|
return hb_unicode_funcs_create (hb_unicode_funcs_get_empty ());
|
|
}
|
|
|
|
|
|
#define MAGIC0 0x12345678
|
|
#define MAGIC1 0x76543210
|
|
|
|
typedef struct {
|
|
int value;
|
|
gboolean freed;
|
|
} data_t;
|
|
|
|
static int global_data;
|
|
|
|
static void global_free_up (void *p G_GNUC_UNUSED)
|
|
{
|
|
global_data++;
|
|
}
|
|
|
|
static void free_up0 (void *p)
|
|
{
|
|
data_t *data = (data_t *) p;
|
|
|
|
g_assert_cmphex (data->value, ==, MAGIC0);
|
|
g_assert_true (!data->freed);
|
|
data->freed = TRUE;
|
|
}
|
|
|
|
static void free_up1 (void *p)
|
|
{
|
|
data_t *data = (data_t *) p;
|
|
|
|
g_assert_cmphex (data->value, ==, MAGIC1);
|
|
g_assert_true (!data->freed);
|
|
data->freed = TRUE;
|
|
}
|
|
|
|
#define OBJECT(name, _hb_object_make_immutable, _hb_object_is_immutable) \
|
|
static void \
|
|
test_object_##name (void) \
|
|
{ \
|
|
typedef hb_##name##_t type_t; \
|
|
typedef type_t *(*create_func_t) (void); \
|
|
typedef type_t *(*reference_func_t) (type_t *obj); \
|
|
typedef void (*destroy_func_t) (type_t *obj); \
|
|
typedef hb_bool_t (*set_user_data_func_t) (type_t *obj, hb_user_data_key_t *key, void *data, hb_destroy_func_t destroy, hb_bool_t replace); \
|
|
typedef void * (*get_user_data_func_t) (const type_t *obj, hb_user_data_key_t *key); \
|
|
typedef void (*make_immutable_func_t) (type_t *obj); \
|
|
typedef hb_bool_t (*is_immutable_func_t) (type_t *obj); \
|
|
\
|
|
struct object_t { \
|
|
create_func_t create; \
|
|
create_func_t create_from_inert; \
|
|
create_func_t get_empty; \
|
|
reference_func_t reference; \
|
|
destroy_func_t destroy; \
|
|
set_user_data_func_t set_user_data; \
|
|
get_user_data_func_t get_user_data; \
|
|
make_immutable_func_t make_immutable; \
|
|
is_immutable_func_t is_immutable; \
|
|
const char *name; \
|
|
} o[1] = {{\
|
|
create_##name, \
|
|
create_##name##_from_inert, \
|
|
hb_##name##_get_empty, \
|
|
hb_##name##_reference, \
|
|
hb_##name##_destroy, \
|
|
hb_##name##_set_user_data, \
|
|
hb_##name##_get_user_data, \
|
|
_hb_object_make_immutable, \
|
|
_hb_object_is_immutable, \
|
|
#name, \
|
|
}}; \
|
|
\
|
|
void *obj; \
|
|
hb_user_data_key_t key[1001]; \
|
|
\
|
|
{ \
|
|
unsigned int j; \
|
|
data_t data[1000] = {{MAGIC0, FALSE}, {MAGIC1, FALSE}}; \
|
|
\
|
|
g_test_message ("Testing object %s", o->name); \
|
|
\
|
|
g_test_message ("->create()"); \
|
|
obj = o->create (); \
|
|
g_assert_true (obj); \
|
|
\
|
|
g_assert_true (obj == o->reference (obj)); \
|
|
o->destroy (obj); \
|
|
\
|
|
if (o->is_immutable) \
|
|
g_assert_true (!o->is_immutable (obj)); \
|
|
\
|
|
g_assert_true (o->set_user_data (obj, &key[0], &data[0], free_up0, TRUE)); \
|
|
g_assert_true (o->get_user_data (obj, &key[0]) == &data[0]); \
|
|
\
|
|
if (o->is_immutable) { \
|
|
o->make_immutable (obj); \
|
|
g_assert_true (o->is_immutable (obj)); \
|
|
} \
|
|
\
|
|
/* Should still work even if object is made immutable */ \
|
|
g_assert_true (o->set_user_data (obj, &key[1], &data[1], free_up1, TRUE)); \
|
|
g_assert_true (o->get_user_data (obj, &key[1]) == &data[1]); \
|
|
\
|
|
g_assert_true (!o->set_user_data (obj, NULL, &data[0], free_up0, TRUE)); \
|
|
g_assert_true (o->get_user_data (obj, &key[0]) == &data[0]); \
|
|
g_assert_true (o->set_user_data (obj, &key[0], &data[1], NULL, TRUE)); \
|
|
g_assert_true (data[0].freed); \
|
|
g_assert_true (o->get_user_data (obj, &key[0]) == &data[1]); \
|
|
g_assert_true (!data[1].freed); \
|
|
\
|
|
data[0].freed = FALSE; \
|
|
g_assert_true (o->set_user_data (obj, &key[0], &data[0], free_up0, TRUE)); \
|
|
g_assert_true (!data[0].freed); \
|
|
g_assert_true (o->set_user_data (obj, &key[0], NULL, NULL, TRUE)); \
|
|
g_assert_true (data[0].freed); \
|
|
\
|
|
data[0].freed = FALSE; \
|
|
global_data = 0; \
|
|
g_assert_true (o->set_user_data (obj, &key[0], &data[0], free_up0, TRUE)); \
|
|
g_assert_true (!o->set_user_data (obj, &key[0], &data[0], free_up0, FALSE)); \
|
|
g_assert_cmpuint (global_data, ==, 0); \
|
|
g_assert_true (o->set_user_data (obj, &key[0], NULL, global_free_up, TRUE)); \
|
|
g_assert_cmpuint (global_data, ==, 0); \
|
|
g_assert_true (o->set_user_data (obj, &key[0], NULL, NULL, TRUE)); \
|
|
g_assert_cmpuint (global_data, ==, 1); \
|
|
\
|
|
global_data = 0; \
|
|
for (j = 2; j < 1000; j++) \
|
|
g_assert_true (o->set_user_data (obj, &key[j], &data[j], global_free_up, TRUE)); \
|
|
for (j = 2; j < 1000; j++) \
|
|
g_assert_true (o->get_user_data (obj, &key[j]) == &data[j]); \
|
|
for (j = 100; j < 1000; j++) \
|
|
g_assert_true (o->set_user_data (obj, &key[j], NULL, NULL, TRUE)); \
|
|
for (j = 2; j < 100; j++) \
|
|
g_assert_true (o->get_user_data (obj, &key[j]) == &data[j]); \
|
|
for (j = 100; j < 1000; j++) \
|
|
g_assert_true (!o->get_user_data (obj, &key[j])); \
|
|
g_assert_cmpuint (global_data, ==, 900); \
|
|
\
|
|
g_assert_true (!data[1].freed); \
|
|
o->destroy (obj); \
|
|
g_assert_true (data[0].freed); \
|
|
g_assert_true (data[1].freed); \
|
|
g_assert_cmpuint (global_data, ==, 1000-2); \
|
|
} \
|
|
\
|
|
{ \
|
|
data_t data[2] = {{MAGIC0, FALSE}, {MAGIC1, FALSE}}; \
|
|
\
|
|
g_test_message ("->get_empty()"); \
|
|
obj = o->get_empty (); \
|
|
g_assert_true (obj); \
|
|
\
|
|
g_assert_true (obj == o->reference (obj)); \
|
|
o->destroy (obj); \
|
|
\
|
|
if (o->is_immutable) \
|
|
g_assert_true (o->is_immutable (obj)); \
|
|
\
|
|
g_assert_true (!o->set_user_data (obj, &key[0], &data[0], free_up0, TRUE)); \
|
|
g_assert_true (!o->get_user_data (obj, &key[0])); \
|
|
\
|
|
o->destroy (obj); \
|
|
o->destroy (obj); \
|
|
o->destroy (obj); \
|
|
o->destroy (obj); \
|
|
o->destroy (obj); \
|
|
\
|
|
g_assert_true (!data[0].freed); \
|
|
} \
|
|
\
|
|
{ \
|
|
data_t data[2] = {{MAGIC0, FALSE}, {MAGIC1, FALSE}}; \
|
|
\
|
|
g_test_message ("->create_from_inert()"); \
|
|
obj = o->create_from_inert (); \
|
|
if (!obj) \
|
|
return; \
|
|
g_assert_true (obj != o->get_empty ()); \
|
|
\
|
|
g_assert_true (obj == o->reference (obj)); \
|
|
o->destroy (obj); \
|
|
\
|
|
if (o->is_immutable) \
|
|
g_assert_true (!o->is_immutable (obj)); \
|
|
\
|
|
g_assert_true (o->set_user_data (obj, &key[0], &data[0], free_up0, TRUE)); \
|
|
g_assert_true (o->get_user_data (obj, &key[0])); \
|
|
\
|
|
o->destroy (obj); \
|
|
\
|
|
g_assert_true (data[0].freed); \
|
|
} \
|
|
}
|
|
|
|
#define OBJECT_WITH_IMMUTABILITY(name) \
|
|
OBJECT (name, hb_##name##_make_immutable, hb_##name##_is_immutable)
|
|
#define OBJECT_WITHOUT_IMMUTABILITY(name) \
|
|
OBJECT (name, NULL, NULL)
|
|
|
|
|
|
OBJECT_WITHOUT_IMMUTABILITY (buffer)
|
|
OBJECT_WITHOUT_IMMUTABILITY (map)
|
|
OBJECT_WITHOUT_IMMUTABILITY (set)
|
|
OBJECT_WITHOUT_IMMUTABILITY (shape_plan)
|
|
|
|
OBJECT_WITH_IMMUTABILITY (blob)
|
|
OBJECT_WITH_IMMUTABILITY (face)
|
|
OBJECT_WITH_IMMUTABILITY (font)
|
|
OBJECT_WITH_IMMUTABILITY (font_funcs)
|
|
OBJECT_WITH_IMMUTABILITY (unicode_funcs)
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
hb_test_init (&argc, &argv);
|
|
|
|
hb_test_add (test_object_buffer);
|
|
hb_test_add (test_object_map);
|
|
hb_test_add (test_object_shape_plan);
|
|
hb_test_add (test_object_set);
|
|
|
|
hb_test_add (test_object_blob);
|
|
hb_test_add (test_object_face);
|
|
hb_test_add (test_object_font);
|
|
hb_test_add (test_object_font_funcs);
|
|
hb_test_add (test_object_unicode_funcs);
|
|
|
|
return hb_test_run ();
|
|
}
|