PS1FontProgram.cpp 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. /*
  2. * Copyright (c) 2022, Julian Offenhäuser <offenhaeuser@protonmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibPDF/CommonNames.h>
  7. #include <LibPDF/Encoding.h>
  8. #include <LibPDF/Fonts/PS1FontProgram.h>
  9. #include <LibPDF/Reader.h>
  10. #include <ctype.h>
  11. #include <math.h>
  12. namespace PDF {
  13. PDFErrorOr<NonnullRefPtr<Type1FontProgram>> PS1FontProgram::create(ReadonlyBytes const& bytes, RefPtr<Encoding> encoding, size_t cleartext_length, size_t encrypted_length)
  14. {
  15. Reader reader(bytes);
  16. if (reader.remaining() == 0)
  17. return error("Empty font program");
  18. reader.move_to(0);
  19. if (reader.remaining() < 2 || !reader.matches("%!"))
  20. return error("Not a font program");
  21. if (!seek_name(reader, CommonNames::Encoding))
  22. return error("Missing encoding array");
  23. auto font_program = adopt_ref(*new PS1FontProgram());
  24. if (encoding) {
  25. // 9.6.6.2 Encodings for Type 1 Fonts:
  26. // An Encoding entry may override a Type 1 font’s mapping from character codes to character names.
  27. font_program->set_encoding(move(encoding));
  28. } else {
  29. if (TRY(parse_word(reader)) == "StandardEncoding") {
  30. font_program->set_encoding(Encoding::standard_encoding());
  31. } else {
  32. auto encoding = Encoding::create();
  33. while (reader.remaining()) {
  34. auto word = TRY(parse_word(reader));
  35. if (word == "readonly") {
  36. break;
  37. } else if (word == "dup") {
  38. u8 char_code = TRY(parse_int(reader));
  39. auto name = TRY(parse_word(reader));
  40. encoding->set(char_code, name.starts_with('/') ? name.substring_view(1) : name.view());
  41. }
  42. }
  43. font_program->set_encoding(move(encoding));
  44. }
  45. }
  46. bool found_font_matrix = seek_name(reader, "FontMatrix");
  47. if (found_font_matrix) {
  48. auto array = TRY(parse_number_array(reader, 6));
  49. font_program->set_font_matrix({ array[0], array[1], array[2], array[3], array[4], array[5] });
  50. } else {
  51. font_program->set_font_matrix({ 0.001f, 0.0f, 0.0f, 0.001f, 0.0f, 0.0f });
  52. }
  53. auto decrypted = TRY(decrypt(reader.bytes().slice(cleartext_length, encrypted_length), 55665, 4));
  54. TRY(font_program->parse_encrypted_portion(decrypted));
  55. return font_program;
  56. }
  57. PDFErrorOr<void> PS1FontProgram::parse_encrypted_portion(ByteBuffer const& buffer)
  58. {
  59. Reader reader(buffer);
  60. if (seek_name(reader, "lenIV"))
  61. m_lenIV = TRY(parse_int(reader));
  62. Vector<ByteBuffer> subroutines;
  63. if (seek_name(reader, "Subrs"))
  64. subroutines = TRY(parse_subroutines(reader));
  65. if (!seek_name(reader, "CharStrings"))
  66. return error("Missing char strings array");
  67. while (reader.remaining()) {
  68. auto word = TRY(parse_word(reader));
  69. VERIFY(!word.is_empty());
  70. if (word == "end")
  71. break;
  72. if (word[0] == '/') {
  73. auto encrypted_size = TRY(parse_int(reader));
  74. auto rd = TRY(parse_word(reader));
  75. if (rd == "-|" || rd == "RD") {
  76. auto line = TRY(decrypt(reader.bytes().slice(reader.offset(), encrypted_size), m_encryption_key, m_lenIV));
  77. reader.move_by(encrypted_size);
  78. auto glyph_name = word.substring_view(1);
  79. GlyphParserState state;
  80. TRY(add_glyph(glyph_name, TRY(parse_glyph(line, subroutines, {}, state, false))));
  81. }
  82. }
  83. }
  84. consolidate_glyphs();
  85. return {};
  86. }
  87. PDFErrorOr<Vector<ByteBuffer>> PS1FontProgram::parse_subroutines(Reader& reader) const
  88. {
  89. if (!reader.matches_number())
  90. return error("Expected array length");
  91. auto length = TRY(parse_int(reader));
  92. VERIFY(length >= 0);
  93. Vector<ByteBuffer> array;
  94. TRY(array.try_resize(length));
  95. while (reader.remaining()) {
  96. auto word = TRY(parse_word(reader));
  97. if (word.is_empty())
  98. VERIFY(0);
  99. if (word == "dup") {
  100. auto index = TRY(parse_int(reader));
  101. auto entry = TRY(parse_word(reader));
  102. if (entry.is_empty())
  103. return error("Empty array entry");
  104. if (index >= length)
  105. return error("Array index out of bounds");
  106. if (isdigit(entry[0])) {
  107. auto maybe_encrypted_size = entry.to_int();
  108. if (!maybe_encrypted_size.has_value())
  109. return error("Malformed array");
  110. auto rd = TRY(parse_word(reader));
  111. if (rd == "-|" || rd == "RD") {
  112. array[index] = TRY(decrypt(reader.bytes().slice(reader.offset(), maybe_encrypted_size.value()), m_encryption_key, m_lenIV));
  113. reader.move_by(maybe_encrypted_size.value());
  114. }
  115. } else {
  116. array[index] = TRY(ByteBuffer::copy(entry.bytes()));
  117. }
  118. } else if (word == "index" || word == "def" || word == "ND") {
  119. break;
  120. }
  121. }
  122. return array;
  123. }
  124. PDFErrorOr<Vector<float>> PS1FontProgram::parse_number_array(Reader& reader, size_t length)
  125. {
  126. Vector<float> array;
  127. TRY(array.try_resize(length));
  128. reader.consume_whitespace();
  129. if (!reader.consume('['))
  130. return error("Expected array to start with '['");
  131. reader.consume_whitespace();
  132. for (size_t i = 0; i < length; ++i)
  133. array.at(i) = TRY(parse_float(reader));
  134. if (!reader.consume(']'))
  135. return error("Expected array to end with ']'");
  136. return array;
  137. }
  138. PDFErrorOr<DeprecatedString> PS1FontProgram::parse_word(Reader& reader)
  139. {
  140. reader.consume_whitespace();
  141. auto start = reader.offset();
  142. reader.move_while([&](char c) {
  143. return !reader.matches_whitespace() && c != '[' && c != ']';
  144. });
  145. auto end = reader.offset();
  146. if (reader.matches_whitespace())
  147. reader.consume();
  148. return StringView(reader.bytes().data() + start, end - start);
  149. }
  150. PDFErrorOr<float> PS1FontProgram::parse_float(Reader& reader)
  151. {
  152. auto word = TRY(parse_word(reader));
  153. return strtof(DeprecatedString(word).characters(), nullptr);
  154. }
  155. PDFErrorOr<int> PS1FontProgram::parse_int(Reader& reader)
  156. {
  157. auto maybe_int = TRY(parse_word(reader)).to_int();
  158. if (!maybe_int.has_value())
  159. return error("Invalid int");
  160. return maybe_int.value();
  161. }
  162. PDFErrorOr<ByteBuffer> PS1FontProgram::decrypt(ReadonlyBytes const& encrypted, u16 key, size_t skip)
  163. {
  164. auto decrypted = TRY(ByteBuffer::create_uninitialized(encrypted.size() - skip));
  165. u16 R = key;
  166. u16 c1 = 52845;
  167. u16 c2 = 22719;
  168. for (size_t i = 0; i < encrypted.size(); ++i) {
  169. u8 C = encrypted[i];
  170. u8 P = C ^ (R >> 8);
  171. R = (C + R) * c1 + c2;
  172. if (i >= skip)
  173. decrypted[i - skip] = P;
  174. }
  175. return decrypted;
  176. }
  177. bool PS1FontProgram::seek_name(Reader& reader, DeprecatedString const& name)
  178. {
  179. auto start = reader.offset();
  180. reader.move_to(0);
  181. while (reader.remaining()) {
  182. if (reader.consume('/') && reader.matches(name.characters())) {
  183. // Skip name
  184. reader.move_while([&](char) {
  185. return reader.matches_regular_character();
  186. });
  187. reader.consume_whitespace();
  188. return true;
  189. }
  190. }
  191. // Jump back to where we started
  192. reader.move_to(start);
  193. return false;
  194. }
  195. }