El objetivo es crear un preprocesador para el lenguaje C, lo más pequeño posible en términos de tamaño del código fuente en bytes , en su idioma preferido. Su entrada será un archivo fuente C y su salida será el código fuente preprocesado.
Los elementos que deberá poder procesar serán: eliminación de comentarios (línea / bloque), # incluir directivas (abriendo archivos en rutas relativas y reemplazando texto en el punto necesario), #define, #undef, #if, #elif, #else, #endif, #ifdef, #ifndef y defined (). Se pueden ignorar otras directivas de preprocesador de C como #pragmas o #errors.
No es necesario calcular expresiones aritméticas u operadores de comparación en las directivas #if, asumimos que la expresión se evaluará como verdadera siempre que contenga un número entero distinto de cero (su uso principal será para la directiva definida ()). A continuación se muestran ejemplos de posibles entradas y salidas (se recortaron posibles espacios en blanco adicionales en los archivos de salida para una mejor apariencia, no es necesario que su código lo haga). Un programa capaz de procesar los siguientes ejemplos correctamente se considerará suficiente.
----Input file: foo.c (main file being preprocessed)
#include "bar.h" // Line may or may not exist
#ifdef NEEDS_BAZZER
#include "baz.h"
#endif // NEEDS_BAZZER
#ifdef _BAZ_H_
int main(int argc, char ** argv)
{
/* Main function.
In case that bar.h defined NEEDS_BAZ as true,
we call baz.h's macro BAZZER with the length of the
program's argument list. */
return BAZZER(argc);
}
#elif defined(_BAR_H_)
// In case that bar.h was included but didn't define NEEDS_BAZ.
#undef _BAR_H_
#define NEEDS_BARRER
#include "bar.h"
int main(int argc, char ** argv)
{
return BARRER(argc);
}
#else
// In case that bar.h wasn't included at all.
int main()
{return 0;}
#endif // _BAZ_H_
----Input file bar.h (Included header)
#ifndef _BAR_H_
#define _BAR_H_
#ifdef NEEDS_BARRER
int bar(int * i)
{
*i += 4 + *i;
return *i;
}
#define BARRER(i) (bar(&i), i*=2, bar(&i))
#else
#define NEEDS_BAZZER // Line may or may not exist
#endif // NEEDS_BARRER
#endif // _BAR_H_
----Input file baz.h (Included header)
#ifndef _BAZ_H_
#define _BAZ_H_
int baz(int * i)
{
*i = 4 * (*i + 2);
return *i;
}
#define BAZZER(i) (baz(&i), i+=2, baz(&i))
#endif // _BAZ_H_
----Output file foopp.c (no edits)
int baz(int * i)
{
*i = 4 * (*i + 2);
return *i;
}
int main(int argc, char ** argv)
{
return (baz(&argc), argc+=2, baz(&argc));
}
----Output file foopp2.c (with foo.c's first line removed)
int main()
{return 0;}
----Output file foopp3.c (with bar.h's line "#define NEEDS_BAZZER" removed)
int bar(int * i)
{
*i += 4 + *i;
return *i;
}
int main(int argc, char ** argv)
{
return (bar(&argc), argc*=2, bar(&argc));
}
#if
necesidades deben ser apoyadas? es decir, ¿el preprocesador necesita admitir expresiones con operaciones aritméticas, bit a bit, etc.?Respuestas:
Flex, 1170 + 4 = 1174
1170 caracteres en el código flexible + 4 caracteres para un indicador de compilación. Para producir un ejecutable, ejecute
flex pre.l ; gcc lex.yy.c -lfl
.La entrada pierde memoria como un tamiz y no cierra los archivos incluidos.Pero de lo contrario, debería ser completamente funcional según las especificaciones.Alguna explicación:
a
yb
son temporales para contener cadenas de la entrada.a
también se usa como parámetro para funcionarf
.v
contiene los nombres de las macros yV
contiene los valores "V" de las macrost
yT
somos titulares temporales de cuando crecemosv
yV
i
es un 'i'ncrementer para bucless
es el tamaño de la matriz de macroso
es el recuento de los 'o'penif
s dentro de un condicional falsog()
'G'rows las matrices de macrosf()
'f' encuentra una macro con el mismo valorv
quea
d(y)
'quita los últimosy
caracteres de la entrada actualD
es para dentro de un 'D'efineF
es para ignorar un 'F'alse condicionalI
es para 'Ignorarelse
/elif
después de que se haya encontrado un verdadero condicional.EDITAR1: limpió muchas de las pérdidas de memoria e implementó el cierre de archivos
EDIT2: código modificado para manejar macros anidadas más correctamente
EDIT3: cantidad loca de golf
EDIT4: más golf
EDIT5: más golf; También he notado que mi llamado a fclose () causa problemas en algunas computadoras ... investigando esto.
fuente
#include
relleno, pero supongo que esto está relacionado con el error en la edición # 5. Además, no sustituye las macros, a pesar de que procesa con éxito los bloques #if, a menos que esté haciendo algo mal ... pero en general se ve muy bien y da una idea aproximada de lo que puede hacer un lexer, ya que Puedo entenderlo incluso en su forma de golf. Intente ver si los errores se pueden corregir, de lo contrario, está bien, ya que el código se explica bien, probablemente esta sea la respuesta elegida, ya que no hay otras entradas.