Como cifrar letra por letra en C++

Iniciado por Royca3, 10 Junio 2013, 06:31 AM

0 Miembros y 1 Visitante están viendo este tema.

amchacon

Cita de: cpu2 en 17 Junio 2013, 08:47 AMPodrías utilizar los parametros argv, te ahorras ese fgets y esa aberración llamada scanf.
Perdiendo facilidad y sencillez...

Personalmente, si eres un poco peresozo y no quieres usar teclado. Simplemente pones esto al ejecutar el programa

programa < fichero.txt

Y en fichero pones todos los parámetros del teclado.

Cita de: cpu2 en 17 Junio 2013, 08:47 AMwhile (getchar()!='\n');

Venga hombre.
Es un método tan válido como cualquier otro para "limpiar" el buffer del teclado.

Cita de: cpu2 en 17 Junio 2013, 08:47 AMP.D: Yo dejo el tema, desde mi punto de vista un desplazamiento de bits o un xor, es mucho mejor que el de César, yo no daría más importancia a esto de verdad.
Bueno es curiosidad educativa  :silbar:
Por favor, no me manden MP con dudas. Usen el foro, gracias.

¡Visita mi programa estrella!

Rar File Missing: Esteganografía en un Rar

leosansan

#61
Cita de: cpu2 en 17 Junio 2013, 08:47 AM
Código demasiado largo y poco eficiente, para algo tan simple como el cifrado de César, y sin hablar del array.


Ya,ya, pero para mejor eficiencia ya están los dos posteados con anterioridad, el de CCross/cypscal/mío con una función y el uso de la librería ctype y el anterior mío sin más uso que el de el caracter ascii. Aún espero una propuesta "completa" por parte tuya, donde se pueda introducir la frase y el desplazamiento, no tan solo mover tres posiciones los caracteres sin tener en cuenta que si se llega al final se vuelve al comienzo del abecedario e incluir tanto minúsculas como mayúsculas e incluya sus printf y toda esa retaila.

De todas formas olvidas lo fundamental:

Cita de: leosansan en 13 Junio 2013, 16:48 PM

Creo que no has pillado la "intención" de la propuesta. Se trata de ver el "potencial" de los intervinientes en el foro y que en base a este tema hagan propuestas ingeniosasy/o "raritas" con la única finalidad de exprimir el ingenio, plasmar los conocimientos y ayudar/enseñar unos de otros.

Cita de: cpu2 en 17 Junio 2013, 08:47 AM

       printf("Ingrese una cadena: ");
       fgets (frase,255,stdin);
   
        puts("Ingrese desplazamiento: (1-25) ");
        scanf ("%d",&desplazamiento);


Podrías utilizar los parametros argv, te ahorras ese fgets y esa aberración llamada scanf.

Sí hombre, y volver cuarenta años atrás a programar tipo MS-DOS metiendo todo a mano y a base de comandos. No, lo siento, pero yo ya pasé esa época, tanto que por líneas de comandos no hagoun código ni de coñ*. Como ya he menciona en alguna ocasión, en mi caso al menos, un pasito atrás ni "pá" coger impulso.

Cita de: cpu2 en 17 Junio 2013, 08:47 AM
while (getchar()!='\n');

Venga hombre.


¿Alguna sugerencia diferente para limpiar el buffer?. No creo que seas de los que usan fflush (stdin). Reconozco humildemente que mis conocimientos en este mundillo del C/C++ son aún escasos, pero ¿alguna propuesta alternativa que nos ilumine a los neófitos?

Cita de: cpu2 en 17 Junio 2013, 08:47 AM
Un saludo.

Yo un "Saludito", como se dice por mi tierra. Lamento que hallas "agriado" un poco el tema con tu actitud ¿prepotente?, cuando mi única finalidad era que el personal que nos sigue se animase al "juego" de proponer alternativas al tema del cifrado César, como podría haber sido cualquier otro. Era sólo un reto que, por lo que veo y leo, tú te lo has tomado como algo personal, cosa que no entiendo. Y menos aún el que, repito/insisto no hallas hecho una propuesta alternativa en toda regla. Para criticar por criticar y deshagorte creo que existen otros foros más apropiados.



