2021SC@SDUSC 山东大学软件学院软件工程应用与实践

1、mrflsh3.c结构

mrflsh3.c的总体结构如下,,主要实现了facos()、fasin()、fatan()、fcos()、fsin()、ftan()、norm()、tan1()  几个在miracl开源库中比较重要的函数,这一次的博客就是读一下这函数的功能。

2、源代码


static int norm(_MIPD_ int type,flash y)
{ /* convert y to first quadrant angle ** and return sign of result         */int s;
#ifdef MR_OS_THREADSmiracl *mr_mip=get_mip();
#endifif (mr_mip->ERNUM) return 0;s=PLUS;if (size(y)<0){negify(y,y);if (type!=COS) s=(-s);}fpi(_MIPP_ mr_mip->pi);fpmul(_MIPP_ mr_mip->pi,1,2,mr_mip->w8);if (fcomp(_MIPP_ y,mr_mip->w8)<=0) return s;fpmul(_MIPP_ mr_mip->pi,2,1,mr_mip->w8);if (fcomp(_MIPP_ y,mr_mip->w8)>0){ /* reduce mod 2.pi */fdiv(_MIPP_ y,mr_mip->w8,mr_mip->w9);ftrunc(_MIPP_ mr_mip->w9,mr_mip->w9,mr_mip->w9);fmul(_MIPP_ mr_mip->w9,mr_mip->w8,mr_mip->w9);fsub(_MIPP_ y,mr_mip->w9,y);}if (fcomp(_MIPP_ y,mr_mip->pi)>0){ /* if greater than pi */fsub(_MIPP_ y,mr_mip->pi,y);if (type!=TAN) s=(-s);}fpmul(_MIPP_ mr_mip->pi,1,2,mr_mip->w8);if (fcomp(_MIPP_ y,mr_mip->w8)>0){ /* if greater than pi/2 */fsub(_MIPP_ mr_mip->pi,y,y);if (type!=SIN) s=(-s);}return s;
}

norm()方法地主要作用就是将输入的角度y转换成第一象限的角度,并且返回结果的符号。首先观察y是否小于0,是的话调用negify方法获得y的相反数。再调用fpi方法将圆周率p.赋值给mr_mip->pi,调用fpmul方法使得mr_mip->pi*1/2赋值给mr_mip->w8,此时调用fcomp方法看y是否小于等于mr_mip->w8,是的话返回1。再次调用fpmul方法使得mr_mip->pi*2/1赋值给mr_mip->w8。调用调用fcomp方法看y是否大于mr_mip->w8,是的话调用采用下述方法去除输入角度大于360度的部分: fdiv方法使得y/mr_mip->w8赋值给mr_mip->w9,再调用ftrunc方法将mr_mip->w9分解为big和flash数据类型并且将flash部分赋值给mr_mip->w9,调用fmul方法将mr_mip->w9乘以mr_mip->w8赋值给mr_mip->w9,调用fsub方法将 y减去mr_mip->w9赋值给y。调用fcomp方法看此时的y是否大于mr_mip->pi(180度、圆周率),是的话调用fsub方法将y减去mr_mip->pi并且赋值给y。调用fpmul方法将mr_mip->pi*1/2赋值给mr_mip->w8,调用fcomp方法看此时的y是否大于mr_mip->w8(90度、圆周率/2),是的话调用fsub方法将y减去mr_mip->w8并且赋值给y。

static int tan1(_MIPD_ big w,int n)
{  /* generator for C.F. of tan(1) */ if (n==0) return 1;if (n%2==1) return 2*(n/2)+1;else return 1;
}

tan1()方法的主要作用就是根据n的输入值获得不同的输出值。如果n等于0的话返回1,n除以2的余数为1的话返回2*(n/23)+1,否则返回1.

