2019-01-07 04:26:18 +09:00
|
|
|
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
|
|
|
|
import { BufReader } from "./bufio.ts";
|
2019-03-08 09:25:16 +09:00
|
|
|
type Reader = Deno.Reader;
|
|
|
|
type Writer = Deno.Writer;
|
2019-03-06 22:39:50 +01:00
|
|
|
import { assert } from "../testing/asserts.ts";
|
2019-01-07 04:26:18 +09:00
|
|
|
|
2019-02-11 08:49:48 +09:00
|
|
|
/** copy N size at the most. If read size is lesser than N, then returns nread */
|
|
|
|
export async function copyN(
|
|
|
|
dest: Writer,
|
|
|
|
r: Reader,
|
|
|
|
size: number
|
|
|
|
): Promise<number> {
|
|
|
|
let bytesRead = 0;
|
|
|
|
let buf = new Uint8Array(1024);
|
|
|
|
while (bytesRead < size) {
|
|
|
|
if (size - bytesRead < 1024) {
|
|
|
|
buf = new Uint8Array(size - bytesRead);
|
|
|
|
}
|
|
|
|
const { nread, eof } = await r.read(buf);
|
|
|
|
bytesRead += nread;
|
|
|
|
if (nread > 0) {
|
|
|
|
const n = await dest.write(buf.slice(0, nread));
|
2019-03-06 22:39:50 +01:00
|
|
|
assert(n === nread, "could not write");
|
2019-02-11 08:49:48 +09:00
|
|
|
}
|
|
|
|
if (eof) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return bytesRead;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Read big endian 16bit short from BufReader */
|
2019-01-07 04:26:18 +09:00
|
|
|
export async function readShort(buf: BufReader): Promise<number> {
|
|
|
|
const [high, low] = [await buf.readByte(), await buf.readByte()];
|
|
|
|
return (high << 8) | low;
|
|
|
|
}
|
|
|
|
|
2019-02-11 08:49:48 +09:00
|
|
|
/** Read big endian 32bit integer from BufReader */
|
2019-01-07 04:26:18 +09:00
|
|
|
export async function readInt(buf: BufReader): Promise<number> {
|
|
|
|
const [high, low] = [await readShort(buf), await readShort(buf)];
|
|
|
|
return (high << 16) | low;
|
|
|
|
}
|
|
|
|
|
|
|
|
const BIT32 = 0xffffffff;
|
2019-02-11 08:49:48 +09:00
|
|
|
|
|
|
|
/** Read big endian 64bit long from BufReader */
|
2019-01-07 04:26:18 +09:00
|
|
|
export async function readLong(buf: BufReader): Promise<number> {
|
|
|
|
const [high, low] = [await readInt(buf), await readInt(buf)];
|
|
|
|
// ECMAScript doesn't support 64bit bit ops.
|
|
|
|
return high ? high * (BIT32 + 1) + low : low;
|
|
|
|
}
|
|
|
|
|
2019-02-11 08:49:48 +09:00
|
|
|
/** Slice number into 64bit big endian byte array */
|
2019-01-07 04:26:18 +09:00
|
|
|
export function sliceLongToBytes(d: number, dest = new Array(8)): number[] {
|
|
|
|
let mask = 0xff;
|
|
|
|
let low = (d << 32) >>> 32;
|
|
|
|
let high = (d - low) / (BIT32 + 1);
|
|
|
|
let shift = 24;
|
|
|
|
for (let i = 0; i < 4; i++) {
|
|
|
|
dest[i] = (high >>> shift) & mask;
|
|
|
|
dest[i + 4] = (low >>> shift) & mask;
|
|
|
|
shift -= 8;
|
|
|
|
}
|
|
|
|
return dest;
|
|
|
|
}
|