#include <libecc/nn/nn_logical.h>
#include <libecc/nn/nn.h>
int nn_lshift_fixedlen(nn_t out, nn_src_t in, bitcnt_t cnt)
{
int ipos, opos, dec, ret;
bitcnt_t lshift, hshift;
u8 owlen, iwlen;
ret = nn_check_initialized(in); EG(ret, err);
ret = nn_check_initialized(out); EG(ret, err);
owlen = out->wlen;
iwlen = in->wlen;
dec = cnt / WORD_BITS;
hshift = cnt % WORD_BITS;
lshift = (bitcnt_t)(WORD_BITS - hshift);
for (opos = owlen - 1; opos >= 0; opos--) {
word_t hipart = 0, lopart = 0;
ipos = opos - dec - 1;
if ((ipos >= 0) && (ipos < iwlen)) {
lopart = WRSHIFT(in->val[ipos], lshift);
}
ipos = opos - dec;
if ((ipos >= 0) && (ipos < iwlen)) {
hipart = WLSHIFT(in->val[ipos], hshift);
}
out->val[opos] = hipart | lopart;
}
err:
return ret;
}
int nn_lshift(nn_t out, nn_src_t in, bitcnt_t cnt)
{
bitcnt_t lshift, hshift, blen;
int ipos, opos, dec, ret;
u8 owlen, iwlen;
ret = nn_check_initialized(in); EG(ret, err);
iwlen = in->wlen;
if (out != in) {
ret = nn_init(out, 0); EG(ret, err);
}
ret = nn_bitlen(in, &blen); EG(ret, err);
owlen = (u8)LOCAL_MIN(BIT_LEN_WORDS(cnt + blen),
BIT_LEN_WORDS(NN_MAX_BIT_LEN));
out->wlen = owlen;
dec = cnt / WORD_BITS;
hshift = cnt % WORD_BITS;
lshift = (bitcnt_t)(WORD_BITS - hshift);
for (opos = owlen - 1; opos >= 0; opos--) {
word_t hipart = 0, lopart = 0;
ipos = opos - dec - 1;
if ((ipos >= 0) && (ipos < iwlen)) {
lopart = WRSHIFT(in->val[ipos], lshift);
}
ipos = opos - dec;
if ((ipos >= 0) && (ipos < iwlen)) {
hipart = WLSHIFT(in->val[ipos], hshift);
}
out->val[opos] = hipart | lopart;
}
err:
return ret;
}
int nn_rshift_fixedlen(nn_t out, nn_src_t in, bitcnt_t cnt)
{
int ipos, opos, dec, ret;
bitcnt_t lshift, hshift;
u8 owlen, iwlen;
ret = nn_check_initialized(in); EG(ret, err);
ret = nn_check_initialized(out); EG(ret, err);
owlen = out->wlen;
iwlen = in->wlen;
dec = cnt / WORD_BITS;
lshift = cnt % WORD_BITS;
hshift = (bitcnt_t)(WORD_BITS - lshift);
for (opos = 0; opos < owlen; opos++) {
word_t hipart = 0, lopart = 0;
ipos = opos + dec;
if ((ipos >= 0) && (ipos < iwlen)) {
lopart = WRSHIFT(in->val[ipos], lshift);
}
ipos = opos + dec + 1;
if ((ipos >= 0) && (ipos < iwlen)) {
hipart = WLSHIFT(in->val[ipos], hshift);
}
out->val[opos] = hipart | lopart;
}
err:
return ret;
}
int nn_rshift(nn_t out, nn_src_t in, bitcnt_t cnt)
{
int ipos, opos, dec, ret;
bitcnt_t lshift, hshift;
u8 owlen, iwlen;
bitcnt_t blen;
ret = nn_check_initialized(in); EG(ret, err);
iwlen = in->wlen;
if (out != in) {
ret = nn_init(out, 0); EG(ret, err);
}
dec = cnt / WORD_BITS;
lshift = cnt % WORD_BITS;
hshift = (bitcnt_t)(WORD_BITS - lshift);
ret = nn_bitlen(in, &blen); EG(ret, err);
if (cnt > blen) {
owlen = 0;
} else {
owlen = (u8)BIT_LEN_WORDS(blen - cnt);
}
out->wlen = owlen;
for (opos = 0; opos < owlen; opos++) {
word_t hipart = 0, lopart = 0;
ipos = opos + dec;
if ((ipos >= 0) && (ipos < iwlen)) {
lopart = WRSHIFT(in->val[ipos], lshift);
}
ipos = opos + dec + 1;
if ((ipos >= 0) && (ipos < iwlen)) {
hipart = WLSHIFT(in->val[ipos], hshift);
}
out->val[opos] = hipart | lopart;
}
for (opos = owlen; opos < NN_MAX_WORD_LEN; opos++) {
out->val[opos] = 0;
}
err:
return ret;
}
int nn_rrot(nn_t out, nn_src_t in, bitcnt_t cnt, bitcnt_t bitlen)
{
u8 owlen = (u8)BIT_LEN_WORDS(bitlen);
int ret;
nn tmp;
tmp.magic = WORD(0);
MUST_HAVE((bitlen <= NN_MAX_BIT_LEN), ret, err);
MUST_HAVE((cnt < bitlen), ret, err);
ret = nn_check_initialized(in); EG(ret, err);
ret = nn_init(&tmp, 0); EG(ret, err);
ret = nn_lshift(&tmp, in, (bitcnt_t)(bitlen - cnt)); EG(ret, err);
ret = nn_set_wlen(&tmp, owlen); EG(ret, err);
ret = nn_rshift(out, in, cnt); EG(ret, err);
ret = nn_set_wlen(out, owlen); EG(ret, err);
ret = nn_xor(out, out, &tmp); EG(ret, err);
if (((bitlen % WORD_BITS) != 0) && (out->wlen > 0)) {
word_t mask = (word_t)(((word_t)(WORD(1) << (bitlen % WORD_BITS))) - 1);
out->val[out->wlen - 1] &= mask;
}
err:
nn_uninit(&tmp);
return ret;
}
int nn_lrot(nn_t out, nn_src_t in, bitcnt_t cnt, bitcnt_t bitlen)
{
u8 owlen = (u8)BIT_LEN_WORDS(bitlen);
int ret;
nn tmp;
tmp.magic = WORD(0);
MUST_HAVE(!(bitlen > NN_MAX_BIT_LEN), ret, err);
MUST_HAVE(!(cnt >= bitlen), ret, err);
ret = nn_check_initialized(in); EG(ret, err);
ret = nn_init(&tmp, 0); EG(ret, err);
ret = nn_lshift(&tmp, in, cnt); EG(ret, err);
ret = nn_set_wlen(&tmp, owlen); EG(ret, err);
ret = nn_rshift(out, in, (bitcnt_t)(bitlen - cnt)); EG(ret, err);
ret = nn_set_wlen(out, owlen); EG(ret, err);
ret = nn_xor(out, out, &tmp); EG(ret, err);
if (((bitlen % WORD_BITS) != 0) && (out->wlen > 0)) {
word_t mask = (word_t)(((word_t)(WORD(1) << (bitlen % WORD_BITS))) - 1);
out->val[out->wlen - 1] &= mask;
}
err:
nn_uninit(&tmp);
return ret;
}
int nn_xor(nn_t A, nn_src_t B, nn_src_t C)
{
int ret;
u8 i;
ret = nn_check_initialized(B); EG(ret, err);
ret = nn_check_initialized(C); EG(ret, err);
if ((A != B) && (A != C)) {
ret = nn_init(A, 0); EG(ret, err);
}
A->wlen = (C->wlen < B->wlen) ? B->wlen : C->wlen;
for (i = 0; i < A->wlen; i++) {
A->val[i] = (B->val[i] ^ C->val[i]);
}
err:
return ret;
}
int nn_or(nn_t A, nn_src_t B, nn_src_t C)
{
int ret;
u8 i;
ret = nn_check_initialized(B); EG(ret, err);
ret = nn_check_initialized(C); EG(ret, err);
if ((A != B) && (A != C)) {
ret = nn_init(A, 0); EG(ret, err);
}
A->wlen = (C->wlen < B->wlen) ? B->wlen : C->wlen;
for (i = 0; i < A->wlen; i++) {
A->val[i] = (B->val[i] | C->val[i]);
}
err:
return ret;
}
int nn_and(nn_t A, nn_src_t B, nn_src_t C)
{
int ret;
u8 i;
ret = nn_check_initialized(B); EG(ret, err);
ret = nn_check_initialized(C); EG(ret, err);
if ((A != B) && (A != C)) {
ret = nn_init(A, 0); EG(ret, err);
}
A->wlen = (C->wlen < B->wlen) ? B->wlen : C->wlen;
for (i = 0; i < A->wlen; i++) {
A->val[i] = (B->val[i] & C->val[i]);
}
err:
return ret;
}
int nn_not(nn_t A, nn_src_t B)
{
int ret;
u8 i;
ret = nn_check_initialized(B); EG(ret, err);
if (A != B) {
ret = nn_init(A, 0); EG(ret, err);
}
A->wlen = B->wlen;
for (i = 0; i < A->wlen; i++) {
A->val[i] = (word_t)(~(B->val[i]));
}
err:
return ret;
}
ATTRIBUTE_WARN_UNUSED_RET static u8 wclz(word_t A)
{
u8 cnt = WORD_BITS, over = 0;
int i;
for (i = (WORD_BITS - 1); i >= 0; i--) {
u8 mask = (u8)(((A & (WORD(1) << i)) >> i) & 0x1);
over |= mask;
cnt = (u8)(cnt - over);
}
return cnt;
}
int nn_clz(nn_src_t in, bitcnt_t *lz)
{
bitcnt_t cnt = 0;
int ret;
u8 i;
MUST_HAVE((lz != NULL), ret, err);
ret = nn_check_initialized(in); EG(ret, err);
for (i = in->wlen; i > 0; i--) {
if (in->val[i - 1] == 0) {
cnt = (bitcnt_t)(cnt + WORD_BITS);
} else {
cnt = (bitcnt_t)(cnt + wclz(in->val[i - 1]));
break;
}
}
*lz = cnt;
err:
return ret;
}
int nn_bitlen(nn_src_t in, bitcnt_t *blen)
{
bitcnt_t _blen = 0;
int ret;
u8 i;
MUST_HAVE((blen != NULL), ret, err);
ret = nn_check_initialized(in); EG(ret, err);
for (i = in->wlen; i > 0; i--) {
if (in->val[i - 1] != 0) {
_blen = (bitcnt_t)((i * WORD_BITS) - wclz(in->val[i - 1]));
break;
}
}
(*blen) = _blen;
err:
return ret;
}
int nn_getbit(nn_src_t in, bitcnt_t bit, u8 *bitval)
{
bitcnt_t widx = bit / WORD_BITS;
u8 bidx = bit % WORD_BITS;
int ret;
MUST_HAVE((bitval != NULL), ret, err);
ret = nn_check_initialized(in); EG(ret, err);
MUST_HAVE((bit < NN_MAX_BIT_LEN), ret, err);
(*bitval) = (u8)((((in->val[widx]) & (WORD(1) << bidx)) >> bidx) & 0x1);
err:
return ret;
}