void ftan(_MIPD_ flash x,flash y)
{ /* calculates y=tan(x) */int i,n,nsq,m,sqrn,sgn,op[5];
#ifdef MR_OS_THREADSmiracl *mr_mip=get_mip();
#endifcopy(x,y);if (mr_mip->ERNUM || size(y)==0) return;MR_IN(57)sgn=norm(_MIPP_ TAN,y);ftrunc(_MIPP_ y,y,mr_mip->w10);n=size(y);if (n!=0) build(_MIPP_ y,tan1);if (size(mr_mip->w10)==0){insign(sgn,y);MR_OUTreturn;}sqrn=isqrt(mr_mip->lg2b*mr_mip->workprec,mr_mip->lg2b);nsq=0;copy(mr_mip->w10,mr_mip->w8);frecip(_MIPP_ mr_mip->w10,mr_mip->w10);ftrunc(_MIPP_ mr_mip->w10,mr_mip->w10,mr_mip->w10);m=logb2(_MIPP_ mr_mip->w10);if (m<sqrn){ /* scale fraction down */nsq=sqrn-m;expb2(_MIPP_ nsq,mr_mip->w10);fdiv(_MIPP_ mr_mip->w8,mr_mip->w10,mr_mip->w8);}zero(mr_mip->w10);fmul(_MIPP_ mr_mip->w8,mr_mip->w8,mr_mip->w9);negify(mr_mip->w9,mr_mip->w9);op[0]=0x4B;    /* set up for x/(y+C) */op[1]=op[3]=1;op[2]=0;for (m=sqrn;m>1;m--){ /* Unwind C.F for tan(x)=z/(1-z^2/(3-z^2/(5-...)))))) */op[4]=2*m-1;flop(_MIPP_ mr_mip->w9,mr_mip->w10,op,mr_mip->w10);}op[4]=1;flop(_MIPP_ mr_mip->w8,mr_mip->w10,op,mr_mip->w10);op[0]=0x6C;     /* set up for tan(x+y)=tan(x)+tan(y)/(1-tan(x).tan(y)) */op[1]=op[2]=op[3]=1;op[4]=(-1);for (i=0;i<nsq;i++){ /* now square it back up again */flop(_MIPP_ mr_mip->w10,mr_mip->w10,op,mr_mip->w10);}flop(_MIPP_ y,mr_mip->w10,op,y);insign(sgn,y);MR_OUT
}

ftan()方法的主要作用就是获得输入角度x的an值并且赋值给y。首先调用norm方法获得与x相对应的0~90度的角度和符号sgn,再计算出tan(x)的值y后再调用insign方法把符号sgn重新赋值给y。