Cita de: cpu2 en 17 Junio 2013, 08:47 AM
P.D: Yo dejo el tema, desde mi punto de vista un desplazamiento de bits o un xor, es mucho mejor que el de César, yo no daría más importancia a esto de verdad.

Pues ya ves, esa es una sugerencia, que no aportación, que no caerá en saco roto, aunque intuyendo tu preparación habría esperado más de tí.

Como diría amchacon:

Cita de: amchacon en 17 Junio 2013, 11:37 AM
Perdiendo facilidad y sencillez...

Personalmente, si eres un poco peresozo y no quieres usar teclado. Simplemente pones esto al ejecutar el programa

programa < fichero.txt

Y en fichero pones todos los parámetros del teclado.
Es un método tan válido como cualquier otro para "limpiar" el buffer del teclado.
Bueno es curiosidad educativa :silbar:

En fin, que si se te hace muy largo el array del último post, y cuya única finalidad era la que era- releete el post para que lo entiendas bien- te dejo "otro" código con dos arrays, aunque más cortitos que el anterior , eso sí sin usar librerías, ni la operación módulo ni tan siquiera el código ascii, a"a pelo" vamos.....¡y que siga el juego! :

Código (cpp) [Seleccionar]
#include <stdio.h>
int main(void)
{
   char frase[256] = {0};
   int i=0,j=0,desplazamiento=0;
   char ABC[] = {"ABCDEFGHIJKLMNOPQRSTUVWXYZ"};
   char abc[] = {"abcdefghijklmnopqrstuvwxyz"};
   while(1)
   {
       printf("Ingrese una cadena: ");
       fgets (frase,255,stdin);
       do
       {
           puts("Ingrese desplazamiento: (1-25) ");
           scanf ("%d",&desplazamiento);
           while (getchar()!='\n');
       }while (desplazamiento<0 || desplazamiento>25);
       for(i = 0; frase[i]; i++)
       {
           for(j = 0; j<26; j++)
               {

                   if ( frase[i]==' ')
                       break;
                   if (frase[i]==abc[j] && j+ desplazamiento<26)
                   {
                       frase[i] =  abc[j+ desplazamiento];
                       break;
                   }
                   else if (frase[i]==abc[j] && j+ desplazamiento>=26)
                   {
                       frase[i] =  ABC[j+ desplazamiento-26];
                       break;
                   }
                   else if (frase[i]==ABC[j] && j+ desplazamiento<26)
                   {
                       frase[i] =  ABC[j+ desplazamiento];
                       break;
                   }
                   else if (frase[i]==ABC[j] && j+ desplazamiento>=26)
                       {
                           frase[i] =  abc[j+ desplazamiento-26];
                           break;
                   }
               }
       }
        printf("Cadena cifrada: %s\n\n", frase);
        for(i = 0; frase[i]; i++)
       {
           for(j = 0; j<26; j++)
               {

                   if ( frase[i]==' ')
                       break;
                   if (frase[i]==abc[j] && j- desplazamiento>=0)
                   {
                       frase[i] =  abc[j- desplazamiento];
                       break;
                   }
                   else if (frase[i]==abc[j] && j- desplazamiento<0)
                   {
                       frase[i] =  ABC[j-desplazamiento+26];
                       break;
                   }
                   else if (frase[i]==ABC[j] && j- desplazamiento>=0)
                   {
                       frase[i] =  ABC[j- desplazamiento];
                       break;
                   }
                   else if (frase[i]==ABC[j] && j- desplazamiento<=0)
                       {
                           frase[i] =  abc[j-desplazamiento+26];
                           break;
                   }
               }
       }
        printf("Cadena descifrada: %s\n\n", frase);
   }
   return 0;
}


Saluditos!

Royca3

emm! Sera que alguien puede pasarme un manual de C++??
Bien explicado y completo? en los que hayan aprendido ustedes
saludos

