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: por lmateu
