Fix some Unicode font embedding issues:

- Reworked Widths array compression for CID fonts to require at least 4 repeated
  widths.
- Fixed the embedded CMap for Unicode fonts.
This commit is contained in:
Michael R Sweet 2025-03-06 17:09:27 -05:00
parent 4165cd23ba
commit 458f366d78
No known key found for this signature in database
GPG Key ID: BE67C75EC81F3244
2 changed files with 34 additions and 25 deletions

View File

@ -18,6 +18,7 @@ v1.5.0 - 2025-03-06
- Fixed an output issue for extremely small `double` values with the - Fixed an output issue for extremely small `double` values with the
`pdfioContent` APIs. `pdfioContent` APIs.
- Fixed a missing Widths array issue for embedded TrueType fonts. - Fixed a missing Widths array issue for embedded TrueType fonts.
- Fixed some Unicode font embedding issues.
v1.4.1 - 2025-01-24 v1.4.1 - 2025-01-24

View File

@ -1656,8 +1656,7 @@ pdfioFileCreateFontObjFromFile(
*to_unicode; // ToUnicode dictionary *to_unicode; // ToUnicode dictionary
pdfio_obj_t *cid2gid_obj, // CIDToGIDMap object pdfio_obj_t *cid2gid_obj, // CIDToGIDMap object
*to_unicode_obj;// ToUnicode object *to_unicode_obj;// ToUnicode object
size_t i, // Looping var size_t i, j, // Looping vars
start, // Start character
num_cmap; // Number of CMap entries num_cmap; // Number of CMap entries
const int *cmap; // CMap entries const int *cmap; // CMap entries
int min_glyph, // First glyph int min_glyph, // First glyph
@ -1709,7 +1708,7 @@ pdfioFileCreateFontObjFromFile(
for (i = 0, bufptr = buffer, bufend = buffer + sizeof(buffer); i < num_cmap; i ++) for (i = 0, bufptr = buffer, bufend = buffer + sizeof(buffer); i < num_cmap; i ++)
{ {
PDFIO_DEBUG("pdfioFileCreateFontObjFromFile: cmap[%u]=%d\n", (unsigned)i, cmap[i]); PDFIO_DEBUG("pdfioFileCreateFontObjFromFile: cmap[%u]=%d\n", (unsigned)i, cmap[i]);
if (cmap[i] < 0) if (cmap[i] < 0 || cmap[i] >= (int)(sizeof(glyphs) / sizeof(glyphs[0])))
{ {
// Map undefined glyph to .notdef... // Map undefined glyph to .notdef...
*bufptr++ = 0; *bufptr++ = 0;
@ -1788,9 +1787,6 @@ pdfioFileCreateFontObjFromFile(
"1 begincodespacerange\n" "1 begincodespacerange\n"
"<0000> <FFFF>\n" "<0000> <FFFF>\n"
"endcodespacerange\n" "endcodespacerange\n"
"1 beginbfrange\n"
"<0000> <FFFF> <0000>\n"
"endbfrange\n"
"endcmap\n" "endcmap\n"
"CMapName currentdict /CMap defineresource pop\n" "CMapName currentdict /CMap defineresource pop\n"
"end\n" "end\n"
@ -1806,40 +1802,52 @@ pdfioFileCreateFontObjFromFile(
if ((w_array = pdfioArrayCreate(pdf)) == NULL) if ((w_array = pdfioArrayCreate(pdf)) == NULL)
goto done; goto done;
for (start = 0, w0 = ttfGetWidth(font, 0), w1 = 0, i = 1; i < 65536; start = i, w0 = w1, i ++) for (i = 0, w0 = ttfGetWidth(font, 0), w1 = 0; i < 65536; w0 = w1)
{ {
while (i < 65536 && (w1 = ttfGetWidth(font, (int)i)) == w0) for (j = 1; (i + j) < 65536; j ++)
i ++;
if ((i - start) > 1)
{ {
// Encode a repeating sequence... if ((w1 = ttfGetWidth(font, (int)(i + j))) != w0)
pdfioArrayAppendNumber(w_array, (double)start); break;
pdfioArrayAppendNumber(w_array, (double)(i - 1)); }
pdfioArrayAppendNumber(w_array, w0);
if (j >= 4)
{
// Encode a long sequence of zeros...
// Encode a repeating sequence...
pdfioArrayAppendNumber(w_array, (double)i);
pdfioArrayAppendNumber(w_array, (double)(i + j - 1));
pdfioArrayAppendNumber(w_array, w0);
i += j;
} }
else else
{ {
// Encode a non-repeating sequence... // Encode a non-repeating sequence...
pdfioArrayAppendNumber(w_array, (double)start); pdfioArrayAppendNumber(w_array, (double)i);
if ((temp_array = pdfioArrayCreate(pdf)) == NULL) if ((temp_array = pdfioArrayCreate(pdf)) == NULL)
goto done; goto done;
pdfioArrayAppendNumber(temp_array, w0); pdfioArrayAppendNumber(temp_array, w0);
for (w0 = w1, i ++; i < 65536; w0 = w1, i ++) for (i ++; i < 65536 && pdfioArrayGetSize(temp_array) < 8191; i ++, w0 = w1)
{ {
if ((w1 = ttfGetWidth(font, (int)i)) == w0 && i < 65535) if ((w1 = ttfGetWidth(font, (int)i)) == w0 && i < 65530)
break; {
size_t j; // Look-ahead
pdfioArrayAppendNumber(temp_array, w0); for (j = 1; j < 4; j ++)
{
if (ttfGetWidth(font, (int)(i + j)) != w0)
break;
}
if (j >= 4)
break;
}
pdfioArrayAppendNumber(temp_array, w1);
} }
if (i == 65536)
pdfioArrayAppendNumber(temp_array, w0);
else
i --;
pdfioArrayAppendArray(w_array, temp_array); pdfioArrayAppendArray(w_array, temp_array);
} }
} }