(tengo unos pero, no entiendo claramente)
Homo erectus

leosansan

#63
La verdad es que, en principio y sólo en principio, el cifrado César es muy endeble ante un simple ataque por "fuerza bruta". Bastaría ir probando con un desplazamiento de 0 a 26 y ver cuando obtenemos "algo" razonable/entendible.

Como muestra esta imagen en que se observa el cifrado y el "ataque" para descifrarlo:



Cosa que consigo con este simple código:

Código (cpp) [Seleccionar]

#include <stdio.h>
int main (void)
{
   int i,n=0,k=0;
   char frase [256];

   while (1)
   {
       n=0;
       puts ("\nIntroduce una frase: ");
       fgets (frase,255,stdin);
       do
       {
           printf ("\nIndica el desplazamiento: ");
           scanf ("%d",&n);
           while (getchar()!='\n');
       }while (n<0 || n>25);

       for (i=0;frase[i];i++)
       {
           if ((frase [i]>('z'-n) && frase [i]<='z') || (frase [i]>('Z'-n) && frase [i]<='Z' ))
               frase [i]=frase [i]-26+n;
           else if (frase [i]>='a' && frase [i]<=('z'-n) || (frase [i]>='A' && frase [i]<=('Z'-n) ))
               frase [i]+=n;
       }
       printf("\nCodificada con desplazamiento %d \n",n);
       printf ("%s\n",frase);
       puts("\nY ahora descodificada: ");
       for (k=0;k<26;k++)
       {
           for (i=0;frase[i];i++)
           {
               if ((frase [i]<('a'+k) && frase [i]>=('a')) || (frase [i]<('A'+k) && frase [i]>=('A') ))
               frase [i] =frase [i]+26-k;
               else if ((frase [i]>=('a'+k) && frase [i]<=('z') )|| (frase [i]>=('A'+k) && frase [i]<=('Z') ))
                   frase [i]-=k;
           }
           printf ("%s\n\n",frase);
           system ("pause");
       }
   }
   return 0;
}


Pero, pero .... con una pequeña variante lo podríamos hacer "casi" indescifrable, según comenta la Wikipedia:

"El cifrado Vigenère usa el cifrado César con un
desplazamiento diferente en cada posición del texto;
el valor del desplazamiento se define usando una
palabra clave repetitiva. Si la palabra clave fuera
escogida al azar y tan larga como el mensaje
(para que no se repita), el sistema resultante sería,
en teoría, indescifrable."



No creo que sea para tanto, pero cualquiera le lleva la contraria a Wikipedia. Como ya comenté en otro post anterior, más información en Cifrado César cortesía de Wikipedia.

Por si acaso, yo lo he intentado usando un valor aleatorio como desplazamiento para cada letra, con lo que el ataque por fuerza bruta anterior no sirve de nada, tendría que ser algo más refinado y respaldado con "potencia".

Como muestra otra imagen:



Pueden observar que la letra "a", que en el cifrado normal sería siempre la misma letra desplazada,  ahora cambia debido a que he usado un array aleatorio para cifrar la frase y sin ese array se haría muy "cuesta arriba" el descifrado ya que incluso el análisis por frecuencia fallaría, por la aleatoriedad mencionada.

Y he aquí la pequeña, pero "potente", variación:


Código (cpp) [Seleccionar]
#include <stdio.h>
#include <ctype.h>
#include <time.h>
void cesar (char *cadena , int letra, int *key);
int main()
{
   int  i=0,des=0, letra=0;
   srand((unsigned) time(NULL));
   char cadena[0x100]={0};
   puts("Ingrese una cadena: ");
   gets (cadena);
   int key[strlen(cadena)],_key[strlen(cadena)];
    for (i = 0;cadena[i]; i++)
   {
       key[i] = rand() % 0x1A+1 ;
       _key[i] = -key[i];
   }
   cesar (cadena,0x61,key);
   printf("\nCifrado es: %s \n", cadena);
   cesar (cadena,0x7A,_key);
   printf("\nDecifrado es: %s \n", cadena);
   return 0;
}
void cesar (char cadena[] , int letra, int *des)
{
   int i,letra1=0;
   for(i = 0; cadena[i]; i++)
    {
       if ( cadena[i]==' ')
           continue;
       letra1=letra;
       if(isupper(cadena[i])!=0)
           letra1-=0x20;
       cadena[i] =((cadena[i]-letra1+des[i] )%0x1A)+letra1;
     }
}



