aboutsummaryrefslogtreecommitdiffstats
path: root/vigenere.c
blob: 5db3e185cc79822aff010b9bdb5f9a4fccd07e88 (plain)
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
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>

#define CTOI(c) ((c) - 'A')
#define ITOC(c) ((c) + 'A')

void preprocess(char *str);

int main(int argc, char **argv)
{
	char *key, *ciphertext, *decodepos, *decodeneg;
	long offset;
	size_t keylength, textlength;

	if (argc != 4) {
		fprintf(stderr, "Usage: %s <offset> <ciphertext> <key>\n", argv[0]);
		exit(EXIT_FAILURE);
	}

	offset = strtol(argv[1], NULL, 10);
	ciphertext = argv[2];
	key = argv[3];

	preprocess(key);
	preprocess(ciphertext);

	keylength = strlen(key);
	textlength = strlen(ciphertext);

	decodepos = malloc(textlength + 1);
	decodeneg = malloc(textlength + 1);

	for (size_t i = 0; i < textlength; i++) {
		decodepos[i] = ITOC((CTOI(ciphertext[i]) + offset + CTOI(key[i % keylength])) % 26);
		decodeneg[i] = ITOC((CTOI(ciphertext[i]) + 52 - offset - CTOI(key[i % keylength])) % 26);
	}

	decodepos[textlength] = '\0';
	decodeneg[textlength] = '\0';

	printf("+ %s\n- %s\n", decodepos, decodeneg);

	free(decodepos);
	free(decodeneg);

	return EXIT_SUCCESS;
}

void preprocess(char *str)
{
	char *src = str, *dest = str;

	while (*src != '\0') {
		if (isalpha(*src))
			*(dest++) = toupper(*src);
		src++;
	}

	*dest = '\0';
}