📄 bn32.c
字号:
/*
* bn32.c - the high-level bignum interface
*
* Like bni32.c, this reserves the string "32" for textual replacement.
* The string must not appear anywhere unless it is intended to be replaced
* to generate other bignum interface functions.
*
* Written by Colin Plumb
*
* $Id: bn32.c,v 1.10 1998/10/22 19:44:52 heller Exp $
*/
#include "pgpConfig.h"
/*
* Some compilers complain about #if FOO if FOO isn't defined,
* so do the ANSI-mandated thing explicitly...
*/
#ifndef NO_STRING_H
#define NO_STRING_H 0
#endif
#ifndef HAVE_STRINGS_H
#define HAVE_STRINGS_H 0
#endif
#ifndef NEED_MEMORY_H
#define NEED_MEMORY_H 0
#endif
#if !NO_STRING_H
#include <string.h> /* for memmove() in bnMakeOdd */
#elif HAVE_STRINGS_H
#include <strings.h>
#endif
#if NEED_MEMORY_H
#include <memory.h>
#endif
/*
* This was useful during debugging, so it's left in here.
* You can ignore it. DBMALLOC is generally undefined.
*/
#ifndef DBMALLOC
#define DBAMLLOC 0
#endif
#if DBMALLOC
#include "../dbmalloc/malloc.h"
#define MALLOCDB malloc_chain_check(1)
#else
#define MALLOCDB (void)0
#endif
#include "bni.h"
#include "bni32.h"
#include "bnimem.h"
#include "bn32.h"
#include "bn.h"
/* Work-arounds for some particularly broken systems */
#include "bnkludge.h" /* For memmove() */
#include "pgpDebug.h"
/* Functions */
void
bnInit_32(void)
{
bnEnd = bnEnd_32;
bnPrealloc = bnPrealloc_32;
bnCopy = bnCopy_32;
bnNorm = bnNorm_32;
bnExtractBigBytes = bnExtractBigBytes_32;
bnInsertBigBytes = bnInsertBigBytes_32;
bnExtractLittleBytes = bnExtractLittleBytes_32;
bnInsertLittleBytes = bnInsertLittleBytes_32;
bnLSWord = bnLSWord_32;
bnBits = bnBits_32;
bnAdd = bnAdd_32;
bnSub = bnSub_32;
bnCmpQ = bnCmpQ_32;
bnSetQ = bnSetQ_32;
bnAddQ = bnAddQ_32;
bnSubQ = bnSubQ_32;
bnCmp = bnCmp_32;
bnSquare = bnSquare_32;
bnMul = bnMul_32;
bnMulQ = bnMulQ_32;
bnDivMod = bnDivMod_32;
bnMod = bnMod_32;
bnModQ = bnModQ_32;
bnExpMod = bnExpMod_32;
bnDoubleExpMod = bnDoubleExpMod_32;
bnTwoExpMod = bnTwoExpMod_32;
bnGcd = bnGcd_32;
bnInv = bnInv_32;
bnLShift = bnLShift_32;
bnRShift = bnRShift_32;
bnMakeOdd = bnMakeOdd_32;
}
void
bnEnd_32(BigNum *bn)
{
if (bn->ptr) {
BNIFREE((BNWORD32 *)bn->ptr, bn->allocated);
bn->ptr = 0;
}
bn->size = 0;
bn->allocated = 0;
MALLOCDB;
}
/* Internal function. It operates in words. */
static int
bnResize_32(BigNum *bn, unsigned len)
{
void *p;
/* Round size up: most mallocs impose 8-byte granularity anyway */
len = (len + (8/sizeof(BNWORD32) - 1)) & ~(8/sizeof(BNWORD32) - 1);
p = BNIREALLOC( bn->mgr, bn->isSecure,
(BNWORD32 *)bn->ptr, bn->allocated, len);
if (!p)
return -1;
bn->ptr = p;
bn->allocated = len;
MALLOCDB;
return 0;
}
#define bnSizeCheck(bn, size) \
if (bn->allocated < size && bnResize_32(bn, size) < 0) \
return -1
int
bnPrealloc_32(BigNum *bn, unsigned bits)
{
bits = (bits + 32-1)/32;
bnSizeCheck(bn, bits);
MALLOCDB;
return 0;
}
int
bnCopy_32(BigNum *dest, BigNum const *src)
{
bnSizeCheck(dest, src->size);
dest->size = src->size;
bniCopy_32((BNWORD32 *)dest->ptr, (BNWORD32 *)src->ptr, src->size);
MALLOCDB;
return 0;
}
void
bnNorm_32(BigNum *bn)
{
bn->size = bniNorm_32((BNWORD32 *)bn->ptr, bn->size);
}
/*
* Convert a bignum to big-endian bytes. Returns, in big-endian form, a
* substring of the bignum starting from lsbyte and "len" bytes long.
* Unused high-order (leading) bytes are filled with 0.
*/
void
bnExtractBigBytes_32(BigNum const *bn, unsigned char *dest,
unsigned lsbyte, unsigned len)
{
unsigned s = bn->size * (32 / 8);
/* Fill unused leading bytes with 0 */
while (s < lsbyte+len && len) {
*dest++ = 0;
len--;
}
if (len)
bniExtractBigBytes_32((BNWORD32 *)bn->ptr, dest, lsbyte, len);
MALLOCDB;
}
int
bnInsertBigBytes_32(BigNum *bn, unsigned char const *src,
unsigned lsbyte, unsigned len)
{
unsigned s = bn->size;
unsigned words = (len+lsbyte+sizeof(BNWORD32)-1) / sizeof(BNWORD32);
/* Pad with zeros as required */
bnSizeCheck(bn, words);
if (s < words) {
bniZero_32((BNWORD32 *)bn->ptr BIGLITTLE(-s,+s), words-s);
s = words;
}
bniInsertBigBytes_32((BNWORD32 *)bn->ptr, src, lsbyte, len);
bn->size = bniNorm_32((BNWORD32 *)bn->ptr, s);
MALLOCDB;
return 0;
}
/*
* Convert a bignum to little-endian bytes. Returns, in little-endian form, a
* substring of the bignum starting from lsbyte and "len" bytes long.
* Unused high-order (trailing) bytes are filled with 0.
*/
void
bnExtractLittleBytes_32(BigNum const *bn, unsigned char *dest,
unsigned lsbyte, unsigned len)
{
unsigned s = bn->size * (32 / 8);
/* Fill unused leading bytes with 0 */
while (s < lsbyte+len && len)
dest[--len] = 0;
if (len)
bniExtractLittleBytes_32((BNWORD32 *)bn->ptr, dest,
lsbyte, len);
MALLOCDB;
}
int
bnInsertLittleBytes_32(BigNum *bn, unsigned char const *src,
unsigned lsbyte, unsigned len)
{
unsigned s = bn->size;
unsigned words = (len+lsbyte+sizeof(BNWORD32)-1) / sizeof(BNWORD32);
/* Pad with zeros as required */
bnSizeCheck(bn, words);
if (s < words) {
bniZero_32((BNWORD32 *)bn->ptr BIGLITTLE(-s,+s), words-s);
s = words;
}
bniInsertLittleBytes_32((BNWORD32 *)bn->ptr, src, lsbyte, len);
bn->size = bniNorm_32((BNWORD32 *)bn->ptr, s);
MALLOCDB;
return 0;
}
/* Return the least-significant word of the input. */
unsigned
bnLSWord_32(BigNum const *src)
{
return src->size ? (unsigned)((BNWORD32 *)src->ptr)[BIGLITTLE(-1,0)]
: 0;
}
unsigned
bnBits_32(BigNum const *src)
{
return bniBits_32((BNWORD32 *)src->ptr, src->size);
}
int
bnAdd_32(BigNum *dest, BigNum const *src)
{
unsigned s = src->size, d = dest->size;
BNWORD32 t;
if (!s)
return 0;
bnSizeCheck(dest, s);
if (d < s) {
bniZero_32((BNWORD32 *)dest->ptr BIGLITTLE(-d,+d), s-d);
dest->size = d = s;
MALLOCDB;
}
t = bniAddN_32((BNWORD32 *)dest->ptr, (BNWORD32 *)src->ptr, s);
MALLOCDB;
if (t) {
if (d > s) {
t = bniAdd1_32((BNWORD32 *)dest->ptr BIGLITTLE(-s,+s),
d-s, t);
MALLOCDB;
}
if (t) {
bnSizeCheck(dest, d+1);
((BNWORD32 *)dest->ptr)[BIGLITTLE(-1-d,d)] = t;
dest->size = d+1;
}
}
return 0;
}
/*
* dest -= src.
* If dest goes negative, this produces the absolute value of
* the difference (the negative of the true value) and returns 1.
* Otherwise, it returls 0.
*/
int
bnSub_32(BigNum *dest, BigNum const *src)
{
unsigned s = src->size, d = dest->size;
BNWORD32 t;
if (d < s && d < (s = bniNorm_32((BNWORD32 *)src->ptr, s))) {
bnSizeCheck(dest, s);
bniZero_32((BNWORD32 *)dest->ptr BIGLITTLE(-d,+d), s-d);
dest->size = d = s;
MALLOCDB;
}
if (!s)
return 0;
t = bniSubN_32((BNWORD32 *)dest->ptr, (BNWORD32 *)src->ptr, s);
MALLOCDB;
if (t) {
if (d > s) {
t = bniSub1_32((BNWORD32 *)dest->ptr BIGLITTLE(-s,+s),
d-s, t);
MALLOCDB;
}
if (t) {
bniNeg_32((BNWORD32 *)dest->ptr, d);
dest->size = bniNorm_32((BNWORD32 *)dest->ptr,
dest->size);
MALLOCDB;
return 1;
}
}
dest->size = bniNorm_32((BNWORD32 *)dest->ptr, dest->size);
return 0;
}
/*
* Compare the BigNum to the given value, which must be < 65536.
* Returns -1. 0 or 1 if a<b, a == b or a>b.
* a <=> b --> bnCmpQ(a,b) <=> 0
*/
int
bnCmpQ_32(BigNum const *a, unsigned b)
{
unsigned t;
BNWORD32 v;
t = bniNorm_32((BNWORD32 *)a->ptr, a->size);
/* If a is more than one word long or zero, it's easy... */
if (t != 1)
return (t > 1) ? 1 : (b ? -1 : 0);
v = (unsigned)((BNWORD32 *)a->ptr)[BIGLITTLE(-1,0)];
return (v > b) ? 1 : ((v < b) ? -1 : 0);
}
int
bnSetQ_32(BigNum *dest, unsigned src)
{
if (src) {
bnSizeCheck(dest, 1);
((BNWORD32 *)dest->ptr)[BIGLITTLE(-1,0)] = (BNWORD32)src;
dest->size = 1;
} else {
dest->size = 0;
}
return 0;
}
int
bnAddQ_32(BigNum *dest, unsigned src)
{
BNWORD32 t;
if (!dest->size)
return bnSetQ(dest, src);
t = bniAdd1_32((BNWORD32 *)dest->ptr, dest->size, (BNWORD32)src);
MALLOCDB;
if (t) {
src = dest->size;
bnSizeCheck(dest, src+1);
((BNWORD32 *)dest->ptr)[BIGLITTLE(-1-src,src)] = t;
dest->size = src+1;
}
return 0;
}
/*
* Return value as for bnSub: 1 if subtract underflowed, in which
* case the return is the negative of the computed value.
*/
int
bnSubQ_32(BigNum *dest, unsigned src)
{
BNWORD32 t;
if (!dest->size)
return bnSetQ(dest, src) < 0 ? -1 : (src != 0);
t = bniSub1_32((BNWORD32 *)dest->ptr, dest->size, src);
MALLOCDB;
if (t) {
/* Underflow. <= 1 word, so do it simply. */
bniNeg_32((BNWORD32 *)dest->ptr, 1);
dest->size = 1;
return 1;
}
/* Try to normalize? Needing this is going to be pretty damn rare. */
/* dest->size = bniNorm_32((BNWORD32 *)dest->ptr, dest->size); */
return 0;
}
/*
* Compare two BigNums. Returns -1. 0 or 1 if a<b, a == b or a>b.
* a <=> b --> bnCmp(a,b) <=> 0
*/
int
bnCmp_32(BigNum const *a, BigNum const *b)
{
unsigned s, t;
s = bniNorm_32((BNWORD32 *)a->ptr, a->size);
t = bniNorm_32((BNWORD32 *)b->ptr, b->size);
if (s != t)
return s > t ? 1 : -1;
return bniCmp_32((BNWORD32 *)a->ptr, (BNWORD32 *)b->ptr, s);
}
int
bnSquare_32(BigNum *dest, BigNum const *src)
{
unsigned s;
BNWORD32 *srcbuf;
s = bniNorm_32((BNWORD32 *)src->ptr, src->size);
if (!s) {
dest->size = 0;
return 0;
}
bnSizeCheck(dest, 2*s);
if (src == dest) {
BNIALLOC( src->mgr, src->isSecure, srcbuf, BNWORD32, s);
if (!srcbuf)
return -1;
bniCopy_32(srcbuf, (BNWORD32 *)src->ptr, s);
bniSquare_32((BNWORD32 *)dest->ptr, (BNWORD32 *)srcbuf, s);
BNIFREE(srcbuf, s);
} else {
bniSquare_32((BNWORD32 *)dest->ptr, (BNWORD32 *)src->ptr, s);
}
dest->size = bniNorm_32((BNWORD32 *)dest->ptr, 2*s);
MALLOCDB;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -