2021-04-27 20:20:25 +09:00
|
|
|
/* Copyright 2012 Mozilla Foundation
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2021-04-27 23:18:52 +09:00
|
|
|
import { DecodeStream } from "./decode_stream.js";
|
2021-04-27 20:20:25 +09:00
|
|
|
import { isWhiteSpace } from "./core_utils.js";
|
|
|
|
|
2021-04-27 20:26:42 +09:00
|
|
|
class Ascii85Stream extends DecodeStream {
|
|
|
|
constructor(str, maybeLength) {
|
2021-04-27 20:20:25 +09:00
|
|
|
// Most streams increase in size when decoded, but Ascii85 streams
|
|
|
|
// typically shrink by ~20%.
|
|
|
|
if (maybeLength) {
|
2021-07-04 18:55:33 +09:00
|
|
|
maybeLength *= 0.8;
|
2021-04-27 20:20:25 +09:00
|
|
|
}
|
2021-04-27 20:26:42 +09:00
|
|
|
super(maybeLength);
|
2021-04-27 20:20:25 +09:00
|
|
|
|
2021-04-27 20:26:42 +09:00
|
|
|
this.str = str;
|
|
|
|
this.dict = str.dict;
|
|
|
|
this.input = new Uint8Array(5);
|
|
|
|
}
|
2021-04-27 20:20:25 +09:00
|
|
|
|
2021-04-27 20:26:42 +09:00
|
|
|
readBlock() {
|
2021-04-27 20:22:34 +09:00
|
|
|
const TILDA_CHAR = 0x7e; // '~'
|
|
|
|
const Z_LOWER_CHAR = 0x7a; // 'z'
|
|
|
|
const EOF = -1;
|
2021-04-27 20:20:25 +09:00
|
|
|
|
2021-04-27 20:22:34 +09:00
|
|
|
const str = this.str;
|
2021-04-27 20:20:25 +09:00
|
|
|
|
2021-04-27 20:22:34 +09:00
|
|
|
let c = str.getByte();
|
2021-04-27 20:20:25 +09:00
|
|
|
while (isWhiteSpace(c)) {
|
|
|
|
c = str.getByte();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c === EOF || c === TILDA_CHAR) {
|
|
|
|
this.eof = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-04-27 20:22:34 +09:00
|
|
|
const bufferLength = this.bufferLength;
|
|
|
|
let buffer, i;
|
2021-04-27 20:20:25 +09:00
|
|
|
|
|
|
|
// special code for z
|
|
|
|
if (c === Z_LOWER_CHAR) {
|
|
|
|
buffer = this.ensureBuffer(bufferLength + 4);
|
|
|
|
for (i = 0; i < 4; ++i) {
|
|
|
|
buffer[bufferLength + i] = 0;
|
|
|
|
}
|
|
|
|
this.bufferLength += 4;
|
|
|
|
} else {
|
2021-04-27 20:22:34 +09:00
|
|
|
const input = this.input;
|
2021-04-27 20:20:25 +09:00
|
|
|
input[0] = c;
|
|
|
|
for (i = 1; i < 5; ++i) {
|
|
|
|
c = str.getByte();
|
|
|
|
while (isWhiteSpace(c)) {
|
|
|
|
c = str.getByte();
|
|
|
|
}
|
|
|
|
|
|
|
|
input[i] = c;
|
|
|
|
|
|
|
|
if (c === EOF || c === TILDA_CHAR) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
buffer = this.ensureBuffer(bufferLength + i - 1);
|
|
|
|
this.bufferLength += i - 1;
|
|
|
|
|
|
|
|
// partial ending;
|
|
|
|
if (i < 5) {
|
|
|
|
for (; i < 5; ++i) {
|
|
|
|
input[i] = 0x21 + 84;
|
|
|
|
}
|
|
|
|
this.eof = true;
|
|
|
|
}
|
2021-04-27 20:22:34 +09:00
|
|
|
let t = 0;
|
2021-04-27 20:20:25 +09:00
|
|
|
for (i = 0; i < 5; ++i) {
|
|
|
|
t = t * 85 + (input[i] - 0x21);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 3; i >= 0; --i) {
|
|
|
|
buffer[bufferLength + i] = t & 0xff;
|
|
|
|
t >>= 8;
|
|
|
|
}
|
|
|
|
}
|
2021-04-27 20:26:42 +09:00
|
|
|
}
|
|
|
|
}
|
2021-04-27 20:20:25 +09:00
|
|
|
|
|
|
|
export { Ascii85Stream };
|