void fatan(_MIPD_ flash x,flash y)
{ /* calculate y=atan(x) */int s,c,op[5];BOOL inv,hack;
#ifdef MR_OS_THREADSmiracl *mr_mip=get_mip();
#endifcopy(x,y);if (mr_mip->ERNUM || size(y)==0) return;MR_IN(58)s=exsign(y);insign(PLUS,y);inv=FALSE;fpi(_MIPP_ mr_mip->pi);fconv(_MIPP_ 1,1,mr_mip->w11);c=fcomp(_MIPP_ y,mr_mip->w11);if (c==0){ /* atan(1)=pi/4 */fpmul(_MIPP_ mr_mip->pi,1,4,y);insign(s,y);MR_OUTreturn;}if (c>0){ /* atan(x)=pi/2 - atan(1/x) for x>1 */inv=TRUE;frecip(_MIPP_ y,y);}hack=FALSE;if (mr_lent(y)<=2){ /* for 'simple' values of y */hack=TRUE;fconv(_MIPP_ 3,1,mr_mip->w11);froot(_MIPP_ mr_mip->w11,2,mr_mip->w11);op[0]=0xC6;op[2]=op[3]=op[4]=1;op[1]=(-1);flop(_MIPP_ y,mr_mip->w11,op,y);}op[0]=0x4B;op[1]=op[3]=op[4]=1;op[2]=0;mr_mip->workprec=mr_mip->stprec;dconv(_MIPP_ atan(fdsize(_MIPP_ y)),mr_mip->w11);while (mr_mip->workprec!=mr_mip->nib){ /* Newtons iteration w11=w11+(y-tan(w11))/(tan(w11)^2+1) */if (mr_mip->workprec<mr_mip->nib) mr_mip->workprec*=2;if (mr_mip->workprec>=mr_mip->nib) mr_mip->workprec=mr_mip->nib;else if (mr_mip->workprec*2>mr_mip->nib) mr_mip->workprec=(mr_mip->nib+1)/2;ftan(_MIPP_ mr_mip->w11,mr_mip->w12);fsub(_MIPP_ y,mr_mip->w12,mr_mip->w8);fmul(_MIPP_ mr_mip->w12,mr_mip->w12,mr_mip->w12);flop(_MIPP_ mr_mip->w8,mr_mip->w12,op,mr_mip->w12);  /* w12=w8/(w12+1) */fadd(_MIPP_ mr_mip->w12,mr_mip->w11,mr_mip->w11);}copy(mr_mip->w11,y);op[0]=0x6C;op[1]=op[3]=6;op[2]=1;op[4]=0;if (hack) flop(_MIPP_ y,mr_mip->pi,op,y);op[1]=(-2);op[3]=2;if (inv) flop(_MIPP_ y,mr_mip->pi,op,y);insign(s,y);MR_OUT
}

fatan()方法的主要作用就是获得输入角度x的arctan值并且赋值给y。首先调用exsign方法获得与x的符号s,再计算出arctan(x)的值y后再调用insign方法把符号s重新赋值给y。

void fsin(_MIPD_ flash x,flash y)
{ /*  calculate y=sin(x) */int sgn,op[5];
#ifdef MR_OS_THREADSmiracl *mr_mip=get_mip();
#endifcopy(x,y);if (mr_mip->ERNUM || size(y)==0) return;MR_IN(59)sgn=norm(_MIPP_ SIN,y);fpmul(_MIPP_ y,1,2,y);ftan(_MIPP_ y,y);op[0]=0x6C;op[1]=op[2]=op[3]=op[4]=1;flop(_MIPP_ y,y,op,y);insign(sgn,y);MR_OUT
}

fsin()方法的主要作用就是获得输入角度x的sin值并且赋值给y。首先调用copy方法把x复制给y,再调用norm方法获得与x相对应的0~90度的角度和符号sgn,再调用fpmul使y*1/2赋值给y,再调用ftan方法获得y的tan(y)值赋值给y,再调用flop方法根据数组op的值选择不同的函数,通过tan值求出sin值并且赋值给y,最后调用insign方法把符号sgn赋值给y。

void fasin(_MIPD_ flash x,flash y)
{ /* calculate y=asin(x) */int s,op[5];
#ifdef MR_OS_THREADSmiracl *mr_mip=get_mip();
#endifcopy(x,y);if (mr_mip->ERNUM || size(y)==0) return;MR_IN(60)s=exsign(y);insign(PLUS,y);op[0]=0x3C;op[1]=(-1);op[2]=op[3]=1;op[4]=0;flop(_MIPP_ y,y,op,mr_mip->w11);  /* mr_w11 = -(y.y-1) */froot(_MIPP_ mr_mip->w11,2,mr_mip->w11);if (size(mr_mip->w11)==0){fpi(_MIPP_ mr_mip->pi);fpmul(_MIPP_ mr_mip->pi,1,2,y);}else{fdiv(_MIPP_ y,mr_mip->w11,y);fatan(_MIPP_ y,y);}insign(s,y);    MR_OUT
}