Como veis he sustituido los números decimales por sus correspondientes hexadecimales, sólo por darle un aspecto más críptico al tema.

Y el código propuesto tiene un valor añadido y es que cada mensaje usa un array aleatorio diferente en cada caso, por lo que la "interseptación" de dos mensajes no ayudaría en nada a los desencriptadores.

¡Anímate!, e intenta el descifrado de:

Código (cpp) [Seleccionar]
Cifrado es: Uhmvhvcjsakoo Mqoug ctz Xwrvdm Gycgs

Saluditos!
   

amchacon

Muy interesante Leo  ;-)

En mis tiempos hize un programa para descifrar palabras cifradas con César:

https://dl.dropboxusercontent.com/u/69551225/Desencriptador%20Secuencias%20%28WINDOWS%29.rar
https://dl.dropboxusercontent.com/u/69551225/Desencriptador%20Secuencias%20%28SOURCE%29.zip

Tienes que representar las letras con números y el programa busca palabras que sigan ese patrón. Por ejemplo:

CitarAmanda

Introduciríamos en el programa:

Citar121341

(cada letra se corresponde con un número).

El programa entonces, detectaría que la primera letra,tercera y quinta letra son iguales. Entonces buscaría en su base de datos palabras que sigan ese patrón. Los nombres obtenidos se guardan en un txt (deseché mostrarlos por pantalla por si la lista fuese muyy larga).
Por favor, no me manden MP con dudas. Usen el foro, gracias.

¡Visita mi programa estrella!

Rar File Missing: Esteganografía en un Rar

leosansan

#65
Cita de: amchacon en 18 Junio 2013, 22:44 PM
Muy interesante Leo  ;-)


Gracias, gracias e idem por el tuyo.

Ya para finalizar el tema no quería quedarme sin "actualizar" el cifrado César, que por motivos históricos se reducía al abecedario, y permitir el cifrado de buena parte del código ascii. Vamos que al menos contuviera símbolos tan actuales como $, ~, #, @ <, >, " ¿, ¡ etc. Una salida como ejemplo:


Código (cpp) [Seleccionar]

Ingrese una cadena:
"leosan@san.com! ¿san? #leo+san|=\12345!==$$

Cifrado es: Vêt├ôíêc£x}enªºn ╣░®╗M C├ç┼k¢qçðv|3xKfT=pYxx

Decifrado es: "leosan@san.com! ¿san? #leo+san|=\12345!==$$


Y esta vez no voy a colgar el código, creo que si han seguido los posts anteriores no tendrán ninguna dificultad en obtenerlo. Sólo pretendía poner de manifiesto que, como ya comenté, el cifrado César "actual" va más allá del simple abecedario original.


Saluditos!

do-while

#66
(He corregido algun error que había en alguna cadena. Ahora si que creo que está todo bien)

He añadido comprobación de parámetros en la función vigenere, que es donde deberían estar si se quiere añadir a una librería.

Ahora, en lugar de definir un nombre de fichero temporal mediante una constante simbolica, utilizo tmpnam() para asegurarme de que el nombre no se corresponda con ningún archivo existente.

¡Buenas!

Como estáis recopilando códigos de cifrado, aquí os dejo el del cifrado de Vigenere.

El código es un podo largo. El cifrado de Vigenere está al principio. Hay repeticion de código, pero lo he hecho así para evitar introducir demasiadas condiciones en el codigo de cifrado, ya que para archivos extensos el algoritmo sería muy lento. El codigo que hay en main es basicamente para leer y clasificar los datos de la linea de comandos (por eso es tan largo). Al final os dejo la sintaxis de llamada desde la linea de comandos y algunos ejemplos:


