-
Notifications
You must be signed in to change notification settings - Fork 1.1k
/
Affine_Substitution_Cipher.c
184 lines (173 loc) · 4.97 KB
/
Affine_Substitution_Cipher.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
// C program to illustrate Affine Substitution Cipher
/*
The Affine Substitution cipher is a type of monoalphabetic substitution cipher wherein each letter in an alphabet
is mapped to its numeric equivalent encrypted using a simple mathematical function and converted back to a letter.
*/
#include <stdio.h>
#include <string.h>
#include <ctype.h>
#include <stdlib.h>
// function declaration
int CalculateGCD(int);
int CalculateMultiplicativeInverse(int);
int main()
{
int size = 0;
int index, indx, k, gcd, alpha, beta, num_msg[size], numcipher[size], numdecipher[size], decipher[size], alphaInverse;
char *msg, *cipher;
printf("Input:\n");
printf("Enter the size of the array: ");
scanf("%d", &size);
// allocating memory dynamically
msg = (char *)malloc(size * sizeof(char));
if (msg == NULL)
{
printf("memory cannot be allocated\n");
}
else
{
printf("Enter the Cipher message: ");
scanf("%s", &*(msg));
}
//converting entered string to Capital letters
for (index = 0, indx = 0; index < strlen(msg); index++)
{
if (msg[index] != ' ')
{
msg[indx] = toupper(msg[index]);
indx++;
}
else
{
msg[indx] = ' ';
indx++;
}
}
//adding '\0' at end of string
msg[indx] = '\0';
printf("Enter the value of the key alpha: ");
scanf("%d", &alpha);
//Checking whether key alpha is in range of(1-25)
if (alpha < 1 || alpha > 25)
{
printf("Alpha should lie in between 1 and 25\nTry again !\n");
exit(0);
}
// calling CalculateGCD function
gcd = CalculateGCD(alpha);
if (gcd != 1)
{
printf("Mandatory condition : gcd(alpha,26)=1 but \n gcd(%d,26)=%d\n Try again !\n", alpha, gcd);
exit(0);
}
printf("Enter the value of the key beta: ");
scanf("%d", &beta);
//Checking whether key beta is in range of(0-25)
if (beta < 0 || beta > 25)
{
printf("Beta value should lie between 0 and 25\n");
exit(0);
}
//Conditions Over
//Storing message in terms of ASCII and replacing spaces used to -20
for (index = 0; index < strlen(msg); index++)
{
if (msg[index] != ' ')
num_msg[index] = msg[index] - 'A';
else
num_msg[index] = -20;
}
//Ciphering Process
printf("Output:\n");
printf("Encrypted Message is: ");
/* applying encryption formula ( alpha x + beta ) mod m {here x is num_msg[index] and m is 26} and added 'A' to
bring it in range of ASCII alphabet[ 65-90 | A-Z ] */
for (index = 0; index < strlen(msg); index++)
{
if (num_msg[index] != -20)
{
numcipher[index] = ((alpha * num_msg[index]) + beta) % 26;
printf("%c", (numcipher[index] + 'A'));
cipher[index] = numcipher[index] + 'A';
}
else
{
printf(" ");
cipher[index] = ' ';
}
}
//Deciphering Process
for (index = 0; index < strlen(cipher); index++)
{
//Storing encrypted message in terms of ASCII and replacing spaces to -20
if (cipher[index] != ' ')
numdecipher[index] = cipher[index] - 'A';
else
numdecipher[index] = -20;
}
//For Decryption we need to find multiplicative inverse of Alpha
alphaInverse = CalculateMultiplicativeInverse(alpha);
printf("\nDecrypted Message is: ");
/*Applying decryption formula alpha^-1 ( x - beta ) mod m {here x is numdecipher[index] and m is 26} and added 'A'
to bring it in range of ASCII alphabet[ 65-90 | A-Z ] */
for (index = 0; index < strlen(cipher); index++)
{
if (numdecipher[index] != -20)
{
decipher[index] = (alphaInverse * (numdecipher[index] - beta)) % 26;
if (decipher[index] < 0)
{
//To avoid negative numbers
decipher[index] = decipher[index] + 26;
}
printf("%c", (decipher[index] + 'A'));
}
else
//adding of white space
printf(" ");
}
printf("\n");
return 0;
}
// main ends here
//CalculateGCD Function definition
int CalculateGCD(int alpha)
{
int GCD;
int a = alpha;
int m = 26;
while (m != 0)
{
GCD = m;
m = a % m;
a = GCD;
}
return (a);
}
//CalculateMultiplicativeInverse Function definition
int CalculateMultiplicativeInverse(int alpha)
{
int count, a_inv;
for (count = 1; count <= alpha; count++)
{
a_inv = ((count * 26) + 1);
if (a_inv % alpha == 0)
{
break;
}
}
a_inv = a_inv / alpha;
return (a_inv);
}
/*
Input:
Enter the size of the array: 15
Enter the Cipher message: AFFINE CIPHER
Enter the value of the key alpha: 17
Enter the value of the key beta: 20
Output:
Encrypted Message is: UBBAHK CAPJKX
Decrypted Message is: AFFINE CIPHER
Time Complexity: O(n)
Space Complexity: O(1)
*/