fasin()方法的主要作用就是获得输入角度x的arcsin值并且赋值给y。首先调用exsign方法获得与x的符号s,再调用insign方法获得x的相反数,然后调用flop方法根据数组op的值选择不同的函数计算出y*(y-1)赋值给mr_mip->w11,再调用froot方法获得mr_mip->w11的平方根赋值给mr_mip->w11。如果mr_mip->w11等于0的话,调用fpi方法将圆周率pi赋值给mr_mip->pi,再调用 fpmul方法将mr_mip->pi*1/2赋值给y;否则的话调用fdiv方法将y除以mr_mip->w11赋值给y,再调用ftan方法获得y的tan值赋值给y。最后调用insign方法将符号s赋值给y。

void fcos(_MIPD_ flash x,flash y)
{ /* calculate y=cos(x) */int sgn,op[5];
#ifdef MR_OS_THREADSmiracl *mr_mip=get_mip();
#endifcopy(x,y);if (mr_mip->ERNUM || size(y)==0){convert(_MIPP_ 1,y);return;}MR_IN(61)sgn=norm(_MIPP_ COS,y);fpmul(_MIPP_ y,1,2,y);ftan(_MIPP_ y,y);op[0]=0x33;op[1]=op[3]=op[4]=1;op[2]=(-1);flop(_MIPP_ y,y,op,y);insign(sgn,y);MR_OUT
}

fcos()方法的主要作用就是获得输入角度x的cos值并且赋值给y。首先调用copy方法把x复制给y,再调用norm方法获得与x相对应的0~90度的角度和符号sgn,再调用fpmul使y*1/2赋值给y,再调用ftan方法获得y的tan(y)值赋值给y,再调用flop方法根据数组op的值选择不同的函数,通过tan值求出cos值并且赋值给y,最后调用insign方法把符号sgn赋值给y。

void facos(_MIPD_ flash x,flash y)
{ /* calculate y=acos(x) */int op[5];
#ifdef MR_OS_THREADSmiracl *mr_mip=get_mip();
#endifif (mr_mip->ERNUM) return;MR_IN(62)fasin(_MIPP_ x,y);fpi(_MIPP_ mr_mip->pi);op[0]=0x6C;op[1]=(-2);op[2]=1;op[3]=2;op[4]=0;flop(_MIPP_ y,mr_mip->pi,op,y);    /* y = pi/2 - y */MR_OUT
}

facos()方法的主要作用就是获得输入角度x的arccos值并且赋值给y。首先调用fasin方法获得x的arcsin值,再调用fpi方法将圆周率pi赋值给mr_mip->pi,然后调用flop方法根据数组op的值选择不同的函数计算出pi*(2-y)赋值给y。