/*
* Cifrado de Vigenere
*
* Linea de comandos: nombre_programa -a alfabeto -[c|f] cadena|fichero -k clave -s [fichero_salida | std]
*
* Cifra los caracteres de la cadena (parametros -c "Cadena") o el fichero (parametros -f nombre_del_fichero)
* que coincidan con los del alfabeto (- que se le pasa, utilizando la clave dada (parametros -k clave)
*
*/

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define MAX_STRLEN 256

/*
* Las siguientes constantes sirven para decidir si la funcion vigenere actuara sobre una cadena o un fichero
*/
#define IN_CADENA  0
#define IN_FICHERO 1

/*
* Constantes para decidir si la funcion vigenere cifrara o descifrara la entrada
*/
#define ACCION_CIFRAR    0
#define ACCION_DESCIFRAR 1

/*
* vigenere(tipo entrada (cadena o fichero), cadena con texto o nombre de fichero con el input,
* alfabeto, clave, cifrar|descifrar, fichero de salida (stdin o nombre de arcivo))
*
* valor de retorno: 0 si algo falla, 1 si todo va bien.
*/

unsigned long flen(FILE *f)
{
    unsigned long pos,len;

    pos = ftell(f);
    fseek(f,0,SEEK_END);
    len = ftell(f);
    fseek(f,pos,SEEK_SET);

    return len;
}

int vigenere(int modo_input, char *cadena, char *alfabeto, char *clave, char accion, FILE *salida)
{
    FILE *entrada = NULL;
    unsigned long longitud,tam_bloque,tam_clave,tam_alfabeto;
    char *lector,*busqueda;
    int *desplazamiento,i,indice = 0;

    /* comprobamos que en alfabeto no haya letras repetidas */
    for(i = 0 ; alfabeto[i] ; i++)
        if(strchr(alfabeto + i + 1,alfabeto[i]))
            return 0;

    /* comprobamos que los caractreres de la clave esten en el alfabeto */
    for(i = 0 ; clave[i] ; i++)
        if(!strchr(alfabeto,clave[i]))
            return 0;

    if(modo_input == IN_CADENA)
        longitud = strlen(cadena);
    else if(modo_input == IN_FICHERO)
    {
        if(!(entrada = fopen(cadena,"rb")))
            return 0;

        longitud = flen(entrada);
    }
    else
        return 0;

    tam_alfabeto = strlen(alfabeto);
    tam_clave = tam_bloque = strlen(clave);

    if(!(lector = malloc(tam_bloque * sizeof(char))))
        return 0;

    if(!(desplazamiento = malloc(tam_bloque * sizeof(int))))
    {
        free(lector);
        return 0;
    }

    for(i = 0 ; clave[i] ; i++)
    {
        if(!(busqueda = strchr(alfabeto,clave[i])))
        {
            free(lector);
            free(desplazamiento);

            return 0;
        }

        if(accion == ACCION_CIFRAR)
            desplazamiento[i] = busqueda - alfabeto;
        else if(accion == ACCION_DESCIFRAR)
            desplazamiento[i] = alfabeto - busqueda; /* el desplazamiento de descifrado es el opuesto del de cifrado */
        else
        {
            free(lector);
            free(desplazamiento);

            return 0;
        }
    }

    if(modo_input == IN_CADENA)
    {
        while(longitud)
        {
            if(longitud < tam_bloque)
                tam_bloque = longitud;

            memcpy(lector,cadena,tam_bloque * sizeof(char));

            for(i = 0 ; i < tam_bloque ; i++)
            {
                if(busqueda = strchr(alfabeto,lector[i]))
                {
                    lector[i] = alfabeto[((busqueda - alfabeto) + desplazamiento[indice] + tam_alfabeto) % tam_alfabeto];
                    indice = (indice + 1) % tam_clave;
                }
            }

            fwrite(lector , tam_bloque*sizeof(char) , 1 , salida);

            if((tam_bloque == tam_clave) && (longitud > tam_bloque))
                cadena += tam_bloque;

            longitud -= tam_bloque;
        }
    }
    else if(modo_input == IN_FICHERO)
    {
        while(longitud)
        {
            if(longitud < tam_bloque)
                tam_bloque = longitud;

            fread(lector,tam_bloque * sizeof(char),1,entrada);

            for(i = 0 ; i < tam_bloque ; i++)
            {
                if(busqueda = strchr(alfabeto,lector[i]))
                {
                    lector[i] = alfabeto[((busqueda - alfabeto) + desplazamiento[indice] + tam_alfabeto) % tam_alfabeto];
                    indice = (++indice) % tam_clave;
                }
            }

            fwrite(lector , tam_bloque*sizeof(char) , 1 , salida);

            longitud -= tam_bloque;
        }
    }
    else
    {
        if(entrada)
            fclose(entrada);

        free(lector);
        free(desplazamiento);

        return 0;
    }

    if(entrada)
        fclose(entrada);

    free(lector);
    free(desplazamiento);

    return 1;
}

