Erstellen Sie ein C-Programm, dessen Kompilierung in gcc am längsten dauert

27

Erstellen Sie ein kurzes C-Programm, dessen Kompilierung mit gcc absurd lange dauert. Die Einträge werden bewertet, indem die Kompilierung zeitgesteuert und anschließend die Kompilierungszeit des Referenzprogramms abgezogen wird.

Regeln

  • Beliebige C-Sprache oder gcc-Erweiterung
  • gcc 4.2.1
charliehorse55
quelle
1
Das Tagging wurde geändert, da [Code-Golf] explizit "Kurzschlusscode nach (Schlüssel-) Strichzahl" bedeutet.
dmckee
6
Die Division durch die Anzahl der Zeichen ist wenig sinnvoll, da jede vernünftige Herangehensweise an diese Herausforderung natürlich eine Kompilierungszeit-Komplexität von mehr als 0 ( n ) hat ein bisschen länger, was wahrscheinlich immer auf offensichtliche Weise möglich sein wird.
hörte auf,

Antworten:

13
#define a "xxxxxxxxxxx"
#define b a a a a a a a
#define c b b b b b b b
#define d c c c c c c c
#define e d d d d d d d
#define f e e e e e e e
#define g f f f f f f f
#define h g g g g g g g
#define i h h h h h h h
#define j i i i i i i i
z=j;
anonymer Feigling
quelle
Kompiliert nicht auf meinem Computer
FUZxxl
19
Zumindest muss die letzte Zeile in so etwas wie main(){char*z=j;}ein gültiges c-Programm geändert werden.
dmckee
2
Mein VS2012 hat keinen Platz mehr auf dem Haufen. Ich schätze /Zm, wir werden das
Problem
13

Sowohl Charlies Antwort als auch meine frühere Arbeit beschäftigen sich mit dem Prinzip, den Pre-Prozessor viel Code schreiben zu lassen , aber sie üben hauptsächlich den Pre-Prozessor selbst, den Lexer (gute Idee, da dieser Schritt traditionell langsam war) und den Parser aus. Meins versucht auch, die Optimierungs- und Codegenerierungsschritte durchzuführen, aber es gewinnt dort offensichtlich nicht viel.

Als ich darüber nachdachte, wie ein typischer c-Compiler funktioniert, stellte ich fest, dass wir dem Code für die Symboltabelle nichts zu tun hatten. Dieser Eintrag ist ein Versuch, dem abzuhelfen. Es soll an die grundlegende Objektorientierung in der c-Implementierung erinnern, macht aber nichts Interessantes: Verwendet die Präprozessor-Erweiterungstechnik, um eine Reihe von Objekten zu deklarieren und trivial (und fälschlicherweise) zu initialisieren. Objekt, das komplizierte Typen auf vielen Ebenen des Bereichs verwendet und sich gegenseitig bei verschiedenen Entfernungen abschattet. Es sollte der Symboltabelle eine echte Wirkung verleihen.

#include <stdlib.h>
#include <stdarg.h>
#include <string.h>
// Exercise the symbol table mechanism of the compiler in an effort to
// take a unreasonable about of time compiling

#define PTR(T) T*
#define CONST(T) T const
#define FUNC(NAME,RTYPE,ARG) RTYPE NAME(ARG)
#define FPTR(NAME,RTYPE,ARG) FUNC((*NAME),RTYPE,ARG)

// Forward decalration of repeated OO method pointers
typedef void* (*cctor_ptr)(void*this, void*that, ...);
typedef void* (*dtor_ptr)(void*this);

// Assumes three var-args: sizeof(payload type), cctor, dtor
void* default_ctor(void*this, ...){
  // Pull in variadac bits
  va_list list;
  va_start(list,this);
  int size=va_arg(list,int);
  cctor_ptr cctor=va_arg(list,cctor_ptr);
  dtor_ptr dtor=va_arg(list,dtor_ptr);
  va_end(list);
  // process
  if (!this) this = malloc(size);
  if (this) {
    memset(this,size,0);
    /* various dodges to install the cctor and dtor in the write places */
  }
  return this;
}
// Copies the payload from that to this; 
void* default_cctor(void*restrict this, void* restrict that, ...){
  // Pull in variadac bits
  va_list list;
  va_start(list,that);
  int size=va_arg(list,int);
  cctor_ptr cctor=va_arg(list,cctor_ptr);
  dtor_ptr dtor=va_arg(list,dtor_ptr);
  va_end(list);
  // process
  if (!this) this = malloc(size);
  if (this) {
    memcpy(this,that,size);
    /* various dodges to install the cctor and dtor in the write places */
  }
  return this;
}
// Assumes that his was allocated with malloc, does not use varargs
void* default_dtor(void*this, ...){
  free(this); 
  return NULL;
};

