O objetivo é criar um pré-processador para a linguagem C, o menor possível em termos de tamanho do código-fonte em bytes , no idioma preferido. Sua entrada será um arquivo de origem C e sua saída será o código-fonte pré-processado.
Os itens que ele deverá poder processar são: remoção de comentários (linha / bloco), #include diretivas (abrindo arquivos em caminhos relativos e substituindo o texto no ponto necessário), #define, #undef, #if, #elif, #else, #endif, #ifdef, #ifndef e defined (). Outras diretivas do pré-processador C, como #pragmas ou #errors, podem ser ignoradas.
Não há necessidade de calcular expressões aritméticas ou operadores de comparação nas diretivas #if, assumimos que a expressão será avaliada como verdadeira desde que contenha um número inteiro diferente de zero (seu principal uso será a diretiva definida ()). Exemplos de possíveis entradas e saídas a seguir (possíveis espaços em branco extras nos arquivos de saída foram aparados para melhor aparência, não é necessário que seu código faça isso). Um programa capaz de processar adequadamente os seguintes exemplos será considerado 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
precisa ser suportado? ou seja, o pré-processador precisa suportar expressões com operações aritméticas, bit a bit, etc.?