-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathtea.c
129 lines (112 loc) · 3.7 KB
/
tea.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
/*
* ---------------------------------------------------------------------------
* Tiny Encryption Algorithm (TEA) implementation.
* ---------------------------------------------------------------------------
* Encrypts/Decrypts reading from input stream, writes back to output
* stream.
*
* Author: Arjob Mukherjee (arjobmukherjee@gmail.com)
* Dated : 16th August 2020
* ---------------------------------------------------------------------------
* */
#include "headers/tea.h"
#include "headers/unistd.h" // For read, write, access, close
#include "headers/fcntl.h" // For open
#include <stdio.h> // For printf and NULL, etc..
#include <string.h> // For memset
void decode(uint32_t *v, uint32_t *k)
{
uint32_t v0 = v[0],
v1 = v[1],
delta = 0x9e3779b9,
n = 32, // Invariant: Number of bits remaining
sum = delta * 32;
while(n--) {
v1 -= ((v0<<4) + k[2]) ^ (v0 + sum) ^ ((v0>>5) + k[3]);
v0 -= ((v1<<4) + k[0]) ^ (v1 + sum) ^ ((v1>>5) + k[1]);
sum -= delta;
}
v[0] = v0;
v[1] = v1;
}
void code(uint32_t *v, uint32_t *k)
{
uint32_t v0 = v[0],
v1 = v[1],
delta = 0x9e3779b9,
n = 32, // Invariant: Number of bits remaining
sum = 0;
while(n--) {
sum += delta;
v0 += ((v1<<4) + k[0]) ^ (v1 + sum) ^ ((v1>>5) + k[1]);
v1 += ((v0<<4) + k[2]) ^ (v0 + sum) ^ ((v0>>5) + k[3]);
}
v[0] = v0;
v[1] = v1;
}
bool encrypt_decrypt(int mode, char *key, int flags,
char *in_file, char *out_file)
{
char d[DATA_SIZE];
int len, inf, outf;
outf = (flags & TEA_FLAG_OUTPUT_STDOUT) ? STDOUT_FILENO : 0;
// 1. Check if out_file already exists (skip is output to stdout)
if ( outf == 0 && access(out_file, F_OK) == 0) {
fprintf(stderr,"Warning: Output file already exists %s\n",
out_file);
return false;
}
// Check if input file exists.
if (access(in_file, F_OK) == -1){
fprintf(stderr,"Warning: Skipping, Input file do not exists %s\n",
in_file);
return false;
}
// 2. Open the files
if ((inf = open (in_file, O_RDONLY|_O_BINARY)) == -1) {
perror("open - input");
return false;
}
// Feature output to stdout
if ( outf == 0 ) {
if ((outf = open (out_file,
O_CREAT|O_WRONLY|_O_BINARY,
DEFAULT_FILE_CREATION_MODE)) == -1) {
perror("open - output");
close(inf);
return false;
}
}
// Read 8 bytes from the file, until EOF is reached or some error
// occurs.
while ((len = read (inf,d,DATA_SIZE)) > 0){
// Fill rest of the d array with zero.
memset(&d[len], 0, DATA_SIZE - len);
// Performs Encryption / Decryption operation
if (mode == ENCRYPT)
code ((uint32_t *) d, (uint32_t *) key);
else
decode ((uint32_t *) d, (uint32_t *) key);
/*printf("\nkey:%u %u %u %u data:%u %u\n",
key[0],key[1],key[2],key[3],d[0],d[1]);*/
// Write back
if ((len = write (outf,d,DATA_SIZE)) < 0){
perror("write");
break;
}
}
// If write/read fails.
if (len < 0){
perror("read/write");
goto failed_exit;
}
// Clean up and exit
close (inf);
if ( outf != STDOUT_FILENO ) close (outf); // Do not close stdout
return true;
failed_exit:
// Clean up and exit
close (inf);
if ( outf != STDOUT_FILENO ) close (outf); // Do not close stdout
return false;
}