#define DECLARE_STRUCT(N) struct S##N##_s
#define TYPEDEF_ACCESSOR(N,T) typedef FPTR(f##N##_ptr,CONST(PTR(T)),PTR(CONST(struct S##N##_s)))
#define TYPEDEF_STRUCT(N,T) typedef struct S##N##_s {PTR(T)p; cctor_ptr cctor; dtor_ptr dtor; f##N##_ptr f##N;} S##N
#define OO_STRUCT(N,T) DECLARE_STRUCT(N); TYPEDEF_ACCESSOR(N,T); TYPEDEF_STRUCT(N,T)

OO_STRUCT(1,char);
OO_STRUCT(2,int);
OO_STRUCT(3,double*);
OO_STRUCT(4,S3);
OO_STRUCT(5,S4);
OO_STRUCT(6,S5);
OO_STRUCT(7,S6);
OO_STRUCT(8,S7);

#define SUBSCOPE(A) { \
    S1*A##1=default_ctor(NULL,sizeof(char),default_cctor,default_dtor); \
    S2 A##2; default_ctor(&A##2,sizeof(int),default_cctor,default_dtor); \
    S2*A##3=default_ctor(NULL,sizeof(double*),default_cctor,default_dtor); \
    S8 A##5; default_ctor(&A##5,sizeof(S4),default_cctor,default_dtor); \
    S6 A##6; default_ctor(&A##6,sizeof(S5),default_cctor,default_dtor); \
    S8*A##8=default_ctor(NULL,sizeof(S7),default_cctor,default_dtor); \
  }
#define SUBSCOPE2(A,B)  { \
    S2*B##5=default_ctor(NULL,sizeof(S4),default_cctor,default_dtor); \
    S4 A##7; default_ctor(&A##7,sizeof(S6),default_cctor,default_dtor); \
    SUBSCOPE(A) SUBSCOPE(B);                 \
  }
#define SUBSCOPE6(A,B,C)  { \
    S2*A##3=default_ctor(NULL,sizeof(double*),default_cctor,default_dtor); \
    S2 B##2; default_ctor(&B##2,sizeof(int),default_cctor,default_dtor); \
    S4*C##4=NULL;                           \
    SUBSCOPE2(A,C) SUBSCOPE2(B,C) SUBSCOPE2(A,B); \
  }
#define SUBSCOPE24(A,B,C,D) { \
    S1*D##1=default_ctor(NULL,sizeof(char),default_cctor,default_dtor); \
    S2 C##2; default_ctor(&C##2,sizeof(int),default_cctor,default_dtor); \
    S2*B##3=default_ctor(NULL,sizeof(double*),default_cctor,default_dtor); \
    S4 A##4; default_ctor(&A##4,sizeof(S3),default_cctor,default_dtor); \
    SUBSCOPE6(A,B,C) SUBSCOPE6(A,B,D) SUBSCOPE6(A,C,D) SUBSCOPE6(B,C,D); \
  }
#define SUBSCOPE120(A,B,C,D,E) { \
    S5*A##5=default_ctor(NULL,sizeof(S4),default_cctor,default_dtor); \
    S6*A##6=default_ctor(NULL,sizeof(S5),default_cctor,default_dtor); \
    S8 A##8; default_ctor(&A##8,sizeof(S7),default_cctor,default_dtor); \
    SUBSCOPE24(A,B,C,D) SUBSCOPE24(A,B,C,E) SUBSCOPE24(A,B,D,E); \
    SUBSCOPE24(A,C,D,E) SUBSCOPE24(B,C,D,E); \
  }
#define SUBSCOPE720(A,B,C,D,E,F) { \
    S5 A##5; default_ctor(&A##5,sizeof(S4),default_cctor,default_dtor); \
    S6 A##6; default_ctor(&A##6,sizeof(S5),default_cctor,default_dtor); \
    S8*A##8=default_ctor(NULL,sizeof(S7),default_cctor,default_dtor); \
    SUBSCOPE120(A,B,C,D,E) SUBSCOPE120(A,B,C,D,F) SUBSCOPE120(A,B,C,E,F); \
    SUBSCOPE120(A,B,D,E,F) SUBSCOPE120(A,C,D,E,F) SUBSCOPE120(B,C,D,E,F); \
  }

int main(){
  S4 s4;
  SUBSCOPE720(A,B,C,D,E,F)
}

