2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* Bit rotation operations.
|
|
|
|
*
|
2016-12-06 18:43:55 +00:00
|
|
|
* @author n1474335 [n1474335@gmail.com]
|
2016-11-28 10:42:58 +00:00
|
|
|
* @copyright Crown Copyright 2016
|
|
|
|
* @license Apache-2.0
|
|
|
|
*
|
|
|
|
* @namespace
|
|
|
|
*
|
|
|
|
* @todo Support for UTF16
|
|
|
|
*/
|
2017-03-23 17:52:20 +00:00
|
|
|
const Rotate = {
|
2016-11-28 10:42:58 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @constant
|
|
|
|
* @default
|
|
|
|
*/
|
|
|
|
ROTATE_AMOUNT: 1,
|
|
|
|
/**
|
|
|
|
* @constant
|
|
|
|
* @default
|
|
|
|
*/
|
|
|
|
ROTATE_WHOLE: false,
|
2017-02-09 15:09:33 +00:00
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* Runs rotation operations across the input data.
|
|
|
|
*
|
|
|
|
* @private
|
2017-01-31 18:24:56 +00:00
|
|
|
* @param {byteArray} data
|
2016-11-28 10:42:58 +00:00
|
|
|
* @param {number} amount
|
|
|
|
* @param {function} algo - The rotation operation to carry out
|
2017-01-31 18:24:56 +00:00
|
|
|
* @returns {byteArray}
|
2016-11-28 10:42:58 +00:00
|
|
|
*/
|
|
|
|
_rot: function(data, amount, algo) {
|
2017-04-13 18:08:50 +01:00
|
|
|
const result = [];
|
|
|
|
for (let i = 0; i < data.length; i++) {
|
|
|
|
let b = data[i];
|
|
|
|
for (let j = 0; j < amount; j++) {
|
2016-11-28 10:42:58 +00:00
|
|
|
b = algo(b);
|
|
|
|
}
|
|
|
|
result.push(b);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
},
|
2017-02-09 15:09:33 +00:00
|
|
|
|
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* Rotate right operation.
|
|
|
|
*
|
2017-01-31 18:24:56 +00:00
|
|
|
* @param {byteArray} input
|
2016-11-28 10:42:58 +00:00
|
|
|
* @param {Object[]} args
|
2017-01-31 18:24:56 +00:00
|
|
|
* @returns {byteArray}
|
2016-11-28 10:42:58 +00:00
|
|
|
*/
|
2017-01-31 18:24:56 +00:00
|
|
|
runRotr: function(input, args) {
|
2016-11-28 10:42:58 +00:00
|
|
|
if (args[1]) {
|
2017-01-31 18:24:56 +00:00
|
|
|
return Rotate._rotrWhole(input, args[0]);
|
2016-11-28 10:42:58 +00:00
|
|
|
} else {
|
|
|
|
return Rotate._rot(input, args[0], Rotate._rotr);
|
|
|
|
}
|
|
|
|
},
|
2017-02-09 15:09:33 +00:00
|
|
|
|
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* Rotate left operation.
|
|
|
|
*
|
2017-01-31 18:24:56 +00:00
|
|
|
* @param {byteArray} input
|
2016-11-28 10:42:58 +00:00
|
|
|
* @param {Object[]} args
|
2017-01-31 18:24:56 +00:00
|
|
|
* @returns {byteArray}
|
2016-11-28 10:42:58 +00:00
|
|
|
*/
|
2017-01-31 18:24:56 +00:00
|
|
|
runRotl: function(input, args) {
|
2016-11-28 10:42:58 +00:00
|
|
|
if (args[1]) {
|
2017-01-31 18:24:56 +00:00
|
|
|
return Rotate._rotlWhole(input, args[0]);
|
2016-11-28 10:42:58 +00:00
|
|
|
} else {
|
|
|
|
return Rotate._rot(input, args[0], Rotate._rotl);
|
|
|
|
}
|
|
|
|
},
|
2017-02-09 15:09:33 +00:00
|
|
|
|
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* @constant
|
|
|
|
* @default
|
|
|
|
*/
|
|
|
|
ROT13_AMOUNT: 13,
|
|
|
|
/**
|
|
|
|
* @constant
|
|
|
|
* @default
|
|
|
|
*/
|
|
|
|
ROT13_LOWERCASE: true,
|
|
|
|
/**
|
|
|
|
* @constant
|
|
|
|
* @default
|
|
|
|
*/
|
|
|
|
ROT13_UPPERCASE: true,
|
2016-11-29 17:53:31 +00:00
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* ROT13 operation.
|
|
|
|
*
|
2017-01-31 18:24:56 +00:00
|
|
|
* @param {byteArray} input
|
2016-11-28 10:42:58 +00:00
|
|
|
* @param {Object[]} args
|
2017-01-31 18:24:56 +00:00
|
|
|
* @returns {byteArray}
|
2016-11-28 10:42:58 +00:00
|
|
|
*/
|
2017-01-31 18:24:56 +00:00
|
|
|
runRot13: function(input, args) {
|
2017-04-13 18:08:50 +01:00
|
|
|
let amount = args[2],
|
2016-11-28 10:42:58 +00:00
|
|
|
output = input,
|
|
|
|
chr,
|
2017-01-31 18:24:56 +00:00
|
|
|
rot13Lowercase = args[0],
|
|
|
|
rot13Upperacse = args[1];
|
2017-02-09 15:09:33 +00:00
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
if (amount) {
|
|
|
|
if (amount < 0) {
|
|
|
|
amount = 26 - (Math.abs(amount) % 26);
|
|
|
|
}
|
2017-02-09 15:09:33 +00:00
|
|
|
|
2017-04-13 18:08:50 +01:00
|
|
|
for (let i = 0; i < input.length; i++) {
|
2016-11-28 10:42:58 +00:00
|
|
|
chr = input[i];
|
2017-01-31 18:24:56 +00:00
|
|
|
if (rot13Upperacse && chr >= 65 && chr <= 90) { // Upper case
|
2016-11-28 10:42:58 +00:00
|
|
|
chr = (chr - 65 + amount) % 26;
|
|
|
|
output[i] = chr + 65;
|
2017-01-31 18:24:56 +00:00
|
|
|
} else if (rot13Lowercase && chr >= 97 && chr <= 122) { // Lower case
|
2016-11-28 10:42:58 +00:00
|
|
|
chr = (chr - 97 + amount) % 26;
|
|
|
|
output[i] = chr + 97;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
},
|
2016-11-29 17:53:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @constant
|
|
|
|
* @default
|
|
|
|
*/
|
|
|
|
ROT47_AMOUNT: 47,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ROT47 operation.
|
|
|
|
*
|
2016-12-06 18:43:55 +00:00
|
|
|
* @author Matt C [matt@artemisbot.pw]
|
2017-01-31 18:24:56 +00:00
|
|
|
* @param {byteArray} input
|
2016-11-29 17:53:31 +00:00
|
|
|
* @param {Object[]} args
|
2017-01-31 18:24:56 +00:00
|
|
|
* @returns {byteArray}
|
2016-11-29 17:53:31 +00:00
|
|
|
*/
|
2017-01-31 18:24:56 +00:00
|
|
|
runRot47: function(input, args) {
|
2017-04-13 18:08:50 +01:00
|
|
|
let amount = args[0],
|
2016-11-29 17:53:31 +00:00
|
|
|
output = input,
|
|
|
|
chr;
|
|
|
|
|
|
|
|
if (amount) {
|
|
|
|
if (amount < 0) {
|
|
|
|
amount = 94 - (Math.abs(amount) % 94);
|
|
|
|
}
|
|
|
|
|
2017-04-13 18:08:50 +01:00
|
|
|
for (let i = 0; i < input.length; i++) {
|
2016-11-29 17:53:31 +00:00
|
|
|
chr = input[i];
|
2016-12-03 10:19:26 +00:00
|
|
|
if (chr >= 33 && chr <= 126) {
|
2016-11-29 17:53:31 +00:00
|
|
|
chr = (chr - 33 + amount) % 94;
|
|
|
|
output[i] = chr + 33;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
},
|
|
|
|
|
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* Rotate right bitwise op.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @param {byte} b
|
|
|
|
* @returns {byte}
|
|
|
|
*/
|
|
|
|
_rotr: function(b) {
|
2017-04-13 18:08:50 +01:00
|
|
|
const bit = (b & 1) << 7;
|
2016-11-28 10:42:58 +00:00
|
|
|
return (b >> 1) | bit;
|
|
|
|
},
|
2017-02-09 15:09:33 +00:00
|
|
|
|
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* Rotate left bitwise op.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @param {byte} b
|
|
|
|
* @returns {byte}
|
|
|
|
*/
|
|
|
|
_rotl: function(b) {
|
2017-04-13 18:08:50 +01:00
|
|
|
const bit = (b >> 7) & 1;
|
2016-11-28 10:42:58 +00:00
|
|
|
return ((b << 1) | bit) & 0xFF;
|
|
|
|
},
|
2017-02-09 15:09:33 +00:00
|
|
|
|
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* Rotates a byte array to the right by a specific amount as a whole, so that bits are wrapped
|
|
|
|
* from the end of the array to the beginning.
|
|
|
|
*
|
|
|
|
* @private
|
2017-01-31 18:24:56 +00:00
|
|
|
* @param {byteArray} data
|
2016-11-28 10:42:58 +00:00
|
|
|
* @param {number} amount
|
2017-01-31 18:24:56 +00:00
|
|
|
* @returns {byteArray}
|
2016-11-28 10:42:58 +00:00
|
|
|
*/
|
2017-01-31 18:24:56 +00:00
|
|
|
_rotrWhole: function(data, amount) {
|
2017-04-13 18:08:50 +01:00
|
|
|
let carryBits = 0,
|
2017-01-31 18:24:56 +00:00
|
|
|
newByte,
|
2016-11-28 10:42:58 +00:00
|
|
|
result = [];
|
2017-02-09 15:09:33 +00:00
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
amount = amount % 8;
|
2017-04-13 18:08:50 +01:00
|
|
|
for (let i = 0; i < data.length; i++) {
|
|
|
|
const oldByte = data[i] >>> 0;
|
2017-01-31 18:24:56 +00:00
|
|
|
newByte = (oldByte >> amount) | carryBits;
|
|
|
|
carryBits = (oldByte & (Math.pow(2, amount)-1)) << (8-amount);
|
|
|
|
result.push(newByte);
|
2016-11-28 10:42:58 +00:00
|
|
|
}
|
2017-01-31 18:24:56 +00:00
|
|
|
result[0] |= carryBits;
|
2016-11-28 10:42:58 +00:00
|
|
|
return result;
|
|
|
|
},
|
2017-02-09 15:09:33 +00:00
|
|
|
|
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
/**
|
|
|
|
* Rotates a byte array to the left by a specific amount as a whole, so that bits are wrapped
|
|
|
|
* from the beginning of the array to the end.
|
|
|
|
*
|
|
|
|
* @private
|
2017-01-31 18:24:56 +00:00
|
|
|
* @param {byteArray} data
|
2016-11-28 10:42:58 +00:00
|
|
|
* @param {number} amount
|
2017-01-31 18:24:56 +00:00
|
|
|
* @returns {byteArray}
|
2016-11-28 10:42:58 +00:00
|
|
|
*/
|
2017-01-31 18:24:56 +00:00
|
|
|
_rotlWhole: function(data, amount) {
|
2017-04-13 18:08:50 +01:00
|
|
|
let carryBits = 0,
|
2017-01-31 18:24:56 +00:00
|
|
|
newByte,
|
2016-11-28 10:42:58 +00:00
|
|
|
result = [];
|
2017-02-09 15:09:33 +00:00
|
|
|
|
2016-11-28 10:42:58 +00:00
|
|
|
amount = amount % 8;
|
2017-04-13 18:08:50 +01:00
|
|
|
for (let i = data.length-1; i >= 0; i--) {
|
|
|
|
const oldByte = data[i];
|
2017-01-31 18:24:56 +00:00
|
|
|
newByte = ((oldByte << amount) | carryBits) & 0xFF;
|
|
|
|
carryBits = (oldByte >> (8-amount)) & (Math.pow(2, amount)-1);
|
|
|
|
result[i] = (newByte);
|
2016-11-28 10:42:58 +00:00
|
|
|
}
|
2017-01-31 18:24:56 +00:00
|
|
|
result[data.length-1] = result[data.length-1] | carryBits;
|
2016-11-28 10:42:58 +00:00
|
|
|
return result;
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
2017-03-23 17:52:20 +00:00
|
|
|
|
|
|
|
export default Rotate;
|