forked from prophetss/rsa
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrsa.c
193 lines (150 loc) · 4.46 KB
/
rsa.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <time.h>
#include <gmp.h>
#define MIN_MODLEN 512
#define MAX_MODLEN 16384
/*结尾终止符,变量初始释放使用*/
#define NULL_TERMINATED 0
/*key_size-密钥长度, kn、ke公钥,kn、kd私钥*/
int generatekeypair(const unsigned long int key_size, char **kn, char **ke, char **kd)
{
if (key_size < MIN_MODLEN || key_size > MAX_MODLEN) {
return -1;
}
mpz_t prand, qrand, p, q, p1, q1, phi, n, d, e, gcd;
mpz_inits(prand, qrand, p, q, p1, q1, phi, n, d, e, gcd, NULL_TERMINATED);
/*固定设置e为65537*/
mpz_set_ui(e, 65537);
gmp_randstate_t state;
gmp_randinit_default(state);
/*设置随机数种子*/
clock_t tm = time(NULL);
gmp_randseed_ui(state, tm);
mp_bitcnt_t lp = (key_size + 1) >> 1;
mp_bitcnt_t lq = key_size - lp;
/*2的lp-1次方 < prand < 2的lp次方-1*/
mpz_rrandomb(prand, state, lp);
while (1) {
mpz_nextprime(p, prand);
do {
/*
*如果位数不够对较小值q再随机生成,如果这里能循环
*10次可以去买彩票了^-^
**/
gmp_randseed_ui(state, ++tm);
mpz_rrandomb(qrand, state, lq);
mpz_nextprime(q, qrand);
/*p > q*/
if (mpz_cmp(p, q) < 0) {
mpz_swap(p, q);
}
mpz_mul(n, p, q);
/*由上面随机数范围可以得出n可能为key_size-1位*/
} while (mpz_sizeinbase(n, 2) < key_size);
/*phi = (p-1)*(q-1)*/
mpz_sub_ui(p1, p, 1);
mpz_sub_ui(q1, q, 1);
mpz_mul(phi, p1, q1);
/*求最大公约数判断是否互质*/
mpz_gcd(gcd, phi, e);
if (0 != mpz_cmp_ui(gcd, 1)) {
continue;
}
/*求模反得出私钥d*/
mpz_invert(d, e, phi);
/*转换62进制对应字符串*/
*kn = mpz_get_str(NULL, 62, n);
*ke = mpz_get_str(NULL, 62, e);
*kd = mpz_get_str(NULL, 62, d);
break;
}
/*释放所有资源*/
mpz_clears(prand, p, q, p1, q1, phi, n, d, e, gcd, NULL_TERMINATED);
return 0;
}
/*
*msg-数据,其大小不可超过n大小,例如1024位密钥如果数据为字符串则其长度最大为128(1024/sizeof(char))
*kn、ke-密钥 cryptogram-输出密文
**/
int cipher(const char *msg, const unsigned int msg_len, const char *kn, const char *ke, char **cryptogram)
{
char *km = (char*)malloc(2 * msg_len + 1);
if (NULL == km) {
return -1;
}
/*将char拆成两个16进制表示*/
char hex[] = "0123456789ABCDEF";
unsigned int i, j;
for (i = 0, j = 0; j < msg_len; i += 2, j++) {
km[i] = hex[msg[j] / 16]; //高位
km[i + 1] = hex[msg[j] % 16];
}
km[2 * msg_len] = '\0';
mpz_t m, n, e, res;
mpz_inits(m, n, e, res, NULL_TERMINATED);
/*对应进制字符串转换*/
if (0 != mpz_set_str(n, kn, 62) || 0 != mpz_set_str(e, ke, 62) || 0 != mpz_set_str(m, km, 16)) {
mpz_clears(m, n, e, res, NULL_TERMINATED);
free(km);
return -1;
}
/*计算密文*/
mpz_powm(res, m, e, n);
/*转换为62进制对应字符串*/
*cryptogram = mpz_get_str(NULL, 62, res);
mpz_clears(m, n, e, res, NULL_TERMINATED);
free(km);
return 0;
}
/*单字符转16进制数*/
#define CHAR_TO_INT(c) (c >= 'a' ? c - 'a' + 10 : (c >= 'A' ? c - 'A' + 10 : c - '0'))
/*高低两字符转16进制数*/
#define CHARS_TO_INT(h, l) (CHAR_TO_INT(h)*16 + CHAR_TO_INT(l))
/*cryptogram-密文,kn、kd-密钥,msg-输出*/
int decipher(const char *cryptogram, const char *kn, const char *kd, char **msg)
{
mpz_t m, n, d, out;
mpz_inits(m, n, d, out, NULL_TERMINATED);
/*对应进制字符串转换*/
if (0 != mpz_set_str(n, kn, 62) || 0 != mpz_set_str(d, kd, 62) || 0 != mpz_set_str(m, cryptogram, 62)) {
mpz_clears(m, n, d, out, NULL_TERMINATED);
return -1;
}
/*解密*/
mpz_powm_sec(out, m, d, n);
char *message = mpz_get_str(NULL, 16, out);
unsigned int msg_len = strlen(message);
if (msg_len % 2 != 0) {
mpz_clears(m, n, d, out, NULL_TERMINATED);
return -1;
}
*msg = (char*)malloc(msg_len / 2 + 1);
if (NULL == *msg) {
mpz_clears(m, n, d, out, NULL_TERMINATED);
return -1;
}
/*转回字符串*/
for (unsigned int i = 0, j = 0; i < msg_len; i += 2, j++) {
(*msg)[j] = CHARS_TO_INT(message[i], message[i+1]);
}
(*msg)[msg_len / 2] = '\0';
mpz_clears(m, n, d, out, NULL_TERMINATED);
return 0;
}
typedef char* PCHAR;
int main()
{
PCHAR n, e, d;
generatekeypair(2048, &n, &e, &d);
printf("n: %s\ne: %s\nd: %s\n", n, e, d);
char msg[] = "I am What I am";
printf("\noriginal: %s\n", msg);
PCHAR cryptogram;
cipher(msg, strlen(msg), n, e, &cryptogram);
printf("encode: %s\n", cryptogram);
PCHAR message;
decipher(cryptogram, n, d, &message);
printf("decode: %s\n", message);
}