Die Kompilierungszeit auf meinem Computer beträgt -O3mehr als 4 Sekunden mit und mehr als 1 Sekunde ohne Optimierung.


Offensichtlich besteht der nächste Schritt darin, die OO-Implementierung für eine BCD-Klasse abzuschließen und die Pi-Berechnungen damit erneut durchzuführen, damit beide Effekte hart zum Laufen kommen.

dmckee
quelle
12

Hier ist ein Riff des Exponential-Präprozessor-Expansions-Themas, das etwas minimal Interessantes bewirkt: Berechnet zwei Näherungen an pi nach Serienmethoden und vergleicht sie sowohl mit dem Wert in math.hals auch mit der üblichen Beschwörung.

Ungolfed.

#include <math.h>
#include <stdio.h>

// Some random bits we'll need
#define MINUSONODD(n) (n%2?-1:+1)
#define TWON(n) (2*(n))
#define NPLUSONE(n) ((n)+1)
#define TWONPLUSONE(n) NPLUSONE(TWON(n))
#define FACT(n) (tgamma(NPLUSONE(n)))

// The Euler series
//                           2^(2n) * (n!)^2      z^(2n+1)
// atan(z) = \sum_n=0^\infty --------------- * ---------------
//                               (2n+1)!       (1 + z^2)^(n+1)
#define TERMEULER(n,z) (pow(2,TWON(n))*                 \
            FACT(n)*FACT(n)*                \
            pow((z),TWONPLUSONE(n))/            \
            FACT(TWONPLUSONE(n)) /              \
            pow((1+z*z),NPLUSONE(n)) )

// The naive version
//                           (-1)^n * z^(2n+1)
// atan(z) = \sum_n=0^\infty -----------------
//                                2n + 1
#define TERMNAIVE(n,z) (MINUSONODD(n)*pow(z,TWONPLUSONE(n))/TWONPLUSONE(n))


// Define a set of bifruncations of the sum
#define N2TERMS(n,z,ALG)  (TERM##ALG(TWON(n),(z)) + TERM##ALG(TWONPLUSONE(n),(z)))
#define N4TERMS(n,z,ALG)  (N2TERMS(TWON(n),(z),ALG)+N2TERMS(TWONPLUSONE(n),(z),ALG))
#define N8TERMS(n,z,ALG)  (N4TERMS(TWON(n),(z),ALG)+N4TERMS(TWONPLUSONE(n),(z),ALG))
#define N16TERMS(n,z,ALG) (N8TERMS(TWON(n),(z),ALG)+N8TERMS(TWONPLUSONE(n),(z),ALG))
#define N32TERMS(n,z,ALG) (N16TERMS(TWON(n),(z),ALG)+N16TERMS(TWONPLUSONE(n),(z),ALG))

// Sum the fist 32*2+16 = 80 terms of a series...
#define PARTIALSUM(z,ALG) N32TERMS(0,(z),ALG)+N32TERMS(1,(z),ALG)+N16TERMS(4,(z),ALG)


int main(void){
  const double PI_TRAD = 4.0L * atan(1.0);
  const double PI_NAIVE = 4.0L * PARTIALSUM(0.999999L,NAIVE);
  const double PI_EULER = 4.0L * PARTIALSUM(0.999999L,EULER);
  printf("pi (math.h) = %10.8f\n",M_PI);
  printf("pi (trad.)  = %10.8f\n",PI_TRAD);
  printf("pi (NAIVE)  = %10.8f\n",PI_NAIVE);
  printf("pi (EULER)  = %10.8f\n",PI_EULER);
}

Es wird davon ausgegangen, dass Sie andere Arrangements verwenden gccund glibc/ oder nicht verwenden. Das time (1)Kompilieren mit -031 auf meinem 2,4-GHz-Intel-Core-2-Duo-MacBook dauert ungefähr 1,0 bis 1,1 Sekunden Prozessorzeit (bewertet mit ) . Eine Standardkompilierung benötigt ca. 0,4 Sekunden Prozessorzeit.

Leider kann ich gcc weder zum Auswerten pownoch tgammazur Compiler-Zeit bringen, was wirklich helfen würde.

Wenn Sie es ausführen, ist die Ausgabe:

pi (math.h) = 3.14159265
pi (trad.)  = 3.14159265
pi (NAIVE)  = 3.11503599
pi (EULER)  = 3.14159065

was zeigt, wie langsam die naive Reihe konvergiert.


1 Um eine möglichst konstante Faltung und Eliminierung von Unterausdrücken zu erzielen.

dmckee
quelle