1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135
|
/* Copyright (C) 2001-2012 Artifex Software, Inc.
All Rights Reserved.
This software is provided AS-IS with no warranty, either express or
implied.
This software is distributed under license and may not be copied,
modified or distributed except as expressly authorized under the terms
of the license contained in the file LICENSE in this distribution.
Refer to licensing information at http://www.artifex.com or contact
Artifex Software, Inc., 7 Mt. Lassen Drive - Suite A-134, San Rafael,
CA 94903, U.S.A., +1(415)492-9861, for further information.
*/
/* Support for glyph data access */
#include "memory_.h"
#include "gx.h"
#include "gsgdata.h"
#include "gserrors.h"
#include "gsmatrix.h" /* for gsfont.h */
#include "gsstruct.h"
#include "gxfont.h"
/* GC structure descriptor */
static ENUM_PTRS_WITH(gs_glyph_data_enum_ptrs, gs_glyph_data_t *pgd)
case 0: return ENUM_CONST_BYTESTRING(&pgd->bits);
case 1: return ENUM_OBJ(pgd->proc_data);
ENUM_PTRS_END
static RELOC_PTRS_WITH(gs_glyph_data_reloc_ptrs, gs_glyph_data_t *pgd)
{
RELOC_CONST_BYTESTRING_VAR(pgd->bits);
RELOC_OBJ_VAR(pgd->proc_data);
} RELOC_PTRS_END
gs_public_st_composite(st_glyph_data, gs_glyph_data_t, "gs_glyph_data_t",
gs_glyph_data_enum_ptrs, gs_glyph_data_reloc_ptrs);
/* ------ Client facilities ------ */
/* Replace glyph data by a substring. */
int
gs_glyph_data_substring(gs_glyph_data_t *pgd, uint offset, uint size)
{
if (offset > pgd->bits.size || size > pgd->bits.size - offset)
return_error(gs_error_rangecheck);
return pgd->procs->substring(pgd, offset, size);
}
/* Free the data for a glyph. */
void
gs_glyph_data_free(gs_glyph_data_t *pgd, client_name_t cname)
{
if (pgd != 0) {
if (pgd->procs != 0)
pgd->procs->free(pgd, cname);
gs_glyph_data_from_null(pgd);
}
}
/* ------ Implementor support ------ */
/* Don't manage the glyph data. */
static void
glyph_data_free_permanent(gs_glyph_data_t *pgd, client_name_t cname)
{
}
static int
glyph_data_substring_permanent(gs_glyph_data_t *pgd, uint offset, uint size)
{
pgd->bits.data += offset;
pgd->bits.size = size;
return 0;
}
/* Manage the glyph data using the font's allocator. */
static void
glyph_data_free_by_font(gs_glyph_data_t *pgd, client_name_t cname)
{
gs_free_const_bytestring(((gs_font *)pgd->proc_data)->memory,
&pgd->bits, cname);
}
static int
glyph_data_substring_by_font(gs_glyph_data_t *pgd, uint offset, uint size)
{
gs_font *const font = pgd->proc_data;
byte *data = (byte *)pgd->bits.data; /* break const */
if (pgd->bits.bytes) /* object, not string */
return glyph_data_substring_permanent(pgd, offset, size);
if (offset > 0)
memmove(data, data + offset, size);
pgd->bits.data =
gs_resize_string(font->memory, data, pgd->bits.size, size,
"glyph_data_substring"); /* shortening, can't fail */
pgd->bits.size = size;
return 0;
}
static const gs_glyph_data_procs_t no_free_procs = {
glyph_data_free_permanent, glyph_data_substring_permanent
};
static const gs_glyph_data_procs_t free_by_font_procs = {
glyph_data_free_by_font, glyph_data_substring_by_font
};
/*
* Initialize glyph data from a string or from bytes. If the font is NULL
* (for glyph data that is part of the font), set the glyph data freeing
* procedure to "do not free"; if the font is not NULL (for just-allocated
* glyph data), set the freeing procedure to "free using the font's
* allocator."
*/
void
gs_glyph_data_from_string(gs_glyph_data_t *pgd, const byte *data,
uint size, gs_font *font)
{
gs_bytestring_from_string(&pgd->bits, data, size);
pgd->proc_data = font;
pgd->procs = (font ? &free_by_font_procs : &no_free_procs);
}
void
gs_glyph_data_from_bytes(gs_glyph_data_t *pgd, const byte *bytes,
uint offset, uint size, gs_font *font)
{
gs_bytestring_from_bytes(&pgd->bits, bytes, offset, size);
pgd->proc_data = font;
pgd->procs = (font ? &free_by_font_procs : &no_free_procs);
}
void
gs_glyph_data_from_null(gs_glyph_data_t *pgd)
{
gs_glyph_data_from_string(pgd, NULL, 0, NULL);
}
|