int main(int argc, char *argv[])
{
    char error[] = "%s [+|-]c (cifrar|descifrar) -a alfabeto -[s|f] [cadena|fichero] -k clave -o [fichero_salida|std]\n";
    char alfabeto[MAX_STRLEN] = "", fentrada[MAX_STRLEN] = "", clave[MAX_STRLEN] = "", fsalida[MAX_STRLEN] = "";
    char *texto = NULL,*ftemp = NULL;
    int sobreescribir = 0, modo = ACCION_CIFRAR + ACCION_DESCIFRAR + 1, i, j;
    FILE *f;

    if(argc != 10)
    {
        fprintf(stderr,error,argv[0]);
        return 1;
    }

    for(i = 1 ; i < 10 ; i += 2)
    {
        if(!strcmp(argv[i],"-a"))
        {
            if(strlen(alfabeto))
            {
                fprintf(stderr,"-a: Parametro repetido\n");
                return 2;
            }

            if(!strlen(argv[i + 1]))
            {
                fprintf(stderr,"-a: Alfabeto vacio no permitido\n");
                return 3;
            }

            for(j = 0 ; argv[i + 1][j] ; j++)
            {
                if(strchr(argv[i + 1] + j + 1, argv[i + 1][j]))
                {
                    fprintf(stderr,"-a: No se permiten caracteres repetidos en el alfabeto\n");
                    return 4;
                }
            }

            if(strlen(argv[i + 1]) >= MAX_STRLEN)
            {
                fprintf(stderr,"-a: El alfabeto es demasiado largo. Se truncara a partir del caracter %d.\n",MAX_STRLEN);

                strncpy(alfabeto, argv[i + 1], 255);
                alfabeto[255] = '\0';
            }
            else
                strcpy(alfabeto,argv[i + 1]);
        }
        else if(!strcmp(argv[i],"-s"))
        {
            if(texto || strlen(fentrada))
            {
                fprintf(stderr,"-s: Parametro -[s|f] repetido\n");
                return 5;
            }

            texto = argv[i + 1];
        }
        else if(!strcmp(argv[i],"-f"))
        {
            if(texto || strlen(fentrada))
            {
                fprintf(stderr,"-f: Parametro -[s|f] repetido\n");
                return 6;
            }

            if(strlen(argv[i + 1]) >= MAX_STRLEN)
            {
                fprintf(stderr,"-f: El nombre del fichero de entrada no debe exceder los %d caracteres\n",MAX_STRLEN);
                return 7;
            }

            if(!strlen(argv[i + 1]))
            {
                fprintf(stderr,"-f: El nombre del fichero de entrada no puede estar vacio\n");
                return 8;
            }

            strcpy(fentrada,argv[i + 1]);
        }
        else if(!strcmp(argv[i],"-k"))
        {
            if(strlen(clave))
            {
                fprintf(stderr,"-k: Parametro -k repetido\n");
                return 9;
            }

            if(!strlen(argv[i + 1]))
            {
                fprintf(stderr,"-k: No se permiten claves vacias");
                return 10;
            }

            if(strlen(argv[i + 1]) >= MAX_STRLEN)
            {
                fprintf(stderr,"-k: La clave no debe exceder los %d caracteres\n",MAX_STRLEN);
                return 11;
            }

            if(!strlen(argv[i + 1]))
            {
                fprintf(stderr,"-k: La clave no puede ser vacia\n");
                return 12;
            }

            strcpy(clave,argv[i + 1]);
        }
        else if(!strcmp(argv[i],"-o"))
        {
            if(strlen(fsalida))
            {
                fprintf(stderr,"-o: Parametro -o repetido\n");
                return 13;
            }

            if(strlen(argv[i + 1]) >= MAX_STRLEN)
            {
                fprintf(stderr,"-o: El nombre del fichero de salida no debe exceder los %d caracteres\n",MAX_STRLEN);
                return 14;
            }

            if(!strlen(argv[i + 1]))
            {
                fprintf(stderr,"-o: El nombre del fichero de salida no puede estar vacio\n");
                return 15;
            }

            strcpy(fsalida,argv[i + 1]);
        }
        else if(!strcmp(argv[i],"-c"))
        {
            if(!(modo ^ ACCION_CIFRAR) || !(modo ^ ACCION_DESCIFRAR))
            {
                fprintf(stderr,"-c: Parametro [+|-]c repetido\n");
                return 16;
            }

            modo = ACCION_DESCIFRAR;
            i--; /* este argumento no tiene asociado un parametro */
        }
        else if(!strcmp(argv[i],"+c"))
        {
            if(!(modo ^ ACCION_CIFRAR) || !(modo ^ ACCION_DESCIFRAR))
            {
                fprintf(stderr,"+c: Parametro [+|-]c repetido\n");
                return 17;
            }

            modo = ACCION_CIFRAR;
            i--; /* este argumento no tiene asociado un parametro */
        }
        else
        {
            fprintf(stderr,error,argv[0]);
            return 18;
        }
    }

    //comprobar que los caracteres de la clave estan en el alfabeto
    for(i = 0 ;  clave[i] ; i++)
    {
        if(!strchr(alfabeto,clave[i]))
        {
            fprintf(stderr,"La clave contiene caracteres que no aparecen en el alfabeto\n");
            return 19;
        }
    }

    if(!strcmp(fsalida,"std"))
    {
        f = stdout;
    }
    else
    {
        if(f = fopen(fsalida,"r"))
        {
            char opcion;

            do{
                printf("El fichero de salida ya existe. Deseas sobreescribirlo? (S/N) ");
                opcion = getchar();

                if(opcion > 'Z')
                    opcion += 'S' - 's';

                while(getchar() != '\n');

            }while(opcion!= 'S' && opcion!='N');

            if(opcion == 'S')
            {
                fclose(f);

                ftemp = tmpnam(NULL);

                if(!(f = fopen(ftemp,"wb")))
                {
                    fprintf(stderr,"%s: Error al abrir el fichero de salida\n",argv[0]);
                    return 20;
                }
                sobreescribir = 1;
            }
            else
            {
                fclose(f);
                return 0;
            }
        }
        else
            f = fopen(fsalida,"wb");
    }

    //si fsalida == std llamar a vigenere con los parametro y fsalida stdout
    if(texto)
    {
        if(!vigenere(IN_CADENA,texto,alfabeto,clave,modo,f))
        {
            fprintf(stderr,"%s no ha podido cifrar la informacion dada\n",argv[0]);
            return 21;
        }
    }
    else
    {
        if(!vigenere(IN_FICHERO,fentrada,alfabeto,clave,modo,f))
        {
            fprintf(stderr,"%s no ha podido cifrar la informacion dada\n",argv[0]);
            return 21;
        }
    }

    if(strcmp(fsalida,"std"))
        fclose(f);

    if(sobreescribir)
    {
        remove(fentrada);
        rename(ftemp,fentrada);
        remove(ftemp);
    }

    return 0;
}


