WebKit Bugzilla
New
Browse
Log In
×
Sign in with GitHub
or
Remember my login
Create Account
·
Forgot Password
Forgotten password account recovery
RESOLVED FIXED
161493
[FreeType] Incorrect application of glyph positioning in the Y direction
https://bugs.webkit.org/show_bug.cgi?id=161493
Summary
[FreeType] Incorrect application of glyph positioning in the Y direction
Khaled Hosny
Reported
2016-09-01 10:51:44 PDT
Created
attachment 287632
[details]
Rendering in Epiphany The vertical placement of glyphs seems to be applied wrongly, for example the Arabic glyphs in
https://fonts.google.com/specimen/Aref+Ruqaa
and
https://fonts.google.com/specimen/Reem+Kufi
has their dots moved incorrectly above or below the glyphs. I suspect this is relative to the fact that in fonts Y moves up while in many graphical systems it moves down, so the Y positions returned from HarfBuzz need to be negated at some point.
Attachments
Rendering in Epiphany
(117.33 KB, image/png)
2016-09-01 10:51 PDT
,
Khaled Hosny
no flags
Details
Rendering in Chrome
(122.62 KB, image/png)
2016-09-01 10:52 PDT
,
Khaled Hosny
no flags
Details
Rendering in Firefox
(123.81 KB, image/png)
2016-09-01 10:53 PDT
,
Khaled Hosny
no flags
Details
Proof of concept patch
(4.03 KB, patch)
2016-09-03 07:23 PDT
,
Khaled Hosny
no flags
Details
Formatted Diff
Diff
Patch
(11.36 KB, patch)
2016-09-26 05:10 PDT
,
Frédéric Wang (:fredw)
no flags
Details
Formatted Diff
Diff
Patch
(8.77 KB, patch)
2016-09-26 12:59 PDT
,
Frédéric Wang (:fredw)
mmaxfield
: review-
Details
Formatted Diff
Diff
Test patch
(2.22 KB, patch)
2018-02-28 02:41 PST
,
Khaled Hosny
no flags
Details
Formatted Diff
Diff
Patch
(5.38 KB, patch)
2019-03-28 05:05 PDT
,
Carlos Garcia Campos
mcatanzaro
: review+
Details
Formatted Diff
Diff
Patch for landing
(50.24 KB, patch)
2019-03-28 05:15 PDT
,
Carlos Garcia Campos
no flags
Details
Formatted Diff
Diff
Show Obsolete
(4)
View All
Add attachment
proposed patch, testcase, etc.
Khaled Hosny
Comment 1
2016-09-01 10:52:22 PDT
Created
attachment 287633
[details]
Rendering in Chrome
Khaled Hosny
Comment 2
2016-09-01 10:53:49 PDT
Created
attachment 287634
[details]
Rendering in Firefox
Khaled Hosny
Comment 3
2016-09-02 17:08:25 PDT
Upon further examination, it seems that Y offsets returned from HarfBuzz are eventually discarded when creating the GlyphBuffer. In HarfBuzzShaper::fillGlyphBufferFromHarfBuzzRun(), the Y offset is (incorrectly?) used to calculate the Y advance of the glyph. Also it seems that support for Y offsets in GlyphBuffer is enabled only on Windows. So probably what is needed here is to enable GlyphBuffer offsets also for Gtk, populate the from HarfBuzz runs, and finally using them in FontCascade::drawGlyphs(). This is all pure speculation from just reading the code.
Khaled Hosny
Comment 4
2016-09-02 17:10:47 PDT
By Gtk I meant Cairo.
Khaled Hosny
Comment 5
2016-09-03 07:23:01 PDT
Created
attachment 287860
[details]
Proof of concept patch This is a proof of concept patch that fixes the Y positioning issue for me. It seems just setting the offsets is all needed, probably other parts of the code will apply them when present. I haven’t tested what is the effect of setting the X offset as well (e.g. on kerning).
Myles C. Maxfield
Comment 6
2016-09-06 14:12:36 PDT
Comment on
attachment 287860
[details]
Proof of concept patch View in context:
https://bugs.webkit.org/attachment.cgi?id=287860&action=review
> Source/WebCore/platform/graphics/GlyphBuffer.h:257 > Vector<FloatSize, 2048> m_offsets;
By the way, I'm doing something similar here:
https://bugs.webkit.org/show_bug.cgi?id=161119
Michael Catanzaro
Comment 7
2016-09-08 19:46:33 PDT
Myles, we don't have any GTK+ reviewers who understand fonts, let alone complex layouts or Arabic fonts. I know you don't know much or anything about FreeType or Harbuzz or Cairo, but I think you're nevertheless the reviewer for this. Do you mind keeping an eye on this bug, and review Khaled's work when he thinks it's ready to land? (It's not obsoleted by your work in
bug #161119
, is it?)
Myles C. Maxfield
Comment 8
2016-09-09 11:47:51 PDT
(In reply to
comment #7
)
> Myles, we don't have any GTK+ reviewers who understand fonts, let alone > complex layouts or Arabic fonts. I know you don't know much or anything > about FreeType or Harbuzz or Cairo, but I think you're nevertheless the > reviewer for this. Do you mind keeping an eye on this bug, and review > Khaled's work when he thinks it's ready to land? (It's not obsoleted by your > work in
bug #161119
, is it?)
I feel honored 😊 Yeah, I'll do my best to review it. Luckily, I have machines which can run FreeType, HarfBuzz, & Cairo, so I'll spend some time getting to know those libraries in order to review this. The patch you linked to should be orthogonal to this one. Khaled: am I correct in assuming this bug only reproduces with specific fonts? Because of licensing issues, we can't include arbitrary fonts in our tests. I would be happy to work with you to make a test font which exhibits this bug.
Myles C. Maxfield
Comment 9
2016-09-09 13:41:18 PDT
Comment on
attachment 287860
[details]
Proof of concept patch View in context:
https://bugs.webkit.org/attachment.cgi?id=287860&action=review
>> Source/WebCore/platform/graphics/GlyphBuffer.h:257 >> Vector<FloatSize, 2048> m_offsets; > > By the way, I'm doing something similar here:
https://bugs.webkit.org/show_bug.cgi?id=161119
Whoops, this was the wrong link. I meant to link to
https://trac.webkit.org/changeset/205396
Khaled Hosny
Comment 10
2016-09-09 13:45:05 PDT
The issue happens in any font that makes use of Y direction movement, and most Arabic fonts do that so it can be quite visible there bit it is not limited to them. I actually already built a test font that can be used to test this bug and I can provide it under whatever suitable license. I just need to figure out how to do the tests part then provide a finished patch.
Myles C. Maxfield
Comment 11
2016-09-09 13:52:36 PDT
GlyphBuffer is only used for painting. As such, we can save space if, instead of storing separate advances / origins in it, we collapse these into "painting advances" which are the offset from each glyph's paint position to the next. I considered renaming GlyphBuffer to something like PaintingGlyphBuffer but ultimately decided it wasn't worth the effort. Ultimately, we should remove the m_offsets member from GlyphBuffer in the Windows port. I plan on doing this at some point in the future. Rather than increasing the usage of this variable, I would suggest computing each glyph's paint position inside HarfBuzzShaper (thereby taking into account the offsets) and storing that in GlyphBuffer.
Myles C. Maxfield
Comment 12
2016-09-09 13:54:02 PDT
(In reply to
comment #10
)
> The issue happens in any font that makes use of Y direction movement, and > most Arabic fonts do that so it can be quite visible there bit it is not > limited to them. > > I actually already built a test font that can be used to test this bug and I > can provide it under whatever suitable license. I just need to figure out > how to do the tests part then provide a finished patch.
This is fantastic! Please let me know if you have any questions about our testing infrastructure. I would be glad to help out :D
Myles C. Maxfield
Comment 13
2016-09-09 14:03:52 PDT
Comment on
attachment 287860
[details]
Proof of concept patch View in context:
https://bugs.webkit.org/attachment.cgi?id=287860&action=review
> Source/WebCore/platform/graphics/harfbuzz/HarfBuzzShaper.cpp:580 > + glyphBuffer->add(glyphs[i], currentRun->fontData(), createGlyphBufferAdvance(glyphAdvanceX, glyphAdvanceY), currentCharacterIndex, &offset);
It appears that FontCascade::drawGlyphBuffer() doesn't actually move any glyphs according to the offsets. How does this work?
Myles C. Maxfield
Comment 14
2016-09-09 14:04:43 PDT
> > Whoops, this was the wrong link. I meant to link to >
https://trac.webkit.org/changeset/205396
This commit does the same thing that this patch does, except for Cocoa ports.
Myles C. Maxfield
Comment 15
2016-09-09 14:16:03 PDT
Comment on
attachment 287860
[details]
Proof of concept patch View in context:
https://bugs.webkit.org/attachment.cgi?id=287860&action=review
>> Source/WebCore/platform/graphics/harfbuzz/HarfBuzzShaper.cpp:580 >> + glyphBuffer->add(glyphs[i], currentRun->fontData(), createGlyphBufferAdvance(glyphAdvanceX, glyphAdvanceY), currentCharacterIndex, &offset); > > It appears that FontCascade::drawGlyphBuffer() doesn't actually move any glyphs according to the offsets. How does this work?
There is a different add() function which takes a GlyphBufferAdvance, which should be 2-dimensional.
Myles C. Maxfield
Comment 16
2016-09-09 14:16:58 PDT
Comment on
attachment 287860
[details]
Proof of concept patch View in context:
https://bugs.webkit.org/attachment.cgi?id=287860&action=review
>>> Source/WebCore/platform/graphics/harfbuzz/HarfBuzzShaper.cpp:580 >>> + glyphBuffer->add(glyphs[i], currentRun->fontData(), createGlyphBufferAdvance(glyphAdvanceX, glyphAdvanceY), currentCharacterIndex, &offset); >> >> It appears that FontCascade::drawGlyphBuffer() doesn't actually move any glyphs according to the offsets. How does this work? > > There is a different add() function which takes a GlyphBufferAdvance, which should be 2-dimensional.
Hopefully, if you migrate to this other add() function, you can remove the entire add() function which takes the float width.
Myles C. Maxfield
Comment 17
2016-09-09 14:20:03 PDT
(In reply to
comment #11
)
> GlyphBuffer is only used for painting. As such, we can save space if, > instead of storing separate advances / origins in it, we collapse these into > "painting advances" which are the offset from each glyph's paint position to > the next. > > I considered renaming GlyphBuffer to something like PaintingGlyphBuffer but > ultimately decided it wasn't worth the effort. > > Ultimately, we should remove the m_offsets member from GlyphBuffer in the > Windows port. I plan on doing this at some point in the future. Rather than > increasing the usage of this variable, I would suggest computing each > glyph's paint position inside HarfBuzzShaper (thereby taking into account > the offsets) and storing that in GlyphBuffer.
I'm doing this at
https://bugs.webkit.org/show_bug.cgi?id=161814
Myles C. Maxfield
Comment 18
2016-09-09 17:03:27 PDT
(In reply to
comment #17
)
> (In reply to
comment #11
) > > GlyphBuffer is only used for painting. As such, we can save space if, > > instead of storing separate advances / origins in it, we collapse these into > > "painting advances" which are the offset from each glyph's paint position to > > the next. > > > > I considered renaming GlyphBuffer to something like PaintingGlyphBuffer but > > ultimately decided it wasn't worth the effort. > > > > Ultimately, we should remove the m_offsets member from GlyphBuffer in the > > Windows port. I plan on doing this at some point in the future. Rather than > > increasing the usage of this variable, I would suggest computing each > > glyph's paint position inside HarfBuzzShaper (thereby taking into account > > the offsets) and storing that in GlyphBuffer. > > I'm doing this at
https://bugs.webkit.org/show_bug.cgi?id=161814
Whoops, looks like it's used in FontCGWin.cpp... not sure how I missed that...
Frédéric Wang (:fredw)
Comment 19
2016-09-26 05:10:09 PDT
Created
attachment 289820
[details]
Patch
Khaled Hosny
Comment 20
2016-09-26 09:21:40 PDT
Since advances and offsets are different things, if we need to get rid of the separate offsets we will need to store the absolute glyph position in the GlyphBuffer instead. Is this what you are suggesting?
Michael Catanzaro
Comment 21
2016-09-26 11:51:57 PDT
Did you intend to request a review on this, Fred?
Frédéric Wang (:fredw)
Comment 22
2016-09-26 12:20:00 PDT
(In reply to
comment #21
)
> Did you intend to request a review on this, Fred?
No, I was just uploading Khaled's test into LayoutTests. However, the patch does not change the render tree so it should probably be verified with a reftest instead.
Frédéric Wang (:fredw)
Comment 23
2016-09-26 12:59:15 PDT
Created
attachment 289849
[details]
Patch
Myles C. Maxfield
Comment 24
2016-10-03 22:54:02 PDT
(In reply to
comment #20
)
> Since advances and offsets are different things, if we need to get rid of > the separate offsets we will need to store the absolute glyph position in > the GlyphBuffer instead. Is this what you are suggesting?
Not quite. Let me approach it from a different way: If you have base advances, along with offsets, you can find absolute glyph positions. Then, once you have absolute glyph positions, you can find glyph paint-time advances by subtracting successive pairs of absolute positions. I'm suggesting putting glyph paint-time advances into the GlyphBuffer.
Myles C. Maxfield
Comment 25
2016-10-03 22:56:17 PDT
Comment on
attachment 289849
[details]
Patch r- because the the approach I detailed earlier. Please let me know if I can offer any more assistance on this patch - you can find me on the Freenode IRC server with the username "litherum". I am very excited for this work! Thanks for helping us out with this! :D
Khaled Hosny
Comment 26
2016-10-04 10:06:57 PDT
(In reply to
comment #24
)
> I'm suggesting putting glyph paint-time advances into the GlyphBuffer.
But that wouldn’t help with the bug here, since the issue is related to Y offsets, not X offsets so storing the paint advances would gain me nothing as Y offsets would still be unhandled (unless it is vertical text of course, but that is not what I’m dealing with here).
Myles C. Maxfield
Comment 27
2016-10-04 13:05:34 PDT
(In reply to
comment #26
)
> (In reply to
comment #24
) > > I'm suggesting putting glyph paint-time advances into the GlyphBuffer. > > But that wouldn’t help with the bug here, since the issue is related to Y > offsets, not X offsets so storing the paint advances would gain me nothing > as Y offsets would still be unhandled (unless it is vertical text of course, > but that is not what I’m dealing with here).
GlyphBuffer's "m_advances" member variable is a Vector of GlyphBufferAdvances. It appears that on all ports, each GlyphBufferAdvance holds both an X component and a Y component.
Khaled Hosny
Comment 28
2016-10-08 08:41:03 PDT
(In reply to
comment #27
)
> (In reply to
comment #26
) > > (In reply to
comment #24
) > > > I'm suggesting putting glyph paint-time advances into the GlyphBuffer. > > > > But that wouldn’t help with the bug here, since the issue is related to Y > > offsets, not X offsets so storing the paint advances would gain me nothing > > as Y offsets would still be unhandled (unless it is vertical text of course, > > but that is not what I’m dealing with here). > > GlyphBuffer's "m_advances" member variable is a Vector of > GlyphBufferAdvances. It appears that on all ports, each GlyphBufferAdvance > holds both an X component and a Y component.
I still don’t see how this can be used here, GlyphBufferAdvance stores width and height, I can see how width can be used, e.g. instead of doing: x = 0 y = 0 for glyph in buffer: draw(glyph, x + glyph.xoffset, y + glyph.yoffset) x += glyph.width We can resolve glyph widths to take into account x offsets and then do: x = 0 y = 0 for glyph in buffer: draw(glyph, x, y + glyph.yoffset) x += glyph.width But I don’t see how this works for y offsets since y (unlike x) is not incremented by previous glyph y advance, so whatever value we calculate for the y advance is not going to affect the y placement of the glyph, unless I’m missing something here.
Myles C. Maxfield
Comment 29
2016-10-10 14:37:19 PDT
> for glyph in buffer: > draw(glyph, x + glyph.xoffset, y + glyph.yoffset)
In WebKit, glyphs are not drawn individually. Inside FontCascade::drawGlyphBuffer(), WebKit breaks up text into runs which all use the same font, and then calls the platform-specific implementation of FontCascade::drawGlyphs() on each run. As you can see in the Windows/Cocoa implementations of FontCascade::drawGlyphs() (inside FontCGWin.cpp), a position as well as a series of X and Y advances are given to the platform drawing routine.
> y (unlike x) is not incremented by previous glyph y advance
Inside FontCascade::drawGlyphBuffer(), you can see[1] that both X and Y advances are summed to produce the position of each subsequent run. Therefore, each X advance and Y advance are treated the same way - both contribute to the position of the next run. "nextY += glyphBuffer.advanceAt(nextGlyph).height();" The semantics of the GlyphBuffer are such that the X and Y components of a glyph's advance form a vector which represents the displacement of that glyph's position to the next. Therefore, both X and Y are incremented by the advance. Each individual platform may require a different format of input (meaning: your claim may be true for the input of a particular platform's drawing routine). However, since all ports of WebKit share GlyphBuffer, its own semantics must be consistent across all ports. The semantics of GlyphBuffer are such that its advances' X and Y components form a displacement vector. If a particular platform's drawing routine requires the input to be in a different form, then that platform-specific implementation of FontCascade::drawGlyphs() must do any work to convert WebKit's GlyphBuffer format to whatever format the platform requires. Fortunately, the Mac, iOS, and Windows code all share the same format as GlyphBuffer, so those ports perform minimal conversion inside their platform-specific code. The previous paragraph describes the semantics of GlyphBuffer today. Perhaps the current semantics of GlyphBuffer can be improved to a form which matches your expectations. However, such a discussion should be discussed in a separate bug since it is a separate issue. I hope this is helpful; please let me know if you have any confusion or any questions about this. Thanks for taking such an interest in this! I am very excited to see this problem fixed :D [1]
https://trac.webkit.org/browser/trunk/Source/WebCore/platform/graphics/FontCascade.cpp#L1306
Myles C. Maxfield
Comment 30
2016-10-10 14:55:07 PDT
Currently, it looks like the Cairo implementation of FontCascade::drawGlyphs() disregards the Y component of the advances in the GlyphBuffer. FontCascade::drawGlyphs() will need to be changed to consider these Y components in order to fix this bug. In particular, the line "glyphs[i].y = point.y();" will ignore the glyph's y position and will simply force the glyph's position to be the position of the baseline. According to the Cairo documentation[1], it looks like the X and Y coordinates of the cairo_glyph_t structure (which is used in drawGlyphsToContext() by cairo_show_glyphs() to draw a run of glyphs) have a similar meaning. These seem to be a position for the glyph, which can be created by summing up all the Y advances in the GlyphBuffer. It seems that the modification to FontCascade::drawGlyphs() will be a one-line change. [1]
https://www.cairographics.org/manual/cairo-text.html#cairo-glyph-t
Michael Catanzaro
Comment 31
2017-01-27 07:16:02 PST
Khaled, any interest in updating this patch?
Khaled Hosny
Comment 32
2017-01-27 11:06:15 PST
(In reply to
comment #31
)
> Khaled, any interest in updating this patch?
Unfortunately I’m still not sure I follow the proposed changes and I don’t have much time to look into it anytime soon.
Myles C. Maxfield
Comment 33
2017-01-28 18:48:20 PST
(In reply to
comment #32
)
> (In reply to
comment #31
) > > Khaled, any interest in updating this patch? > > Unfortunately I’m still not sure I follow the proposed changes and I don’t > have much time to look into it anytime soon.
I just looked into it and I actually want to make ComplexTextController work with Uniscribe and Harfbuzz. I will try to do the Uniscribe piece tomorrow. If this works, the Windows and Linux ports will get a lot of stuff for free, including vertical glyph origins.
Michael Catanzaro
Comment 34
2017-01-28 19:37:05 PST
Thanks a bunch Myles, much appreciated!
Myles C. Maxfield
Comment 35
2017-01-31 17:03:00 PST
(In reply to
comment #33
)
> (In reply to
comment #32
) > > (In reply to
comment #31
) > > > Khaled, any interest in updating this patch? > > > > Unfortunately I’m still not sure I follow the proposed changes and I don’t > > have much time to look into it anytime soon. > > I just looked into it and I actually want to make ComplexTextController work > with Uniscribe and Harfbuzz. I will try to do the Uniscribe piece tomorrow. > If this works, the Windows and Linux ports will get a lot of stuff for free, > including vertical glyph origins.
Work in progress at
https://bugs.webkit.org/show_bug.cgi?id=167566
Khaled Hosny
Comment 36
2018-02-28 02:41:35 PST
Created
attachment 334733
[details]
Test patch Here is a different and simpler batch based on the changed to text layout since the first patch. It works most of the time except when the first glyph in the run is shifted vertically (e.g. the last character in the Arabic text is a combining mark), then the mark will be drawn at the baseline and the rest of the glyphs will be shifted relative to it (the patch has some debug code to draw a baseline to make spotting this easier). I’m not sure how to fix this. I tried to offset all the glyphs by the height of the first one, which fixes the case of the last character being a mark, but break when the before last character is a mark. I appreciate any suggestions how to move this forward.
Carlos Garcia Campos
Comment 37
2019-03-27 09:52:08 PDT
I think the problem is that ComplexTextController adds glyph origins to their advance in a single advances. But in cairo we need origins and advances separately. We need to apply the current origin when drawing, and then advance. We are using the current point as origin, and then advancing the advance + origin. In these examples there's a y origin we are ignoring. So, we have: glyphs[i] = { glyphsData[i], xOffset, yOffset }; xOffset += advances[i].width(); this should be glyphs[i] = { glyphsData[i], xOffset + xOrigin, yOffset + yOrigin }; xOffset += advances[i].width(); but we don't have the origins in GlyphBuffer.
Carlos Garcia Campos
Comment 38
2019-03-28 05:00:29 PDT
(In reply to Carlos Garcia Campos from
comment #37
)
> I think the problem is that ComplexTextController adds glyph origins to > their advance in a single advances. But in cairo we need origins and > advances separately. We need to apply the current origin when drawing, and > then advance. We are using the current point as origin, and then advancing > the advance + origin. In these examples there's a y origin we are ignoring. > > So, we have: > > glyphs[i] = { glyphsData[i], xOffset, yOffset }; > xOffset += advances[i].width(); > > this should be > > glyphs[i] = { glyphsData[i], xOffset + xOrigin, yOffset + yOrigin }; > xOffset += advances[i].width(); > > but we don't have the origins in GlyphBuffer.
Ok, now I understand how the paint advances work. The origins are taken into account in the advances, but we need to use the first origin as the initial advance of the run.
Carlos Garcia Campos
Comment 39
2019-03-28 05:05:14 PDT
Created
attachment 366166
[details]
Patch
Carlos Garcia Campos
Comment 40
2019-03-28 05:15:54 PDT
Created
attachment 366167
[details]
Patch for landing
Carlos Garcia Campos
Comment 41
2019-03-28 06:58:45 PDT
Committed
r243602
: <
https://trac.webkit.org/changeset/243602
>
Note
You need to
log in
before you can comment on or make changes to this bug.
Top of Page
Format For Printing
XML
Clone This Bug