bug 974575 - backport Behdad's patches for color bitmap glyph support from current cairo trunk. r=jrmuizel

* * *
[ft] Fix resizing of bitmap fonts
From b94a519aad3d5b50aa6de47ee16ee6a099de9791 Mon Sep 17 00:00:00 2001
Say, you were asking cairo for a font at 200px.  For bitmap-only fonts,
cairo was finding the closes strike size and using it.  If the strike
was at 20px, well, that's what you were getting.  We now scale that 20px
strike by a factor of 10 to get the correct size rendering.

Note that by itself this patch doesn't change much on the Linux desktop.
The reason is that the size you are interested in (eg. 200px) is lost by
fontconfig.  When you request a font at 200px, fontconfig returns a font
pattern that says 20px, and so the next layers thing you want a font at
20px.  To address that, one also needs a piece of fontconfig config that
puts the 200 back into the pixelsize.  Something like this:

<match target="font">
  <test name="scalable" mode="eq">
    <bool>false</bool>
  </test>
  <edit name="pixelsize" mode="assign">
    <times>
      <name>size</name>
      <name>dpi</name>
      <double>0.0138888888888</double> <!--1/72.-->
    </times>
  </edit>
</match>

I'm going to try to upstream this config so it will be enabled by
default.  The config can be a bit smarter.  For example, if
metricshinting is enabled and the size difference is small, we may as
well not scale.

The nice thing about this is that the configuration of whether and when
to scale bitmaps will be done in fontconfig, not cairo / Qt / ... code.
---
* * *
[FT] Prefer downscaling bitmap glyphs to upscaling

From a8f1b456db744e33a10b2301df03528787e5b1ca Mon Sep 17 00:00:00 2001
Say, you have bitmap strikes for sizes 50ppem and 100ppem.
To render at 60ppem, it's much better to downscale the 100ppem
bitmap than upscale 50ppem one.  Prefer downscaling.
---
* * *
[ft] I meant fabs(), not abs()

From 13bd8d09b44e50649f6fc4d58d036bc32c1d5c5b Mon Sep 17 00:00:00 2001
---
* * *
[ft] Fix memory bug in copying bitmaps

From a0f556f37fb7016aa304b7cf0e811c0d38f0b969 Mon Sep 17 00:00:00 2001
---
* * *
[ft] Fix wrong assumptions

From e738079302a968b7b1fb9101cd4d92a8887bedce Mon Sep 17 00:00:00 2001
If subpixel rendering is enabled, but FT returns a 8bit gray bitmap
(perhaps because the font has 8bit embedded bitmaps) we were hitting
the assertions because the assumptions made were wrong.  Fix up.
---
* * *
Towards support loading color glyphs from FreeType

From 2cc353c3dbe01b4d8f65d6de800f2b1d6004a1c2 Mon Sep 17 00:00:00 2001
See comments.
---
* * *
Support 2bit and 4bit embedded bitmaps

From 9444ef09ccde2735258cc1bd2f1912119a32dd88 Mon Sep 17 00:00:00 2001
---
* * *
[ft] Fix math

From 7d26341072b13a78d4b3fe58779057ac020be487 Mon Sep 17 00:00:00 2001
---
* * *
[ft] Add missing include

From 0554d76402321b25cc952180e4d19436a9038d1a Mon Sep 17 00:00:00 2001
---
* * *
[ft] Fix alignment

From 34a747e7bdeba1cfe17318f80fbe6720d47bc023 Mon Sep 17 00:00:00 2001
---
* * *
[ft] Ensure alignment of bitmaps received from FreeType

From 46d9db96d460fea72f0420102e8a90c6a7231f79 Mon Sep 17 00:00:00 2001
---
This commit is contained in:
Jonathan Kew 2014-03-06 12:29:22 +00:00
Родитель 4654925890
Коммит d9227c7961
1 изменённых файлов: 143 добавлений и 61 удалений

Просмотреть файл

