/******************************************************************************
Welcome to GDB Online.
GDB online is an online compiler and debugger tool for C/C++.
Code, Compile, Run and Debug online from anywhere in world.
*******************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include <time.h>
long modulo( long a, long b)
{
return a % b;
}
long euclides( long a, long b)
{
while ( b != 0 )
{
long aux = b;
b = a % b;
a = aux;
}
return a;
}
void eratostenes( long p)
{
int * numeros
= ( int * ) malloc ( p
* sizeof ( int ) ) ; if ( numeros == NULL) {
printf ( "Error en la asignación de memoria.\n " ) ; }
for ( int i = 2 ; i < p; i++ )
{
numeros[ i] = 1 ;
}
for ( int i = 2 ; i * i < p; i++ )
{
if ( numeros[ i] == 1 )
{
for ( int j = i * i; j < p; j += i)
{
numeros[ j] = 0 ;
}
}
}
for ( int i = 2 ; i < p; i++ )
{
if ( numeros[ i] == 1 )
{
}
}
}
void factorizacion( long p)
{
printf ( "Factores primos de %ld: " , p
) ; for ( long i
= 2 ; i
<= sqrt ( p
) ; i
++ ) {
while ( p % i == 0 )
{
p /= i;
}
}
if ( p > 1 )
{
}
}
long entero_menor( long p)
{
}
int es_primo( long n)
{
if ( n < 2 ) return 0 ;
for ( long i
= 2 ; i
<= sqrt ( n
) ; i
++ ) {
if ( n % i == 0 ) return 0 ;
}
return 1 ;
}
long primo_menor( long p)
{
if ( p <= 1 )
{
printf ( "El número debe ser mayor que 1.\n " ) ; }
long n;
do
{
n = entero_menor( p) ;
} while ( ! es_primo( n) ) ;
return n;
}
void inversa( long a, long b)
{
long r0 = b, r1 = a, x0 = 1 ;
long q1 = r0 / r1;
long r2 = modulo( r0, r1) ;
long x1 = - q1;
long x2;
while ( r2!= 0 )
{
r0 = r1;
r1 = r2;
q1 = r0 / r1;
r2 = modulo( r0, r1) ;
x2 = x0 - ( q1* x1) ;
x0 = x1;
x1 = x2;
}
if ( r1!= 1 )
{
printf ( "No existe la inversa.\n " ) ; }
if ( x0> 0 )
{
printf ( "La inversa es %ld.\n " , x0
) ;
}
else
{
printf ( "La inversa es %ld.\n " , x0
+ b
) ; }
}
long exponenciacion( long a, long x, long n)
{
long y = 1 ;
long u = a % n;
while ( x!= 0 )
{
if ( ( x% 2 ) == 1 ) y = ( y* u) % n;
x = x/ 2 ;
u = ( u* u) % n;
}
return y;
}
long es_generador( long g, long p)
{
long p1 = p - 1 ;
for ( long q = 2 ; q * q <= p1; q++ )
{
if ( p1 % q == 0 )
{
// Si g^((p-1)/q) ≡ 1 mod p → no es generador
if ( exponenciacion( g, p1 / q, p) == 1 )
return 0 ;
// Eliminar potencias de q para evitar redundancia
while ( p1 % q == 0 )
p1 /= q;
}
}
// Verificar el último factor si quedó uno mayor que sqrt(p1)
if ( p1 > 1 )
{
if ( exponenciacion( g, ( p - 1 ) / p1, p) == 1 )
return 0 ;
}
return 1 ;
}
void generadores( long p)
{
if ( ! es_primo( p) )
{
printf ( "El número %ld no es primo.\n " , p
) ; return ;
}
for ( long g = 2 ; g < p; g++ )
{
if ( es_generador( g, p) )
{
printf ( "%ld es un generador de Z%ld\n " , g
, p
) ; }
}
}
void legendre( long a, long p)
{
if ( euclides( a, p) != 1 )
{
printf ( "El símbolo de Legendre no está definido para estos valores.\n " ) ; return ;
}
long resultado = exponenciacion( a, ( p - 1 ) / 2 , p) ;
if ( resultado == p - 1 )
{
printf ( "El símbolo de Legendre (%ld/%ld) es -1.\n " , a
, p
) ; }
else
{
printf ( "El símbolo de Legendre (%ld/%ld) es 1.\n " , a
, p
) ; }
}
int jacobi( long a, long n)
{
if ( n <= 1 || n % 2 == 0 )
{
printf ( "El valor de n debe ser un número impar mayor que 1.\n " ) ; return 0 ;
}
int j = 1 ;
while ( a != 0 )
{
while ( a % 2 == 0 )
{
a /= 2 ;
if ( n % 8 == 3 || n % 8 == 5 )
{
j = - j;
}
}
long temp = a;
a = n % a;
n = temp;
if ( a % 4 == 3 && n % 4 == 3 )
{
j = - j;
}
}
if ( n == 1 )
{
return j;
}
return 0 ;
}
int main( )
{
char letra;
long resultado, a, b, c;
printf ( " b) MCD de a y b usando el algoritmo de Euclides\n " ) ; printf ( " c) Criba de Eratóstenes dado un número máximo p\n " ) ; printf ( " d) Factorización de un número p por fuerza bruta\n " ) ; printf ( " e) Generar un número entero aleatorio x menor que p\n " ) ; printf ( " f) Generar, por fuerza bruta, un número primo x menor que p\n " ) ; printf ( " g) Calcular, si existe, la inversa x = a-1 mod b\n " ) ; printf ( " h) Calcular la exponenciación Y = a^x mod n\n " ) ; printf ( " i) Calcular por fuerza bruta todos los generadores del grupo Zp con p primo\n " ) ; printf ( " j) Verificar analiticamente si un número x es generador del grupo Zp\n " ) ; printf ( " k) Calcular el símbolo de Legendre(a/p)\n " ) ; printf ( " l) Calcular el símbolo de Jacobi(a/n)\n " ) ; switch ( letra)
{
case 'a' :
printf ( "Introduce el primer número:\n " ) ; printf ( "Introduce el segundo número:\n " ) ; resultado = modulo( a, b) ;
break ;
case 'b' :
printf ( "Introduce el primer número:\n " ) ; printf ( "Introduce el segundo número:\n " ) ; resultado = euclides( a, b) ;
break ;
case 'c' :
printf ( "Introduce un número:\n " ) ; eratostenes( a) ;
break ;
case 'd' :
printf ( "Introduce un número:\n " ) ; factorizacion( a) ;
break ;
case 'e' :
printf ( "Introduce un número:\n " ) ; resultado = entero_menor( a) ;
break ;
case 'f' :
printf ( "Introduce un número:\n " ) ; resultado = primo_menor( a) ;
break ;
case 'g' :
printf ( "Introduce el primer número:\n " ) ; printf ( "Introduce el segundo número:\n " ) ; inversa( a, b) ;
break ;
case 'h' :
printf ( "Introduce el número a:\n " ) ; printf ( "Introduce el número x:\n " ) ; printf ( "Introduce el número n:\n " ) ; resultado = exponenciacion( a, b, c) ;
break ;
case 'i' :
printf ( "Introduce un número:\n " ) ; generadores( a) ;
break ;
case 'j' :
printf ( "Introduce un número x:\n " ) ; printf ( "Introduce un número p:\n " ) ; if ( es_generador( a, b) )
{
printf ( "%ld es un generador de Z%ld\n " , a
, b
) ; }
else
{
printf ( "%ld no es un generador de Z%ld\n " , a
, b
) ; }
break ;
case 'k' :
printf ( "Introduce el primer número:\n " ) ; printf ( "Introduce el segundo número:\n " ) ; legendre( a, b) ;
break ;
case 'l' :
printf ( "Introduce el primer número:\n " ) ; printf ( "Introduce el segundo número:\n " ) ; resultado = jacobi( a, b) ;
if ( resultado != 0 )
{
printf ( "El símbolo de Jacobi (%ld/%ld) es %ld.\n " , a
, b
, resultado
) ; }
else
{
printf ( "Error en el cálculo del símbolo de Jacobi.\n " ) ; }
break ;
case 'z' :
return 0 ;
default :
printf ( "Opción no válida.\n " ) ; break ;
}
return 0 ;
}
/******************************************************************************

Welcome to GDB Online.
GDB online is an online compiler and debugger tool for C/C++.
Code, Compile, Run and Debug online from anywhere in world.

*******************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include <time.h>

long modulo(long a, long b)
{
  return a % b;
}

long euclides(long a, long b)
{
    while (b != 0)
    {
        long aux = b;
        b = a % b;
        a = aux;
    }
    return a;
}

void eratostenes(long p)
{
    int *numeros = (int*) malloc(p * sizeof(int));
    if (numeros == NULL) {
        printf("Error en la asignación de memoria.\n");
        exit(1);
    }
    for (int i = 2; i < p; i++)
    {
        numeros[i] = 1;
    }

    for (int i = 2; i * i < p; i++)
    {
        if (numeros[i] == 1)
        {
            for (int j = i * i; j < p; j += i)
            {
                numeros[j] = 0;
            }
        }
    }

    for (int i = 2; i < p; i++)
    {
        if (numeros[i] == 1)
        {
            printf("%d ", i);
        }
    }
	free(numeros);
}

void factorizacion(long p)
{
    printf("Factores primos de %ld: ", p);
    for (long i = 2; i <= sqrt(p); i++)
    {
        while (p % i == 0)
        {
            printf("%ld ", i);
            p /= i;
        }
    }
	if (p > 1)
    {
        printf("%ld", p);
    }
    printf("\n");
}

long entero_menor(long p)
{
    return rand() % p;
}

int es_primo(long n)
{
    if (n < 2) return 0;
    for (long i = 2; i <= sqrt(n); i++)
    {
        if (n % i == 0) return 0;
    }
    return 1;
}

long primo_menor(long p)
{
	if (p <= 1)
	{
		printf("El número debe ser mayor que 1.\n");
		exit(1);
	}
    long n;
    do
    {
        n = entero_menor(p);
    } while (!es_primo(n));

    return n;
}

void inversa(long a, long b)
{
	long r0 = b, r1 = a, x0 = 1;
	long q1 = r0 / r1;
	long r2 = modulo(r0,r1);
	long x1 = -q1;
	long x2;
	while (r2!=0)
	{
		r0 = r1;
		r1 = r2;
		q1 = r0 / r1;
		r2 = modulo(r0,r1);
		x2 = x0 - (q1*x1);
		x0 = x1;
		x1 = x2;
	}
	if (r1!=1)
	{
		printf("No existe la inversa.\n");
		exit(0);
	}
	if (x0>0)
	{
		printf("La inversa es %ld.\n", x0);
		
	}
	else
	{
		printf("La inversa es %ld.\n", x0 + b);
	}
}

long exponenciacion(long a, long x, long n)
{
	long y = 1;
	long u = a % n;
	while(x!=0)
	{
		if((x%2)==1) y = (y*u) % n;
		x = x/2;
		u = (u*u) % n;
	}
	return y;
}

long es_generador(long g, long p)
{
    long p1 = p - 1;
    for (long q = 2; q * q <= p1; q++)
    {
        if (p1 % q == 0)
        {
            // Si g^((p-1)/q) ≡ 1 mod p → no es generador
            if (exponenciacion(g, p1 / q, p) == 1)
                return 0;

            // Eliminar potencias de q para evitar redundancia
            while (p1 % q == 0)
                p1 /= q;
        }
    }

    // Verificar el último factor si quedó uno mayor que sqrt(p1)
    if (p1 > 1)
    {
        if (exponenciacion(g, (p - 1) / p1, p) == 1)
            return 0;
    }

    return 1;
}

void generadores(long p)
{
    if (!es_primo(p))
    {
        printf("El número %ld no es primo.\n", p);
        return;
    }

    for (long g = 2; g < p; g++)
    {
        if (es_generador(g, p))
        {
            printf("%ld es un generador de Z%ld\n", g, p);
        }
    }
}

void legendre(long a, long p)
{
    if (euclides(a, p) != 1)
	{
        printf("El símbolo de Legendre no está definido para estos valores.\n");
        return;
    }
    long resultado = exponenciacion(a, (p - 1) / 2, p);
    if (resultado == p - 1)
	{
        printf("El símbolo de Legendre (%ld/%ld) es -1.\n", a, p);
    }
	else
	{
        printf("El símbolo de Legendre (%ld/%ld) es 1.\n", a, p);
    }
}

int jacobi(long a, long n)
{
    if (n <= 1 || n % 2 == 0)
	{
        printf("El valor de n debe ser un número impar mayor que 1.\n");
        return 0;
    }

    int j = 1;
    while (a != 0)
	{
        while (a % 2 == 0)
		{
            a /= 2;
            if (n % 8 == 3 || n % 8 == 5)
			{
                j = -j;
            }
        }
        long temp = a;
        a = n % a;
        n = temp;
        if (a % 4 == 3 && n % 4 == 3)
		{
            j = -j;
        }
    }
    if (n == 1)
	{
        return j;
    }
    return 0;
}

int main()
{
	srand(time(NULL));
	char letra;
	long resultado, a, b, c;
	printf("Pulse una letra:\n");
	printf("	a) a mod b\n");
	printf("	b) MCD de a y b usando el algoritmo de Euclides\n");
	printf("	c) Criba de Eratóstenes dado un número máximo p\n");
	printf("	d) Factorización de un número p por fuerza bruta\n");
	printf("	e) Generar un número entero aleatorio x menor que p\n");
	printf("	f) Generar, por fuerza bruta, un número primo x menor que p\n");
	printf("	g) Calcular, si existe, la inversa x = a-1 mod b\n");
	printf("	h) Calcular la exponenciación Y = a^x mod n\n");
	printf("	i) Calcular por fuerza bruta todos los generadores del grupo Zp con p primo\n");
	printf("	j) Verificar analiticamente si un número x es generador del grupo Zp\n");
	printf("	k) Calcular el símbolo de Legendre(a/p)\n");
	printf("	l) Calcular el símbolo de Jacobi(a/n)\n");
	printf("	z) Salir\n");
	scanf("%c",&letra);
	switch (letra)
	{
		case 'a':
			printf("Introduce el primer número:\n");
			scanf("%ld",&a);
			printf("Introduce el segundo número:\n");
			scanf("%ld",&b);
			resultado = modulo(a, b);
			printf("%ld\n", resultado);
			break;
			
		case 'b':
			printf("Introduce el primer número:\n");
			scanf("%ld",&a);
			printf("Introduce el segundo número:\n");
			scanf("%ld",&b);
			resultado = euclides(a, b);
			printf("%ld\n", resultado);
			break;
			
		case 'c':
			printf("Introduce un número:\n");
			scanf("%ld",&a);
			eratostenes(a);
			break;
			
		case 'd':
			printf("Introduce un número:\n");
			scanf("%ld",&a);
			factorizacion(a);
			break;
		
		case 'e':
			printf("Introduce un número:\n");
			scanf("%ld",&a);
			resultado = entero_menor(a);
			printf("%ld\n", resultado);
			break;
			
		case 'f':
			printf("Introduce un número:\n");
			scanf("%ld",&a);
			resultado = primo_menor(a);
			printf("%ld\n", resultado);
			break;
		
		case 'g':
			printf("Introduce el primer número:\n");
			scanf("%ld",&a);
			printf("Introduce el segundo número:\n");
			scanf("%ld",&b);
			inversa(a, b);
			break;
		
		case 'h':
			printf("Introduce el número a:\n");
			scanf("%ld",&a);
			printf("Introduce el número x:\n");
			scanf("%ld",&b);
			printf("Introduce el número n:\n");
			scanf("%ld",&c);
			resultado = exponenciacion(a,b,c);
			printf("%ld\n", resultado);
			break;
			
		case 'i':
			printf("Introduce un número:\n");
			scanf("%ld",&a);
			generadores(a);
			break;
			
		case 'j':
			printf("Introduce un número x:\n");
			scanf("%ld",&a);
			printf("Introduce un número p:\n");
			scanf("%ld",&b);
			if(es_generador(a,b))
			{
				printf("%ld es un generador de Z%ld\n", a, b);
			}
			else
			{
				printf("%ld no es un generador de Z%ld\n", a, b);
			}
			break;
			
		case 'k':
			printf("Introduce el primer número:\n");
			scanf("%ld",&a);
			printf("Introduce el segundo número:\n");
			scanf("%ld",&b);
			legendre(a, b);
			break;
			
		case 'l':
			printf("Introduce el primer número:\n");
			scanf("%ld",&a);
			printf("Introduce el segundo número:\n");
			scanf("%ld",&b);
			resultado = jacobi(a, b);
			if (resultado != 0)
	        {
                printf("El símbolo de Jacobi (%ld/%ld) es %ld.\n", a, b, resultado);
            }
	        else
	        {
                printf("Error en el cálculo del símbolo de Jacobi.\n");
            }
			break;
			
		case 'z':
			return 0;
			
		default:
			printf("Opción no válida.\n");
            break;
	}
	return 0;
}