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 03:51] – [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 148: | Línea 148: | ||
pendientes, es decir las llamadas a funciones definidas en otros archivos. | pendientes, es decir las llamadas a funciones definidas en otros archivos. | ||
- | Cuidado, en esta fase no se realiza ninguna verificación de tipos. | + | Por ejemplo supongamos que tenemos 2 archivos a.c y b.c: |
+ | |||
+ | /* Este es a.c */ | ||
+ | int g(int x); | ||
+ | int a= 1; | ||
+ | float b; | ||
+ | |||
+ | int f() { | ||
+ | b= 1.0; | ||
+ | return g(a); | ||
+ | } | ||
+ | |||
+ | En a.c la referencia pendiente es g porque el ensamblador no puede determinar cual es la dirección de | ||
+ | la función g. | ||
+ | |||
+ | /* Este es b.c */ | ||
+ | #include < | ||
+ | extern int a; | ||
+ | float b; | ||
+ | |||
+ | int main() { | ||
+ | int x= f(); | ||
+ | printf(" | ||
+ | return x; | ||
+ | } | ||
+ | |||
+ | int g(int x) { | ||
+ | a= 2; | ||
+ | return x; | ||
+ | } | ||
+ | |||
+ | 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 | ||
+ | la va a declarar. | ||
+ | ' | ||
+ | |||
+ | 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. | ||
+ | |||
+ | % ld a.o b.o ... otros argumentos ... | ||
+ | |||
+ | |||
+ | ¿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 de tipos === | ||
+ | |||
+ | Cuidado, el linker no realiza ninguna verificación de tipos. | ||
que el archivo a.c contiene: | que el archivo a.c contiene: | ||
Línea 166: | 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. |
- | ar r bib.a f1.o f2.o f3.o | + | |
+ | void g(); | ||
+ | |||
+ | void f() { | ||
+ | g(); | ||
+ | } | ||
+ | |||
+ | /* bib2.c */ | ||
+ | void g() { | ||
+ | } | ||
+ | |||
+ | /* bib3.c */ | ||
+ | void h() { | ||
+ | } | ||
+ | |||
+ | Compilamos ambos archivos con: | ||
+ | |||
+ | % 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(); | ||
+ | } | ||
- | Luego se pueden especificar en gcc: | + | Compilamos incluyendo la biblioteca con: |
- | gcc a.c b.c bib.a | + | |
- | Si los archivos | + | Como a.o incluye |
- | contenido | + | en bib1.o, por lo tanto agrega bib1.o completo al binario. |
+ | referencia pendiente a g y la encuentra | ||
+ | lado, el archivo bib3.o no se agrega al binario | ||
- | Uno de los bugs más difíciles de diagnosticar que me ha tocado presenciar | + | 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 196: | 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.1346903493.txt.gz · Última modificación: 2012/09/06 03:51 por lmateu