@ -52,6 +52,7 @@
#include FT_FREETYPE_H #include FT_FREETYPE_H
#include FT_OUTLINE_H #include FT_OUTLINE_H
#include FT_IMAGE_H #include FT_IMAGE_H
#include FT_BITMAP_H
#include FT_TRUETYPE_TABLES_H #include FT_TRUETYPE_TABLES_H
#if HAVE_FT_GLYPHSLOT_EMBOLDEN #if HAVE_FT_GLYPHSLOT_EMBOLDEN
#include FT_SYNTHESIS_H #include FT_SYNTHESIS_H
@ -702,7 +703,8 @@ _cairo_ft_unscaled_font_unlock_face (cairo_ft_unscaled_font_t *unscaled)
static cairo_status_t static cairo_status_t
_compute_transform (cairo_ft_font_transform_t *sf, _compute_transform (cairo_ft_font_transform_t *sf,
cairo_matrix_t *scale) cairo_matrix_t *scale,
cairo_ft_unscaled_font_t *unscaled)
{ {
cairo_status_t status; cairo_status_t status;
double x_scale, y_scale; double x_scale, y_scale;
@ -730,6 +732,39 @@ _compute_transform (cairo_ft_font_transform_t *sf,
if (y_scale < 1.0) if (y_scale < 1.0)
y_scale = 1.0; y_scale = 1.0;
if (unscaled && (unscaled->face->face_flags & FT_FACE_FLAG_SCALABLE) == 0) {
double min_distance = DBL_MAX;
cairo_bool_t magnify = TRUE;
int i;
int best_i = 0;
double best_x_size = 0;
double best_y_size = 0;
for (i = 0; i < unscaled->face->num_fixed_sizes; i++) {
double x_size = unscaled->face->available_sizes[i].y_ppem / 64.;
double y_size = unscaled->face->available_sizes[i].y_ppem / 64.;
double distance = y_size - y_scale;
/*
* distance is positive if current strike is larger than desired
* size, and negative if smaller.
*
* We like to prefer down-scaling to upscaling.
*/
if ((magnify && distance >= 0) || fabs (distance) <= min_distance) {
magnify = distance < 0;
min_distance = fabs (distance);
best_i = i;
best_x_size = x_size;
best_y_size = y_size;
}
}
x_scale = best_x_size;
y_scale = best_y_size;
}
sf->x_scale = x_scale; sf->x_scale = x_scale;
sf->y_scale = y_scale; sf->y_scale = y_scale;
@ -754,6 +789,7 @@ _cairo_ft_unscaled_font_set_scale (cairo_ft_unscaled_font_t *unscaled,
cairo_ft_font_transform_t sf; cairo_ft_font_transform_t sf;
FT_Matrix mat; FT_Matrix mat;
FT_Error error; FT_Error error;
double x_scale, y_scale;
assert (unscaled->face != NULL); assert (unscaled->face != NULL);
@ -767,7 +803,7 @@ _cairo_ft_unscaled_font_set_scale (cairo_ft_unscaled_font_t *unscaled,
unscaled->have_scale = TRUE; unscaled->have_scale = TRUE;
unscaled->current_scale = *scale; unscaled->current_scale = *scale;
status = _compute_transform (&sf, scale); status = _compute_transform (&sf, scale, unscaled);
if (unlikely (status)) if (unlikely (status))
return status; return status;
@ -792,46 +828,14 @@ _cairo_ft_unscaled_font_set_scale (cairo_ft_unscaled_font_t *unscaled,
FT_Set_Transform(unscaled->face, &mat, NULL); FT_Set_Transform(unscaled->face, &mat, NULL);
if ((unscaled->face->face_flags & FT_FACE_FLAG_SCALABLE) != 0) { x_scale = MIN(sf.x_scale, MAX_FONT_SIZE);
double x_scale = MIN(sf.x_scale, MAX_FONT_SIZE); y_scale = MIN(sf.y_scale, MAX_FONT_SIZE);
double y_scale = MIN(sf.y_scale, MAX_FONT_SIZE); error = FT_Set_Char_Size (unscaled->face,
error = FT_Set_Char_Size (unscaled->face, x_scale * 64.0 + .5,
x_scale * 64.0 + .5, y_scale * 64.0 + .5,
y_scale * 64.0 + .5, 0, 0);
0, 0); if (error)
if (error) return _cairo_error (CAIRO_STATUS_NO_MEMORY);
return _cairo_error (CAIRO_STATUS_NO_MEMORY);
} else {
double min_distance = DBL_MAX;
int i;
int best_i = 0;
for (i = 0; i < unscaled->face->num_fixed_sizes; i++) {
#if HAVE_FT_BITMAP_SIZE_Y_PPEM
double size = unscaled->face->available_sizes[i].y_ppem / 64.;
#else
double size = unscaled->face->available_sizes[i].height;
#endif
double distance = fabs (size - sf.y_scale);
if (distance <= min_distance) {
min_distance = distance;
best_i = i;
}
}
#if HAVE_FT_BITMAP_SIZE_Y_PPEM
error = FT_Set_Char_Size (unscaled->face,
unscaled->face->available_sizes[best_i].x_ppem,
unscaled->face->available_sizes[best_i].y_ppem,
0, 0);
if (error)
#endif
error = FT_Set_Pixel_Sizes (unscaled->face,
unscaled->face->available_sizes[best_i].width,
unscaled->face->available_sizes[best_i].height);
if (error)
return _cairo_error (CAIRO_STATUS_NO_MEMORY);
}
return CAIRO_STATUS_SUCCESS; return CAIRO_STATUS_SUCCESS;
} }
@ -1114,6 +1118,7 @@ _fill_xrender_bitmap(FT_Bitmap *target,
*/ */
static cairo_status_t static cairo_status_t
_get_bitmap_surface (FT_Bitmap *bitmap, _get_bitmap_surface (FT_Bitmap *bitmap,
FT_Library library,
cairo_bool_t own_buffer, cairo_bool_t own_buffer,
cairo_font_options_t *font_options, cairo_font_options_t *font_options,
cairo_image_surface_t **surface) cairo_image_surface_t **surface)
@ -1122,6 +1127,7 @@ _get_bitmap_surface (FT_Bitmap *bitmap,
unsigned char *data; unsigned char *data;
int format = CAIRO_FORMAT_A8; int format = CAIRO_FORMAT_A8;
cairo_image_surface_t *image; cairo_image_surface_t *image;
cairo_bool_t component_alpha = FALSE;
width = bitmap->width; width = bitmap->width;
height = bitmap->rows; height = bitmap->rows;
@ -1152,9 +1158,7 @@ _get_bitmap_surface (FT_Bitmap *bitmap,
source = bitmap->buffer; source = bitmap->buffer;
dest = data; dest = data;
for (i = height; i; i--) { for (i = height; i; i--) {
memcpy (dest, source, bitmap->pitch); memcpy (dest, source, stride);
memset (dest + bitmap->pitch, '\0', stride - bitmap->pitch);
source += bitmap->pitch; source += bitmap->pitch;
dest += stride; dest += stride;
} }
@ -1178,8 +1182,18 @@ _get_bitmap_surface (FT_Bitmap *bitmap,
case FT_PIXEL_MODE_LCD: case FT_PIXEL_MODE_LCD:
case FT_PIXEL_MODE_LCD_V: case FT_PIXEL_MODE_LCD_V:
case FT_PIXEL_MODE_GRAY: case FT_PIXEL_MODE_GRAY:
if (font_options->antialias != CAIRO_ANTIALIAS_SUBPIXEL) { if (font_options->antialias != CAIRO_ANTIALIAS_SUBPIXEL ||
bitmap->pixel_mode == FT_PIXEL_MODE_GRAY)
{
stride = bitmap->pitch; stride = bitmap->pitch;
/* We don't support stride not multiple of 4. */
if (stride & 3)
{
assert (!own_buffer);
goto convert;
}
if (own_buffer) { if (own_buffer) {
data = bitmap->buffer; data = bitmap->buffer;
} else { } else {
@ -1190,21 +1204,72 @@ _get_bitmap_surface (FT_Bitmap *bitmap,
memcpy (data, bitmap->buffer, stride * height); memcpy (data, bitmap->buffer, stride * height);
} }
format = CAIRO_FORMAT_A8; format = CAIRO_FORMAT_A8;
} else { } else {
/* if we get there, the data from the source bitmap data = bitmap->buffer;
* really comes from _fill_xrender_bitmap, and is stride = bitmap->pitch;
* made of 32-bit ARGB or ABGR values */ format = CAIRO_FORMAT_ARGB32;
assert (own_buffer != 0); component_alpha = TRUE;
assert (bitmap->pixel_mode != FT_PIXEL_MODE_GRAY);
data = bitmap->buffer;
stride = bitmap->pitch;
format = CAIRO_FORMAT_ARGB32;
} }
break; break;
#ifdef FT_LOAD_COLOR
case FT_PIXEL_MODE_BGRA:
stride = width * 4;
if (own_buffer) {
data = bitmap->buffer;
} else {
data = _cairo_malloc_ab (height, stride);
if (!data)
return _cairo_error (CAIRO_STATUS_NO_MEMORY);
memcpy (data, bitmap->buffer, stride * height);
}
format = CAIRO_FORMAT_ARGB32;
break;
#endif
case FT_PIXEL_MODE_GRAY2: case FT_PIXEL_MODE_GRAY2:
case FT_PIXEL_MODE_GRAY4: case FT_PIXEL_MODE_GRAY4:
convert:
if (!own_buffer && library)
{
/* This is pretty much the only case that we can get in here. */
/* Convert to 8bit grayscale. */
FT_Bitmap tmp;
FT_Int align;
format = CAIRO_FORMAT_A8;
align = cairo_format_stride_for_width (format, bitmap->width);
FT_Bitmap_New( &tmp );
if (FT_Bitmap_Convert( library, bitmap, &tmp, align ))
return _cairo_error (CAIRO_STATUS_NO_MEMORY);
FT_Bitmap_Done( library, bitmap );
*bitmap = tmp;
stride = bitmap->pitch;
data = _cairo_malloc_ab (height, stride);
if (!data)
return _cairo_error (CAIRO_STATUS_NO_MEMORY);
if (bitmap->num_grays != 256)
{
unsigned int x, y;
unsigned int mul = 255 / (bitmap->num_grays - 1);
FT_Byte *p = bitmap->buffer;
for (y = 0; y < height; y++) {
for (x = 0; x < width; x++)
p[x] *= mul;
p += bitmap->pitch;
}
}
memcpy (data, bitmap->buffer, stride * height);
break;
}
/* These could be triggered by very rare types of TrueType fonts */ /* These could be triggered by very rare types of TrueType fonts */
default: default:
if (own_buffer) if (own_buffer)
@ -1222,7 +1287,7 @@ _get_bitmap_surface (FT_Bitmap *bitmap,
return (*surface)->base.status; return (*surface)->base.status;
} }
if (format == CAIRO_FORMAT_ARGB32) if (component_alpha)
pixman_image_set_component_alpha (image->pixman_image, TRUE); pixman_image_set_component_alpha (image->pixman_image, TRUE);
_cairo_image_surface_assume_ownership_of_data (image); _cairo_image_surface_assume_ownership_of_data (image);
@ -1403,7 +1468,7 @@ _render_glyph_outline (FT_Face face,
/* Note: /* Note:
* _get_bitmap_surface will free bitmap.buffer if there is an error * _get_bitmap_surface will free bitmap.buffer if there is an error
*/ */
status = _get_bitmap_surface (&bitmap, TRUE, font_options, surface); status = _get_bitmap_surface (&bitmap, NULL, TRUE, font_options, surface);
if (unlikely (status)) if (unlikely (status))
return status; return status;
@ -1444,6 +1509,7 @@ _render_glyph_bitmap (FT_Face face,
return _cairo_error (CAIRO_STATUS_NO_MEMORY); return _cairo_error (CAIRO_STATUS_NO_MEMORY);
status = _get_bitmap_surface (&glyphslot->bitmap, status = _get_bitmap_surface (&glyphslot->bitmap,
glyphslot->library,
FALSE, font_options, FALSE, font_options,
surface); surface);
if (unlikely (status)) if (unlikely (status))
@ -1485,7 +1551,7 @@ _transform_glyph_bitmap (cairo_matrix_t * shape,
* the "shape" portion of the font transform * the "shape" portion of the font transform
*/ */
original_to_transformed = *shape; original_to_transformed = *shape;
cairo_surface_get_device_offset (&(*surface)->base, &origin_x, &origin_y); cairo_surface_get_device_offset (&(*surface)->base, &origin_x, &origin_y);
orig_width = (*surface)->width; orig_width = (*surface)->width;
orig_height = (*surface)->height; orig_height = (*surface)->height;
@ -1535,7 +1601,11 @@ _transform_glyph_bitmap (cairo_matrix_t * shape,
if (unlikely (status)) if (unlikely (status))
return status; return status;
image = cairo_image_surface_create (CAIRO_FORMAT_A8, width, height); if (cairo_image_surface_get_format (*surface) == CAIRO_FORMAT_ARGB32 &&
!pixman_image_get_component_alpha ((*surface)->pixman_image))
image = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, width, height);
else
image = cairo_image_surface_create (CAIRO_FORMAT_A8, width, height);
if (unlikely (image->status)) if (unlikely (image->status))
return image->status; return image->status;
@ -2169,6 +2239,18 @@ _cairo_ft_scaled_glyph_init (void *abstract_font,
vertical_layout = TRUE; vertical_layout = TRUE;
} }
#ifdef FT_LOAD_COLOR
/* Color-glyph support:
*
* This flags needs plumbing through fontconfig (does it?), and
* maybe we should cache color and grayscale bitmaps separately
* such that users of the font (ie. the surface) can choose which
* version to use based on target content type.
*/
load_flags |= FT_LOAD_COLOR;
#endif
error = FT_Load_Glyph (scaled_font->unscaled->face, error = FT_Load_Glyph (scaled_font->unscaled->face,
_cairo_scaled_glyph_index(scaled_glyph), _cairo_scaled_glyph_index(scaled_glyph),
load_flags); load_flags);
@ -2926,7 +3008,7 @@ _cairo_ft_resolve_pattern (FcPattern *pattern,
font_matrix, font_matrix,
&scale); &scale);
status = _compute_transform (&sf, &scale); status = _compute_transform (&sf, &scale, NULL);
if (unlikely (status)) if (unlikely (status))
return (cairo_font_face_t *)&_cairo_font_face_nil; return (cairo_font_face_t *)&_cairo_font_face_nil;