from ctypes import * from base64 import * DECAF = CDLL("build/lib/libdecaf.so") F = GF(2^448-2^224-1) d = -39081 E = EllipticCurve(F,[0,2-4*d,0,1,0]) p_tor4 = E.lift_x(-1) Tor = [p_tor4 * i for i in xrange(4)] q = 2^446-0x8335dc163bb124b65129c96fde933d8d723a70aadc873d6d54a7bb0d FQ = GF(q) isoMagic = 1/sqrt(F(39082/39081)-1) passing = True # TODO: pathological cases # TODO: Elligator # TODO: double scalar mul # TODO: Curve25519 def random_array(length): answer = "".join([chr(randint(0,255)) for i in xrange(length)]) return answer def from_le(buf): return sum([256^i * ord(x) for i,x in enumerate(buf)]) def youfail(why,n): print ("Fail on test %d!"%n), why global passing passing = False def run_test(i): try: s = DecafScalar.random() t = DecafScalar.random() p = DecafPoint.random() q = DecafPoint.random() s*p + t*q if s*(t*p) != (s*t)*p: raise Exception("Mul doesn't work") (p+q-p-q).ser() # i guess... except Exception, e: youfail(e,i) def run_all_tests(n = 100): for testno in xrange(n): run_test(testno) if passing: print "Passed all %d tests." % n def to_le(x,n): x = int(x) if x >= 256^n: raise Exception("Integer too big in to_le(%d,%d)" % (x,n)) return "".join([chr(x>>(8*i) & 255) for i in xrange(n)]) class DecafScalar(): _UNDER = c_uint64 * int(7) def __init__(self,cstruct=None,scalar=None): if cstruct is None: cstruct = DecafScalar._UNDER() memmove(addressof(cstruct), DECAF.decaf_448_scalar_zero, 8*7 ) if scalar is None: scalar = E(0) self.cstruct = cstruct self.scalar = scalar self._check() @staticmethod def _c_deser(str): buffer = (c_uint8*int(56)).from_buffer_copy(str) cstruct = DecafScalar._UNDER() ret = DECAF.decaf_448_scalar_decode(cstruct,buffer,c_uint64(-1)) if ret != -1: raise Exception("scalar didn't decode") return cstruct @staticmethod def _sage_deser(str): s = from_le(str) if s >= FQ.cardinality(): raise Exception("scalar didn't decode") return FQ(s) def __eq__(self,other): csays = bool(DECAF.decaf_448_scalar_eq(self.cstruct,other.cstruct)) sagesays = any([self.scalar == other.scalar + t for t in Tor]) if csays != sagesays: raise Exception("C and SAGE don't agree: %d %d" % (csays, sagesays)) return csays def __ne__(self,other): return not self==other def __add__(self,other): cstruct = DecafScalar._UNDER() DECAF.decaf_448_scalar_add(cstruct,self.cstruct,other.cstruct) return DecafScalar(cstruct,self.scalar + other.scalar) def __sub__(self,other): cstruct = DecafScalar._UNDER() DECAF.decaf_448_scalar_sub(cstruct,self.cstruct,other.cstruct) return DecafScalar(cstruct,self.scalar - other.scalar) def __mul__(self,other): if isinstance(other,DecafScalar): cstruct = DecafScalar._UNDER() DECAF.decaf_448_scalar_mul(cstruct,self.cstruct,other.cstruct) return DecafScalar(cstruct,self.scalar * other.scalar) elif isinstance(other,DecafPoint): cstruct = DecafPoint._UNDER() DECAF.decaf_448_point_scalarmul(cstruct,other.cstruct,self.cstruct) return DecafPoint(cstruct,int(self.scalar) * other.point) else: raise Exception("Nope") def __div__(self,other): return self / other.inverse() def inverse(self): cstruct = DecafScalar._UNDER() z = DECAF.decaf_448_scalar_invert(cstruct,self.cstruct) if bool(z) != (self.scalar == 0): raise Exception("C and SAGE don't agree") return DecafScalar(cstruct,1/self.scalar) def __neg__(self): cstruct = DecafScalar._UNDER() DECAF.decaf_448_scalar_negate(cstruct,self.cstruct) return DecafScalar(cstruct,-self.scalar) def __str__(self): return " ".join(["%02x"%ord(b) for b in self.ser()]) def __repr__(self): return "DecafScalar.fromInt(%d)" % self.scalar @classmethod def fromInt(cls,i): return cls.deser(to_le(i,56)) def to64(self): return b64encode(self.ser()) @classmethod def from64(cls,str): return cls.deser(b64decode(str)) @classmethod def deser(cls,str): good = True try: cstruct = cls._c_deser(str) except Exception: good = False good2 = True try: scalar = cls._sage_deser(str) except Exception: good2 = False if good != good2: raise Exception("C and SAGE don't agree") elif not good: raise Exception("scalar didn't decode") return cls(cstruct,scalar) @classmethod def random(cls): while True: try: return cls.deser(random_array(56)) except Exception: pass @staticmethod def _c_ser(cstruct): buffer = (c_uint8*int(56))() DECAF.decaf_448_scalar_encode(buffer,cstruct) return str(bytearray(buffer)) def ser(self): return self._c_ser(self.cstruct) @staticmethod def _sage_ser(P): return to_le(P,56) def _check(self): ss = self._sage_ser(self.scalar) cs = self._c_ser(self.cstruct) if ss != cs: print ss print cs raise Exception("Check failed!") return True class DecafPoint(): @staticmethod def _UNDER(): size = int(8*8*4) alignment = 32 buf1 = bytearray(size+alignment-1) buf2 = (c_char * int(size+alignment-1)).from_buffer(buf1) raw_addr = addressof(buf2) offset = (-raw_addr) % alignment return (c_char*size).from_buffer(buf2,int(offset)) def __init__(self,cstruct=None,point=None): if cstruct is None: cstruct = DecafPoint._UNDER() memmove(addressof(cstruct), DECAF.decaf_448_point_identity, 8*8*4 ) if point is None: point = E(0) self.cstruct = cstruct self.point = point self._check() @staticmethod def _c_deser(str): buffer = (c_uint8*int(56)).from_buffer_copy(str) cstruct = DecafPoint._UNDER() ret = DECAF.decaf_448_point_decode(cstruct,buffer,c_uint64(-1)) if ret != -1: raise Exception("Point didn't decode") return cstruct @staticmethod def _sage_deser(str): s = from_le(str) if is_odd(s): raise Exception("Point didn't decode") if (s==0): return E(0) if not E.is_x_coord(s^2): raise Exception("Point didn't decode") P = E.lift_x(s^2) if is_odd(int(2*s^2*isoMagic/P.xy()[1])): P = -P return P def __eq__(self,other): csays = bool(DECAF.decaf_448_point_eq(self.cstruct,other.cstruct)) sagesays = any([self.point == other.point + t for t in Tor]) if csays != sagesays: raise Exception("C and SAGE don't agree: %d %d" % (csays, sagesays)) return csays def __ne__(self,other): return not self==other def __add__(self,other): cstruct = DecafPoint._UNDER() DECAF.decaf_448_point_add(cstruct,self.cstruct,other.cstruct) return DecafPoint(cstruct,self.point + other.point) def __sub__(self,other): cstruct = DecafPoint._UNDER() DECAF.decaf_448_point_sub(cstruct,self.cstruct,other.cstruct) return DecafPoint(cstruct,self.point - other.point) def __mul__(self,other): if isinstance(other,DecafScalar): return other*self else: raise Exception("nope") def __div__(self,other): if isinstance(other,DecafScalar): return other.inverse()*self else: raise Exception("nope") def __neg__(self): cstruct = DecafPoint._UNDER() DECAF.decaf_448_point_negate(cstruct,self.cstruct) return DecafPoint(cstruct,-self.point) def __str__(self): return " ".join(["%02x"%ord(b) for b in self.ser()]) def __repr__(self): return "DecafPoint.from64('%s')" % self.to64() def to64(self): return b64encode(self.ser()) @classmethod def from64(cls,str): return cls.deser(b64decode(str)) @classmethod def deser(cls,str): good = True try: cstruct = cls._c_deser(str) except Exception: good = False good2 = True try: point = cls._sage_deser(str) except Exception: good2 = False if good != good2: raise Exception("C and SAGE don't agree") elif not good: raise Exception("Point didn't decode") return cls(cstruct,point) @classmethod def random(cls): while True: try: return cls.deser(random_array(56)) except Exception: pass @staticmethod def _c_ser(cstruct): buffer = (c_uint8*int(56))() DECAF.decaf_448_point_encode(buffer,cstruct) return str(bytearray(buffer)) def ser(self): return self._c_ser(self.cstruct) @staticmethod def _sage_ser(P): if P == E(0): return to_le(0,56) x,y = P.xy() s = sqrt(x) if s==0: return to_le(0,56) if is_odd(int(2*s^2*isoMagic/y)): s = 1/s if is_odd(int(s)): s = -s return to_le(s,56) def _check(self): ss = self._sage_ser(self.point) cs = self._c_ser(self.cstruct) if ss != cs: print "SAGE",b64encode(ss) print "C ",b64encode(cs) raise Exception("Check failed!") return True run_all_tests()