Fix subtraction of numbers with different signs
This commit is contained in:
parent
77dca31261
commit
a2dadda511
12
README.md
12
README.md
@ -3,14 +3,18 @@
|
|||||||
[](https://travis-ci.org/caraus-ecms/tanya)
|
[](https://travis-ci.org/caraus-ecms/tanya)
|
||||||
[](https://raw.githubusercontent.com/caraus-ecms/tanya/master/LICENSE)
|
[](https://raw.githubusercontent.com/caraus-ecms/tanya/master/LICENSE)
|
||||||
|
|
||||||
tanya is a general purpose library for D programming language that doesn't rely on the Garbage Collector.
|
Tanya is a general purpose library for D programming language that doesn't
|
||||||
|
rely on the Garbage Collector.
|
||||||
|
|
||||||
# Overview
|
The library is currently in development.
|
||||||
|
|
||||||
The library consists of the following packages:
|
## Overview
|
||||||
|
|
||||||
|
Tanya consists of the following packages:
|
||||||
|
|
||||||
* `async`: Event loop.
|
* `async`: Event loop.
|
||||||
* `containers`: Queue, Vector, Singly linked list.
|
* `container`: Queue, Vector, Singly linked list.
|
||||||
* `crypto`: Work in progress TLS implementation.
|
* `crypto`: Work in progress TLS implementation.
|
||||||
|
* `math`: Multiple precision integer and a set of functions.
|
||||||
* `memory`: Tools for manual memory management (allocator, reference counting, helper functions).
|
* `memory`: Tools for manual memory management (allocator, reference counting, helper functions).
|
||||||
* `network`: URL-Parsing, sockets.
|
* `network`: URL-Parsing, sockets.
|
||||||
|
@ -18,9 +18,12 @@ import std.range;
|
|||||||
import std.traits;
|
import std.traits;
|
||||||
import tanya.memory;
|
import tanya.memory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Mutliple precision integer.
|
||||||
|
*/
|
||||||
struct Integer
|
struct Integer
|
||||||
{
|
{
|
||||||
private RefCounted!(ubyte[]) rep;
|
private RefCounted!(ubyte[]) rep;
|
||||||
private bool sign;
|
private bool sign;
|
||||||
private shared Allocator allocator;
|
private shared Allocator allocator;
|
||||||
|
|
||||||
@ -47,11 +50,7 @@ struct Integer
|
|||||||
body
|
body
|
||||||
{
|
{
|
||||||
this(allocator);
|
this(allocator);
|
||||||
|
assignInt(value);
|
||||||
T absolute = value;
|
|
||||||
immutable size = calculateSizeFromInt(absolute);
|
|
||||||
allocator.resizeArray(rep, size);
|
|
||||||
assignInt(absolute);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private @nogc unittest
|
private @nogc unittest
|
||||||
@ -93,10 +92,10 @@ struct Integer
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* Figures out the minimum amount of space this value will take
|
* Figures out the minimum amount of space this value will take
|
||||||
* up in bytes. Set the sign.
|
* up in bytes and resizes the internal storage. Sets the sign.
|
||||||
*/
|
*/
|
||||||
private ubyte calculateSizeFromInt(T)(ref T value)
|
private void assignInt(T)(in ref T value)
|
||||||
pure nothrow @safe @nogc
|
nothrow @safe @nogc
|
||||||
in
|
in
|
||||||
{
|
{
|
||||||
static assert(isIntegral!T);
|
static assert(isIntegral!T);
|
||||||
@ -104,40 +103,36 @@ struct Integer
|
|||||||
body
|
body
|
||||||
{
|
{
|
||||||
ubyte size = ulong.sizeof;
|
ubyte size = ulong.sizeof;
|
||||||
|
ulong mask;
|
||||||
|
|
||||||
static if (isSigned!T)
|
static if (isSigned!T)
|
||||||
{
|
{
|
||||||
sign = value < 0 ? true : false;
|
sign = value < 0 ? true : false;
|
||||||
value = abs(value);
|
immutable absolute = value.abs;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
sign = false;
|
sign = false;
|
||||||
|
alias absolute = value;
|
||||||
}
|
}
|
||||||
for (ulong mask = 0xff00000000000000; mask >= 0xff; mask >>= 8)
|
for (mask = 0xff00000000000000; mask >= 0xff; mask >>= 8)
|
||||||
{
|
{
|
||||||
if (value & mask)
|
if (absolute & mask)
|
||||||
{
|
{
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
--size;
|
--size;
|
||||||
}
|
}
|
||||||
return size;
|
allocator.resizeArray(rep.get, size);
|
||||||
}
|
|
||||||
|
/* Work backward through the int, masking off each byte (up to the
|
||||||
/*
|
first 0 byte) and copy it into the internal representation in
|
||||||
* Work backward through the int, masking off each byte
|
big-endian format. */
|
||||||
* (up to the first 0 byte) and copy it into the internal
|
mask = 0xff;
|
||||||
* representation in big-endian format.
|
|
||||||
*/
|
|
||||||
private void assignInt(in ulong value)
|
|
||||||
pure nothrow @safe @nogc
|
|
||||||
{
|
|
||||||
ulong mask = 0xff;
|
|
||||||
ushort shift;
|
ushort shift;
|
||||||
for (auto i = length; i; --i, mask <<= 8, shift += 8)
|
for (auto i = rep.length; i; --i, mask <<= 8, shift += 8)
|
||||||
{
|
{
|
||||||
rep[i - 1] = cast(ubyte) ((value & mask) >> shift);
|
rep[i - 1] = cast(ubyte) ((absolute & mask) >> shift);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -153,12 +148,8 @@ struct Integer
|
|||||||
ref Integer opAssign(T)(in T value) nothrow @safe @nogc
|
ref Integer opAssign(T)(in T value) nothrow @safe @nogc
|
||||||
if (isIntegral!T)
|
if (isIntegral!T)
|
||||||
{
|
{
|
||||||
T absolute = value;
|
|
||||||
immutable size = calculateSizeFromInt(absolute);
|
|
||||||
|
|
||||||
checkAllocator();
|
checkAllocator();
|
||||||
allocator.resizeArray(rep.get, size);
|
assignInt(value);
|
||||||
assignInt(absolute);
|
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
@ -278,8 +269,6 @@ struct Integer
|
|||||||
uint sum;
|
uint sum;
|
||||||
uint carry = 0;
|
uint carry = 0;
|
||||||
|
|
||||||
// Adding h2 to h1. If h2 is > h1 to begin with, resize h1
|
|
||||||
|
|
||||||
if (h.length > length)
|
if (h.length > length)
|
||||||
{
|
{
|
||||||
auto tmp = allocator.makeArray!ubyte(h.length);
|
auto tmp = allocator.makeArray!ubyte(h.length);
|
||||||
@ -311,7 +300,7 @@ struct Integer
|
|||||||
{
|
{
|
||||||
// Still overflowed; allocate more space
|
// Still overflowed; allocate more space
|
||||||
auto tmp = allocator.makeArray!ubyte(length + 1);
|
auto tmp = allocator.makeArray!ubyte(length + 1);
|
||||||
tmp[1..$] = rep[0..length];
|
tmp[1 .. $] = rep[0..length];
|
||||||
tmp[0] = 0x01;
|
tmp[0] = 0x01;
|
||||||
rep = tmp;
|
rep = tmp;
|
||||||
}
|
}
|
||||||
@ -373,6 +362,11 @@ struct Integer
|
|||||||
*/
|
*/
|
||||||
ref Integer opOpAssign(string op)(in Integer h) nothrow @safe @nogc
|
ref Integer opOpAssign(string op)(in Integer h) nothrow @safe @nogc
|
||||||
if ((op == "+") || (op == "-"))
|
if ((op == "+") || (op == "-"))
|
||||||
|
out
|
||||||
|
{
|
||||||
|
assert(!rep.length || rep[0]);
|
||||||
|
}
|
||||||
|
body
|
||||||
{
|
{
|
||||||
checkAllocator();
|
checkAllocator();
|
||||||
static if (op == "+")
|
static if (op == "+")
|
||||||
@ -414,7 +408,7 @@ struct Integer
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
subtract(h.rep);
|
add(h.rep);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return this;
|
return this;
|
||||||
@ -481,6 +475,11 @@ struct Integer
|
|||||||
/// Ditto.
|
/// Ditto.
|
||||||
ref Integer opOpAssign(string op)(in Integer h) nothrow @safe @nogc
|
ref Integer opOpAssign(string op)(in Integer h) nothrow @safe @nogc
|
||||||
if (op == "*")
|
if (op == "*")
|
||||||
|
out
|
||||||
|
{
|
||||||
|
assert(!rep.length || rep[0]);
|
||||||
|
}
|
||||||
|
body
|
||||||
{
|
{
|
||||||
auto i = h.rep.length;
|
auto i = h.rep.length;
|
||||||
auto temp = Integer(this, allocator);
|
auto temp = Integer(this, allocator);
|
||||||
@ -598,6 +597,11 @@ struct Integer
|
|||||||
/// Ditto.
|
/// Ditto.
|
||||||
ref Integer opOpAssign(string op)(in Integer exp) nothrow @safe @nogc
|
ref Integer opOpAssign(string op)(in Integer exp) nothrow @safe @nogc
|
||||||
if (op == "^^")
|
if (op == "^^")
|
||||||
|
out
|
||||||
|
{
|
||||||
|
assert(!rep.length || rep[0]);
|
||||||
|
}
|
||||||
|
body
|
||||||
{
|
{
|
||||||
auto i = exp.rep.length;
|
auto i = exp.rep.length;
|
||||||
auto tmp1 = Integer(this, allocator);
|
auto tmp1 = Integer(this, allocator);
|
||||||
@ -738,6 +742,11 @@ struct Integer
|
|||||||
*/
|
*/
|
||||||
ref Integer opUnary(string op)() nothrow @safe @nogc
|
ref Integer opUnary(string op)() nothrow @safe @nogc
|
||||||
if ((op == "++") || (op == "--"))
|
if ((op == "++") || (op == "--"))
|
||||||
|
out
|
||||||
|
{
|
||||||
|
assert(!rep.length || rep[0]);
|
||||||
|
}
|
||||||
|
body
|
||||||
{
|
{
|
||||||
checkAllocator();
|
checkAllocator();
|
||||||
|
|
||||||
@ -878,6 +887,11 @@ struct Integer
|
|||||||
*/
|
*/
|
||||||
ref Integer opOpAssign(string op)(in size_t n) nothrow @safe @nogc
|
ref Integer opOpAssign(string op)(in size_t n) nothrow @safe @nogc
|
||||||
if (op == ">>")
|
if (op == ">>")
|
||||||
|
out
|
||||||
|
{
|
||||||
|
assert(!rep.length || rep[0]);
|
||||||
|
}
|
||||||
|
body
|
||||||
{
|
{
|
||||||
immutable step = n / 8;
|
immutable step = n / 8;
|
||||||
|
|
||||||
@ -950,6 +964,11 @@ struct Integer
|
|||||||
/// Ditto.
|
/// Ditto.
|
||||||
ref Integer opOpAssign(string op)(in size_t n) nothrow @safe @nogc
|
ref Integer opOpAssign(string op)(in size_t n) nothrow @safe @nogc
|
||||||
if (op == "<<")
|
if (op == "<<")
|
||||||
|
out
|
||||||
|
{
|
||||||
|
assert(!rep.length || rep[0]);
|
||||||
|
}
|
||||||
|
body
|
||||||
{
|
{
|
||||||
ubyte carry;
|
ubyte carry;
|
||||||
auto i = rep.length;
|
auto i = rep.length;
|
||||||
|
@ -316,10 +316,6 @@ private unittest
|
|||||||
* $(D_PSYMBOL RefCounted) using $(D_PARAM args) as the parameter list for
|
* $(D_PSYMBOL RefCounted) using $(D_PARAM args) as the parameter list for
|
||||||
* the constructor of $(D_PARAM T).
|
* the constructor of $(D_PARAM T).
|
||||||
*
|
*
|
||||||
* This function is more efficient than using $(D_PSYMBOL RefCounted)
|
|
||||||
* for the new objects directly, since $(D_PSYMBOL refCounted) allocates
|
|
||||||
* only once (the object and the internal storage are allocated at once).
|
|
||||||
*
|
|
||||||
* Params:
|
* Params:
|
||||||
* T = Type of the constructed object.
|
* T = Type of the constructed object.
|
||||||
* A = Types of the arguments to the constructor of $(D_PARAM T).
|
* A = Types of the arguments to the constructor of $(D_PARAM T).
|
||||||
|
Loading…
x
Reference in New Issue
Block a user