Sintaxis (mi programa se llama vigenere.exe):

vigenere [+|-]c (cifrar|descifrar) -a alfabeto -[s|f] [cadena|fichero] -k clave -o [fichero_salida|std]

+c cifrar
-c descifrar

-a "alfabeto de cifrado": Los caracteres que se consideraran para cifrar.

-s "cadena": La entrada es la cadena que sigua a -s
-f "fichero": La entrada se tomara del nombre de fichero que siga a -f

-k "clave": Indicara el desplazamiento de los caracteres del texto en claro

-o "fichero": La salida se producira en el fichero indicado.
-o std: La salida se producira por pantalla.


Ejemplos:

vigenere +c -a "abcdefghijklmnopqrstuvwxyz .:,;-+" -s "el simbolo de la suma es + y el de la resta -" -k "ejemplo de clave" -o std

iu;;xxphostfp +e;.yypesl,dt ee+;hn;xpe-;vx :j


vigenere +c -a "abcdefghijklmnopqrstuvwxyz .:,;-+" -s "el simbolo de la suma es + y el de la resta -" -k "ejemplo de clave" -o prueba.txt

type prueba.txt
iu;;xxphostfp +e;.yypesl,dt ee+;hn;xpe-;vx :j


vigenere -c -a "abcdefghijklmnopqrstuvwxyz .:,;-+" -s "iu;;xxphostfp +e;.yypesl,dt ee+;hn;xpe-;vx :j" -k "ejemplo de clave" -o std

