123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197(*****************************************************************************)(* *)(* Copyright (c) 2020-2021 Danny Willems <be.danny.willems@gmail.com> *)(* *)(* Permission is hereby granted, free of charge, to any person obtaining a *)(* copy of this software and associated documentation files (the "Software"),*)(* to deal in the Software without restriction, including without limitation *)(* the rights to use, copy, modify, merge, publish, distribute, sublicense, *)(* and/or sell copies of the Software, and to permit persons to whom the *)(* Software is furnished to do so, subject to the following conditions: *)(* *)(* The above copyright notice and this permission notice shall be included *)(* in all copies or substantial portions of the Software. *)(* *)(* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR*)(* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, *)(* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL *)(* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER*)(* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING *)(* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER *)(* DEALINGS IN THE SOFTWARE. *)(* *)(*****************************************************************************)moduletypeRAW_BASE=sigvalorder:Z.tvalsize_in_bytes:intvalcheck_bytes:Bytes.t->boolvalis_zero:Bytes.t->boolvalis_one:Bytes.t->boolvalrandom:unit->Bytes.tvalzero:unit->Bytes.tvalone:unit->Bytes.tvaladd:Bytes.t->Bytes.t->Bytes.tvalmul:Bytes.t->Bytes.t->Bytes.tvalunsafe_inverse:Bytes.t->Bytes.tvaleq:Bytes.t->Bytes.t->boolvalnegate:Bytes.t->Bytes.tvalsquare:Bytes.t->Bytes.tvaldouble:Bytes.t->Bytes.tvalpow:Bytes.t->Bytes.t->Bytes.tendmoduleMake(Stubs:RAW_BASE):Ff_sig.BASE=structtypet=Bytes.texceptionNot_in_fieldofBytes.tletorder=Stubs.orderletsize_in_bytes=Stubs.size_in_bytesletpad_if_requirebs=(* Pad to 32 bytes. In anycase, copy the bytes to a new buffer *)ifBytes.lengthbs<size_in_bytesthen(letpadded_bytes=Bytes.makesize_in_bytes'\000'inBytes.blitbs0padded_bytes0(Bytes.lengthbs);padded_bytes)elseBytes.copybsletcheck_bytesbs=ifBytes.lengthbs=size_in_bytesthenStubs.check_bytesbselsefalseletof_bytes_optbs=letbs=pad_if_requirebsinifcheck_bytesbsthenSomebselseNoneletof_bytes_exn(g:Bytes.t):t=letg=pad_if_requireginifcheck_bytesgthengelseraise(Not_in_fieldg)letto_bytesg=gletis_zerog=assert(Bytes.lengthg=Stubs.size_in_bytes);Stubs.is_zerogletis_oneg=assert(Bytes.lengthg=Stubs.size_in_bytes);Stubs.is_onegletzero=letg=Stubs.zero()inassert(Bytes.lengthg=Stubs.size_in_bytes);gletone=letg=Stubs.one()inassert(Bytes.lengthg=Stubs.size_in_bytes);gletrandom?state()=ignorestate;letg=Stubs.random()inassert(Bytes.lengthg=Stubs.size_in_bytes);gletrecnon_null_random?state()=ignorestate;letr=random()inifis_zerorthennon_null_random()elserletaddxy=assert(Bytes.lengthx=Stubs.size_in_bytes);assert(Bytes.lengthy=Stubs.size_in_bytes);letres=Stubs.addxyinassert(Bytes.lengthres=Stubs.size_in_bytes);reslet(+)=addletmulxy=assert(Bytes.lengthx=Stubs.size_in_bytes);assert(Bytes.lengthy=Stubs.size_in_bytes);letres=Stubs.mulxyinassert(Bytes.lengthres=Stubs.size_in_bytes);reslet(*)=mulletinverse_exng=assert(Bytes.lengthg=Stubs.size_in_bytes);letres=Stubs.unsafe_inverseginassert(Bytes.lengthres=Stubs.size_in_bytes);resletinverse_optg=assert(Bytes.lengthg=Stubs.size_in_bytes);ifis_zerogthenNoneelseletres=Stubs.unsafe_inverseginassert(Bytes.lengthres=Stubs.size_in_bytes);Someresletnegateg=assert(Bytes.lengthg=Stubs.size_in_bytes);letres=Stubs.negateginassert(Bytes.lengthres=Stubs.size_in_bytes);reslet(-)=negateletsubab=adda(negateb)letsquareg=assert(Bytes.lengthg=Stubs.size_in_bytes);letres=Stubs.squareginassert(Bytes.lengthres=Stubs.size_in_bytes);resletdoubleg=assert(Bytes.lengthg=Stubs.size_in_bytes);letres=Stubs.doubleginassert(Bytes.lengthres=Stubs.size_in_bytes);resleteqxy=Stubs.eqxylet(=)=eqletpowxn=letn=Z.eremn(Z.predStubs.order)in(* sign is removed by to_bits, but that's fine because we used mod before *)letn=Bytes.of_string(Z.to_bitsn)inletbytes_size_n=Bytes.lengthninletpadded_n=Bytes.initStubs.size_in_bytes(funi->ifi<bytes_size_nthenBytes.getnielsechar_of_int0)inletres=Stubs.pow(to_bytesx)padded_nin(* assert (Bytes.length res = Stubs.size_in_bytes) ; *)reslet(**)=powletdiv_exnab=ifb=zerothenraiseDivision_by_zeroelsemula(inverse_exnb)letdiv_optab=ifb=zerothenNoneelseSome(mula(inverse_exnb))let(/)=div_exnend