/src/qpdf/libqpdf/Pl_ASCII85Decoder.cc
Line | Count | Source |
1 | | #include <qpdf/Pl_ASCII85Decoder.hh> |
2 | | |
3 | | #include <qpdf/QTC.hh> |
4 | | #include <qpdf/Util.hh> |
5 | | |
6 | | #include <cstring> |
7 | | #include <stdexcept> |
8 | | |
9 | | using namespace qpdf; |
10 | | |
11 | | Pl_ASCII85Decoder::Pl_ASCII85Decoder(char const* identifier, Pipeline* next) : |
12 | 1.73k | Pipeline(identifier, next) |
13 | 1.73k | { |
14 | 1.73k | util::assertion(next, "Attempt to create Pl_ASCII85Decoder with nullptr as next"); |
15 | 1.73k | } |
16 | | |
17 | | void |
18 | | Pl_ASCII85Decoder::write(unsigned char const* buf, size_t len) |
19 | 2.02k | { |
20 | 2.02k | if (eod > 1) { |
21 | 194 | return; |
22 | 194 | } |
23 | 365k | for (size_t i = 0; i < len; ++i) { |
24 | 364k | switch (buf[i]) { |
25 | 763 | case ' ': |
26 | 990 | case '\f': |
27 | 1.20k | case '\v': |
28 | 1.43k | case '\t': |
29 | 2.00k | case '\r': |
30 | 3.93k | case '\n': |
31 | 3.93k | QTC::TC("libtests", "Pl_ASCII85Decoder ignore space"); |
32 | | // ignore whitespace |
33 | 3.93k | continue; |
34 | 364k | } |
35 | 360k | if (eod > 1) { |
36 | 62 | break; |
37 | 360k | } else if (eod == 1) { |
38 | 497 | util::no_ci_rt_error_if(buf[i] != '>', "broken end-of-data sequence in base 85 data"); |
39 | 497 | flush(); |
40 | 497 | eod = 2; |
41 | 360k | } else { |
42 | 360k | switch (buf[i]) { |
43 | 521 | case '~': |
44 | 521 | eod = 1; |
45 | 521 | break; |
46 | | |
47 | 1.70k | case 'z': |
48 | 1.70k | if (pos != 0) { |
49 | 31 | throw std::runtime_error("unexpected z during base 85 decode"); |
50 | 31 | } |
51 | 1.67k | unsigned char zeroes[4]; |
52 | 1.67k | memset(zeroes, '\0', 4); |
53 | 1.67k | next()->write(zeroes, 4); |
54 | 1.67k | break; |
55 | | |
56 | 357k | default: |
57 | 357k | if (buf[i] < 33 || buf[i] > 117) { |
58 | 731 | error = true; |
59 | 731 | throw std::runtime_error("character out of range during base 85 decode"); |
60 | 357k | } else { |
61 | 357k | this->inbuf[this->pos++] = buf[i]; |
62 | 357k | if (pos == 5) { |
63 | 70.9k | flush(); |
64 | 70.9k | } |
65 | 357k | } |
66 | 357k | break; |
67 | 360k | } |
68 | 360k | } |
69 | 360k | } |
70 | 1.82k | } |
71 | | |
72 | | void |
73 | | Pl_ASCII85Decoder::flush() |
74 | 72.4k | { |
75 | 72.4k | if (this->pos == 0) { |
76 | 817 | return; |
77 | 817 | } |
78 | 71.6k | unsigned long lval = 0; |
79 | 429k | for (int i = 0; i < 5; ++i) { |
80 | 358k | lval *= 85; |
81 | 358k | lval += (this->inbuf[i] - 33U); |
82 | 358k | } |
83 | | |
84 | 71.6k | unsigned char outbuf[4]; |
85 | 71.6k | memset(outbuf, 0, 4); |
86 | 358k | for (int i = 3; i >= 0; --i) { |
87 | 286k | outbuf[i] = lval & 0xff; |
88 | 286k | lval >>= 8; |
89 | 286k | } |
90 | | |
91 | 71.6k | QTC::TC("libtests", "Pl_ASCII85Decoder partial flush", (this->pos == 5) ? 0 : 1); |
92 | | // Reset before calling getNext()->write in case that throws an exception. |
93 | 71.6k | auto t = this->pos - 1; |
94 | 71.6k | this->pos = 0; |
95 | 71.6k | memset(this->inbuf, 117, 5); |
96 | | |
97 | 71.6k | next()->write(outbuf, t); |
98 | 71.6k | } |
99 | | |
100 | | void |
101 | | Pl_ASCII85Decoder::finish() |
102 | 1.53k | { |
103 | 1.53k | if (error) { |
104 | 482 | return; |
105 | 482 | } |
106 | 1.04k | flush(); |
107 | 1.04k | next()->finish(); |
108 | 1.04k | } |