el simbolo de la suma es + y el de la resta -

vigenere -c -a "abcdefghijklmnopqrstuvwxyz .:,;-+" -f prueba.txt -k "ejemplo de clave" -o std

el simbolo de la suma es + y el de la resta -


El orden de los argumentos no tiene que ser el que he utilizado yo, puede ser cualquier otro.

¡Saludos!

PD:

El codigo ensablador que ha dejado cpu2 sigue la sintaxis Intel. GCC utiliza sintaxis AT&T (creo).

cpu2, ¿podrías traducir tu código de Intel a AT&T?
- Doctor, confundo los números y los colores.
- Vaya marrón.
- ¿Marrón? ¡Por el culo te la hinco!

erest0r

#67
No soy tan experto como uds pero siguiendo la regla del cifrado de Cesar logre hacer mi codigo asi:


#include <stdio.h>
#include <string.h>
#include <stdlib.h>

int main(){
   char oracion[100];
   char codificado[100];
   int i, desp;
    for(i = 0; i < 100; i++)
        codificado[i] = '\0';
   printf("Ingrese una oracion para cifrar:\n");
   fgets(oracion, 100, stdin);
   do{
       printf("Ingrese el numero de desplazamientos en el abecedario: ");
       scanf("%d", &desp);
       if(desp > 27)
           printf("El desplazamiento no puede ser mayor a 27\n\n");
   }
   while(desp > 27);
   for(i = 0; i < strlen(oracion); i++){
       if(oracion[i] >= 65 && oracion[i] <= 90){
           if(oracion[i] + desp > 90)
               codificado[i] = (oracion[i] - 90) + desp + 64;
           else
               codificado[i] = oracion[i] + desp;
       }
       else{
           if(oracion[i] >= 97 && oracion[i] <=122){
               if(oracion[i] + desp > 122)
                   codificado[i] = (oracion[i] - 122) + desp + 96;
               else
                   codificado[i] = oracion[i] + desp;
           }
           else
               codificado[i] = oracion[i];
       }
   }
   printf("La oracion codificada es: %s", codificado);
   getchar();
   return 0;
}


EDIT: Olvide inicializar el array "codificado"  ;D
Cruzar la calle junto a mucha gente cuando el semáforo sigue en rojo da seguridad y espíritu de equipo... o cruzamos todos o morimos juntos.