compilacion
Diferencias
Muestra las diferencias entre dos versiones de la página.
Ambos lados, revisión anteriorRevisión previaPróxima revisión | Revisión previa | ||
compilacion [2012/09/06 11:35] – [Link] lmateu | compilacion [2014/09/04 13:42] (actual) – [Preproceso] lmateu | ||
---|---|---|---|
Línea 1: | Línea 1: | ||
- | ===== Fases de la Compilación ===== | + | ===== Etapas |
La compilación de un programa en C pasa por varias etapas desde | La compilación de un programa en C pasa por varias etapas desde | ||
Línea 49: | Línea 49: | ||
Recuerde que cada identificador que se usa en C debe haber sido declarado previamente, | Recuerde que cada identificador que se usa en C debe haber sido declarado previamente, | ||
modo se considera un error. | modo se considera un error. | ||
- | de un tipo en un archivo, pero de otro en otro archivo. | + | de un tipo en un archivo, pero de otro tipo en otro archivo. |
este caso porque simplemente no tiene la información para detectar el error. | este caso porque simplemente no tiene la información para detectar el error. | ||
se denomina compilación **separada**. | se denomina compilación **separada**. | ||
Línea 77: | Línea 77: | ||
% gcc -E prog.c | % gcc -E prog.c | ||
| | ||
- | La salida | + | La salida |
int main(int argc, char **argv) { | int main(int argc, char **argv) { | ||
Línea 153: | Línea 153: | ||
int g(int x); | int g(int x); | ||
int a= 1; | int a= 1; | ||
- | float g; | + | float b; |
| | ||
int f() { | int f() { | ||
+ | b= 1.0; | ||
return g(a); | return g(a); | ||
} | } | ||
Línea 163: | Línea 164: | ||
/* Este es b.c */ | /* Este es b.c */ | ||
+ | #include < | ||
extern int a; | extern int a; | ||
- | float g; | + | float b; |
| | ||
int main() { | int main() { | ||
int x= f(); | int x= f(); | ||
- | | + | |
+ | return x; | ||
+ | } | ||
+ | |||
+ | int g(int x) { | ||
+ | a= 2; | ||
+ | return x; | ||
} | } | ||
Aquí las referencias pendientes son a y f. Observe que a fue declarada como ' | Aquí las referencias pendientes son a y f. Observe que a fue declarada como ' | ||
al ensablador que no reserve espacio para ella porque se trata de una promesa de que otro archivo | al ensablador que no reserve espacio para ella porque se trata de una promesa de que otro archivo | ||
- | la va a declarar. | + | la va a declarar. |
- | ' | + | ' |
- | % ln a.o b.o | + | La fase de link en Unix la realiza el comando '' |
+ | un error histórico, porque su tarea no es la de un //loader// (cargardor). | ||
+ | es la componente del núcleo del sistema operativo que carga un archivo ejecutable en la memoria del | ||
+ | computador para que sea ejecutado. | ||
+ | se conserva quizás por compatibilidad. | ||
- | ¿Pero que pasa con g que aparece declarada en dos archivos? | + | % ld a.o b.o ... otros argumentos ... |
- | eliminando una de las declaraciones. La variable aparecerá una sola vez. | + | |
- | === Errores | + | |
+ | ¿Pero que pasa con b que aparece declarada en dos archivos? | ||
+ | eliminando una de las declaraciones. | ||
+ | se puede hacer si la variable se inicializa en los 2 archivos (aunque sea el mismo valor). | ||
+ | En ese caso el linker reporta la variable como una definición múltiple. | ||
+ | |||
+ | === Inconsistencia | ||
Cuidado, el linker no realiza ninguna verificación de tipos. | Cuidado, el linker no realiza ninguna verificación de tipos. | ||
Línea 201: | Línea 218: | ||
evitar este tipo de errores. | evitar este tipo de errores. | ||
- | Un error típico en grandes proyectos ocurre cuando 2 archivos | + | === Static === |
+ | |||
+ | Un error típico en grandes proyectos ocurre cuando 2 archivos | ||
+ | El linker no puede determinar cual usar, y reporta la función como una definición múltiple. | ||
Para disminuir este tipo de errores se puede limitar la visibilidad de una funciona o variable global | Para disminuir este tipo de errores se puede limitar la visibilidad de una funciona o variable global | ||
a solo el archivo en donde declara: | a solo el archivo en donde declara: | ||
- | static | + | static |
... | ... | ||
} | } | ||
De esta forma si otra función f se declara en otro archivo, no habrá colisión de nombres. | De esta forma si otra función f se declara en otro archivo, no habrá colisión de nombres. | ||
+ | Observe que uso de static acá no tiene nada que ver con el atributo static de Java. | ||
==== Las bibliotecas ==== | ==== Las bibliotecas ==== | ||
Las bibliotecas son archivos con la extensión ' | Las bibliotecas son archivos con la extensión ' | ||
- | de manejarlos como una unidad. | + | de manejarlos como una unidad. |
- | | + | |
+ | void g(); | ||
+ | |||
+ | void f() { | ||
+ | g(); | ||
+ | } | ||
- | Luego se pueden especificar en gcc: | + | /* bib2.c */ |
+ | void g() { | ||
+ | } | ||
- | | + | |
+ | void h() { | ||
+ | } | ||
- | Si los archivos | + | Compilamos ambos archivos |
- | contenido en bib.a, entonces se agrega ese archivo al binario. | + | |
- | Uno de los bugs más difíciles de diagnosticar que me ha tocado presenciar | + | % gcc -c bib1.c bib2.c bib3.c |
+ | |||
+ | Lo que genera los archivos bib1.o y bib2.o. | ||
+ | |||
+ | % ar r bib.a bib1.o bib2.o bib3.o | ||
+ | |||
+ | Ahora podemos usar f o g desde otro programa: | ||
+ | |||
+ | /* a.c */ | ||
+ | int main() { | ||
+ | f(); | ||
+ | } | ||
+ | |||
+ | Compilamos incluyendo la biblioteca con: | ||
+ | |||
+ | % gcc a.c bib.a | ||
+ | |||
+ | Como a.o incluye una referencia pendiente a f, el linker la busca en las bibliotecas y la encuentra | ||
+ | en bib1.o, por lo tanto agrega bib1.o completo al binario. | ||
+ | referencia pendiente a g y la encuentra en bib2.o al binario. | ||
+ | lado, el archivo bib3.o no se agrega al binario ejecutable. | ||
+ | |||
+ | Uno de los bugs más difíciles de diagnosticar que me ha tocado presenciar | ||
un función de biblioteca llamaba a otra función dentro de la misma biblioteca pero que casualmente | un función de biblioteca llamaba a otra función dentro de la misma biblioteca pero que casualmente | ||
se llamaba igual que una variable global de los fuentes. | se llamaba igual que una variable global de los fuentes. | ||
Línea 231: | Línea 283: | ||
¡El programa se caía con un segmentation fault inexplicable! | ¡El programa se caía con un segmentation fault inexplicable! | ||
+ | Esto se puede reproducir cambiando a.c por: | ||
+ | |||
+ | /* a.c */ | ||
+ | float g= 3.14; | ||
+ | int main() { | ||
+ | f(); | ||
+ | } | ||
+ | Ejercicio: Compile y ejecute para observar el segmentation fault. | ||
+ | la ejecución hacia la variable en punto flotante g, como si ahí hubiesen instrucciones. |
compilacion.1346931343.txt.gz · Última modificación: 2012/09/06 11:35 por lmateu