Skip to content

Commit 7fc776e

Browse files
Daniel Gredlerprrace
Daniel Gredler
authored andcommittedMar 13, 2025
8270265: LineBreakMeasurer calculates incorrect line breaks with zero-width characters
Reviewed-by: achung, prr
1 parent 3da5e3f commit 7fc776e

File tree

6 files changed

+22
-10
lines changed

6 files changed

+22
-10
lines changed
 

‎src/java.desktop/share/classes/sun/font/ExtendedTextSourceLabel.java

+4-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1998, 2017, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1998, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -493,10 +493,9 @@ public int getLineBreakIndex(int start, float width) {
493493
--start;
494494
while (width >= -epsilon && ++start < length) {
495495
int cidx = l2v(start) * numvals + advx;
496-
if (cidx >= charinfo.length) {
497-
break; // layout bailed for some reason
498-
}
499-
float adv = charinfo[cidx];
496+
float adv = cidx < charinfo.length ?
497+
charinfo[cidx] : // layout provided info for this glyph
498+
0; // glyph info omitted, assume no advance
500499
if (adv != 0) {
501500
width -= adv + advTracking;
502501
}

‎src/java.desktop/share/native/libfontmanager/HBShaper.c

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -272,6 +272,7 @@ JNIEXPORT jboolean JNICALL Java_sun_font_SunLayoutEngine_shape
272272

273273
buffer = hb_buffer_create();
274274
hb_buffer_set_script(buffer, getHBScriptCode(script));
275+
hb_buffer_set_invisible_glyph(buffer, INVISIBLE_GLYPH_ID);
275276
hb_buffer_set_language(buffer,
276277
hb_ot_tag_to_language(HB_OT_TAG_DEFAULT_LANGUAGE));
277278
if ((flags & TYPO_RTL) != 0) {

‎src/java.desktop/share/native/libfontmanager/HBShaper_Panama.c

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2023, 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2023, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -107,6 +107,7 @@ JDKEXPORT void jdk_hb_shape(
107107

108108
buffer = hb_buffer_create();
109109
hb_buffer_set_script(buffer, getHBScriptCode(script));
110+
hb_buffer_set_invisible_glyph(buffer, INVISIBLE_GLYPH_ID);
110111
hb_buffer_set_language(buffer,
111112
hb_ot_tag_to_language(HB_OT_TAG_DEFAULT_LANGUAGE));
112113
if ((flags & TYPO_RTL) != 0) {

‎src/java.desktop/share/native/libfontmanager/hb-jdk-p.h

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2023, 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2023, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -48,6 +48,8 @@
4848
extern "C" {
4949
#endif
5050

51+
// Matches sun.font.CharToGlyphMapper.INVISIBLE_GLYPH_ID
52+
#define INVISIBLE_GLYPH_ID 0xffff
5153

5254
hb_font_t* jdk_font_create_hbp(
5355
hb_face_t* face,

‎src/java.desktop/share/native/libfontmanager/hb-jdk.h

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -50,6 +50,9 @@ typedef struct JDKFontInfo_Struct {
5050
#define HBFloatToFixedScale ((float)(1 << 16))
5151
#define HBFloatToFixed(f) ((unsigned int)((f) * HBFloatToFixedScale))
5252

53+
// Matches sun.font.CharToGlyphMapper.INVISIBLE_GLYPH_ID
54+
#define INVISIBLE_GLYPH_ID 0xffff
55+
5356
/*
5457
* Note:
5558
*

‎test/jdk/java/awt/font/TextLayout/FormatCharAdvanceTest.java

+7-1
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@
2323

2424
/**
2525
* @test
26-
* @bug 8208377 6562489
26+
* @bug 8208377 6562489 8270265
2727
* @summary Confirm that format-category glyphs are not rendered or measured.
2828
*/
2929

@@ -275,6 +275,12 @@ private static void testChar(char c, BufferedImage image, Graphics2D g2d, Font f
275275
g2d.drawString(as2.getIterator(), w / 2, h / 2);
276276
ab2 = findTextBoundingBox(image).width;
277277
assertEqual(ab1, ab2, "drawString (using AttributedCharacterIterator)", c, font);
278+
279+
int max = metrics.stringWidth("AB") + 2; // add a little wiggle room to the max width
280+
LineBreakMeasurer measurer1 = new LineBreakMeasurer(as1.getIterator(), frc);
281+
LineBreakMeasurer measurer2 = new LineBreakMeasurer(as2.getIterator(), frc);
282+
assertEqual(2, measurer1.nextOffset(max), "nextOffset 1", c, font);
283+
assertEqual(7, measurer2.nextOffset(max), "nextOffset 2", c, font);
278284
}
279285

280286
private static void assertEqual(int i1, int i2, String scenario, char c, Font font) {

1 commit comments

Comments
 (1)

openjdk-notifier[bot] commented on Mar 13, 2025

@openjdk-notifier[bot]
Please sign in to comment.