Cifrado afínEl cifrado afín es un tipo de cifrado por sustitución en el que cada símbolo del alfabeto en claro (el alfabeto del texto en claro) es sustituido por un símbolo del alfabeto cifrado (el alfabeto del texto cifrado) siendo el número de símbolos del alfabeto en claro igual que el número de símbolos del alfabeto cifrado. Para hallar el símbolo del alfabeto cifrado que sustituye a un determinado símbolo del alfabeto en claro, se usa una función matemática afín en aritmética modular. Para poder aplicar la función matemática lo primero que hay que hacer es asignar un orden que a cada símbolo de cada uno de los alfabeto le asocie un número de orden. Una vez establecido esto, la fórmula matemática tiene la siguiente forma: Donde:
Los valores numéricos de y para traducirlos a símbolos de alfabetos, se interpretan como la posición del símbolo en el orden elegido de cada alfabeto. Sobre los valores de las constantes podemos decir:
Por ejemplo, podemos modelizar el cifrado César, suponiendo el orden natural del alfabeto de símbolos, por la siguiente ecuación: En este cifrado, la clave viene definida por los valores enteros y , y por el orden usado en el alfabeto de cifrado y en el alfabeto en claro, ambos alfabetos son el mismo número de símbolos, . Por tanto es un cifrado de clave simétrica. Para descifrar habrá que realizar el proceso inverso que se puede describir con la función matemática:
donde representa al inverso multiplicativo en aritmética modular (el menor número que ). Este tipo de cifrado se ha generalizado para su uso como cifrador de bloque en los llamados cifradores afínes por bloques. ClasificaciónPodemos clasificar los cifradores afines según los valores de y :
EjemploSupongamos que y y usamos el alfabeto castellano con en el orden habitual como alfabeto en claro y como alfabeto cifrado. Usando el cifrado afín definido de esa forma el símbolo '' se convertirá en , por tanto, el carácter asociado será el que ocupa la posición 20 empezando desde 0, la ''. Aplicando el mismo algoritmo podemos obtener que el texto cifrado de 'plantanuclear' es 'ntsdlspctmsb'. Para descifrar aplicamos la fórmula de descifrado. Para el inverso multiplicativo en aritmética modular es . Por tanto para tenemos . El carácter que ocupa la posición 1 del alfabeto es la ''. CriptoanálisisEl cifrado afín es vulnerable a ataques criptoanalíticos, los más habituales son:[2]
Análisis de frecuenciasEl cifrado afín, como cualquier otro cifrado por sustitución, es vulnerable a ataques por análisis de frecuencias. EjemploPor ejemplo, supongamos que observando un texto cifrado podemos inferir que usa un alfabeto de cifrado con m=26 símbolos, lo que nos lleva a suponer que ha usado el alfabeto del inglés. En el inglés los caracteres más frecuentes son la E y después la T. Si suponemos el orden habitual entonces E=4 y T=19. Analizando una cantidad suficiente de mensaje cifrado obtenemos que, por ejemplo, las más frecuentes son la V y después la E. Si suponemos el orden habitual V=21 y E=4. Del resultado anterior podemos ver que probablemente la E esté siendo sustutida por la V y la T por la E. Por tanto podemos obtener las siguientes congruencias: Si restamos de la primera ecuación la segunda, obtenemos lo siguiente: resolviendo obtenemos: Sustituyendo a en la primera ecuación original obtenemos: despejando A partir de estos valores puedo hallar el inverso multiplicativo, por ejemplo usando el algoritmo de Euclides extendido, . Ahora toca verificar nuestras suposiciones, si aplicando los valores obtenidos obtenemos un texto inteligible entonces habíamos acertado en nuestras suposiciones, en otro caso habremos fallado y será necesario hacer otras suposiciones. Búsqueda en el espacio de clavesLa función de Euler nos da el número de enteros de que tienen inverso para la multiplicación. Aplicando dicha fórmula a n=27 el valor de la función de Euler es 27*(1-1/3)=18. De este resultado podemos concluir que el número de posibles cifradores definidos con n=27 es 27(posibles valores de b)*18(posibles valores de a)=486. Este valor es realmente pequeño y un ordenador podría probar todas las posibles combinaciones muy rápidamente. Para hacer crecer este valor podemos usar una clave para que nos sirva como un desplazamiento adicional para cada símbolo. En esta situación, el cifrador se convierte en un cifrador de sustitución monoalfabético general en el que cada símbolo del alfabeto en claro es susituido por un símbolo del alfabeto cifrado. En este tipo de cifradores las posibles sustituciones son n!=27! (el número de las posibles permutaciones). Por ejemplo, si la clave es "HOLA" al primer símbolo de texto cifrado obtenido por la función, se le aplicará un desplazamiento de 8 (la H ocupa la octava posición), y así se sigue sucesivamente y cuando se acabe la clave volvemos a empezar. Implementaciones en CCifrado#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
/*Este código (creado por Arget) compilado tal cual está se convierte en un
cifrador afín usado de la siguiente manera:
affine-cipher a b plaintext
Siendo 'a' y 'b' las claves y 'plaintext' el texto a cifrar.
De manera predeterminada considera este alfabeto
ABCDEFGHIJKLMNOPQRSTUVWXYZ
Con A=1; B=2... sin embargo se puede modificar siguiendo las intrsucciones,
Para usar la 'Ñ' el valor de 'm' se debe modificar por 27 así como
descomentar/modificar las líneas indicadas. También se puede configurar
fácilmente para usar A=0;B=1...
Si se obtiene una '@' es debido a que el valor cifrado es 0, y al realizar
la conversión para el ASCII se debe sumar 64: 0 + 64 = 64, valor de '@'*/
int main(int argc, char** argv)
{
int a = atoi(argv[1]),
b = atoi(argv[2]),
i = 0,
m = 26, //m = 27 para usar Ñ
len = strlen(argv[3]);
char* plaintext;
plaintext = (char*) malloc(len);
strcpy(plaintext, argv[3]);
char ciphertext, x;
i = 0;
while(i < len)
{
x = toupper(plaintext[i]) - 64;//A=1; Cambiar 64 por 65 para A=0.
//if(x > 14) x++;//Para usar 'Ñ''
ciphertext = ((a * x + b) % m) + 64; //Quitar "+ 64" si se usa 'Ñ' / Cambiar 64 por 65 para A=0
printf("%c", ciphertext); //Cambiar "%c" por "%i\n" cuando se usa 'Ñ'
i++;
}
free(plaintext);
return 0;
}
Descifrado#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h> //toupper()
/*
Descifrador de cifrado afín por Arget.
Uso:
affine-decipher a b ciphertext
Usa el siguiente alfabeto
ABCDEFGHIJKLMNOPQRSTUVWXYZ
Con A=1, B=2...
*/
int cim(int a, int m);
int main(int argc, char** argv)
{
if(argc == 1) exit(-1);
int a = atoi(argv[1]),
b = atoi(argv[2]),
m = 26,
i,
len = strlen(argv[3]),
x,
ai = cim(a, m);
char* ciphertext;
ciphertext = (char*) malloc(len);
strcpy(ciphertext, argv[3]);
if(!ai){printf("Error!");exit(-2);}
for(i = 0; i < len; i++)
{
x = ciphertext[i] - 64;
x = ((ai * abs((x - b))) % m) + 64;
printf("%c", x);
}
return 0;
}
int cim(int a, int m)
{
int b, //Almacena el valor de b en (a * b)(mod m)
x; //Almacena el resultado de la op.
for(b = 0; b < m; b++)
{
x = (a * b) % m;
if(x == 1)
return b;
}
return 0;
}
Referencias
|