Miracle密码算法开源库(十二)分析 :mrflsh3.c相关推荐

  1. Miracle密码算法开源库(一)源码分析 :mraes.c

    2021SC@SDUSC 山东大学软件学院软件工程应用与实践 一.mraes.c结构 mraec.c的总体结构如下,具有fbsub.ftable1等数组,主要实现了aes_decrypt .aes_e ...

  2. NTL密码算法开源库——模二整数上的矩阵(mat_GF2)

    2021SC@SDUSC 模二整数上的矩阵(mat_GF2) 矩阵运算 高斯消元 矩阵运算 具体代码 #include <NTL/matrix.h> #include <NTL/ve ...

  3. NTL密码算法开源库(数论库)代码分析项目--综述

    2021SC@SDUSC NTL密码算法开源库(数论库)代码分析项目--综述 项目综述 NTL开源代码库的安装和使用 NTL代码开源库主要解决的问题 项目分工以及核心代码的分配 项目综述 NTL算法库 ...

  4. NTL密码算法开源库--综述

    2021SC@SDUSC NTL密码算法开源库--综述 项目综述 NTL算法库是开放源码的自由软件,具有专业处理任意精度大整数.实数的计算数论与计算代数的高性能可移植c++库,提供了任意大整数.任意精 ...

  5. NTL密码算法开源库-大整数ZZ类(一)

    2021SC@SDUSC NTL密码算法开源库-大整数ZZ类(一) 本章综述 代码分析 贝祖公式 本章综述 大整数ZZ类主要实现了任意长度大整数表示.最大公因数.Jacobi符号和素性检验.笔者将通过 ...

  6. NTL密码算法开源库-大整数ZZ类(三)

    2021SC@SDUSC NTL密码算法开源库-大整数ZZ类(三) 中国剩余定理 一次同余式 乘法逆元 简化剩余的等价描述 二次同余式 雅可比符号 中国剩余定理 //中国剩余定理模板代码 typede ...

  7. NTL密码算法开源库——大整数ZZ类(二)

    2021SC@SDUSC 素数检测 (1)数学基础:费马小定理,二次探测定理,欧拉定理,Miller-Rabin素数测试,同余式, wilson定理,乘法逆元,简化剩余系 费马小定理:若存在整数 a ...

  8. SEAL全同态加密开源库(十二) CKKS-源码浅析

    2021SC@SDUSC 2021-12-19 前言 前两篇我们讨论了SEAL对于BFV的实现.本篇开始我将开始讨论SEAL对于CKKS的实现. 照例我会先补充理论知识,然后分析源码native/ex ...

  9. NTL密码算法开源库——大整数ZZ类(四)

    2021SC@SDUSC RSA算法原理 密钥生成的步骤 第一步,随机选择两个不相等的质数p和q. 爱丽丝选择了61和53.(实际应用中,这两个质数越大,就越难破解.) 第二步,计算p和q的乘积n. ...

  10. 【NTL密码算法开源库-概述】

    NTL库基本数据类型 ZZ: big integers(支持不等式运算) ZZ_p: big integers modulo p zz_p: integers mod "single pre ...

最新文章

  1. 面试数百名NLP工程师发现:90%以上是不合格的
  2. libevent java_libevent简介
  3. 台湾国立大学郭彦甫Matlab教程笔记(8)文件读写
  4. Java时间处理类SimpleDateFormat的parse和format方法的正确使用
  5. Linux系统 iptables 和 firewalld 的那些事
  6. android webview framework,android – Webview导致ANR
  7. c# 写文件注意问题及用例展示
  8. android中自定义RadioButton
  9. SpecFlow的力量
  10. 什么?print 函数还可以打印彩色围棋局面?
  11. delphi打包python_python for delphi 组件安装和调试的那些坑儿 !
  12. 计算机病毒与反病毒技术
  13. 电气专业标准规范大全html,电气专业规范大全
  14. 上传本地文件到服务器:not a regular file
  15. windows 命令行操作
  16. Meta元宇宙副总裁离职了...『Go语言圣经』终于汉化啦;德云社失业警告!AI要说相声了;一键就能AI绘图的网站;前沿论文 | ShowMeAI资讯日报
  17. dtl文件java_数据库之DTL——数据事务语言 事务
  18. 武林传奇之七剑下天山java游戏开发商_武林传奇2之七剑下天山的配方
  19. csv文件中文乱码转换
  20. 微信卡券怎么制作以及卡券封号推送消息技术分享

热门文章

  1. 数独-比回溯法更优的人类思维逻辑的数独解法
  2. 深蓝-视觉slam-第七节习题
  3. Vue脚手架安装流程详解
  4. vuecli-脚手架,安装使用及目录详解
  5. win10如何调整计算机时间同步,Win10如何修改时间同步服务器?Windows时间同步出错解决方法...
  6. 爬取武汉大学教务系统数据
  7. java 线程的插队运行_java笔记–线程的插队行为
  8. 【树莓派不吃灰】基础篇⑲ 搭建usb摄像头MJPG-streamer图片流监控,支持远程视频监控访问
  9. python 爬虫。爬取小说--斗破苍穹
  10. 《单片机原理及应用(魏洪磊)》第六章第10题