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));
}

#ifnecesidades 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:
aybson temporales para contener cadenas de la entrada.atambién se usa como parámetro para funcionarf.vcontiene los nombres de las macros yVcontiene los valores "V" de las macrostyTsomos titulares temporales de cuando crecemosvyVies un 'i'ncrementer para buclesses el tamaño de la matriz de macrosoes el recuento de los 'o'penifs dentro de un condicional falsog()'G'rows las matrices de macrosf()'f' encuentra una macro con el mismo valorvquead(y)'quita los últimosycaracteres de la entrada actualDes para dentro de un 'D'efineFes para ignorar un 'F'alse condicionalIes para 'Ignorarelse/elifdespué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
#includerelleno, 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.