PC Laborator 2

De la WikiLabs
Versiunea din 25 septembrie 2017 11:28, autor: Rhobincu (discuție | contribuții) (Pagină nouă: = Obiective = În urma parcurgerii acestui laborator veți fi capabili: * să compilați un program C în linie de comandă; * să utilizați fișiere <code>Makefile</code> pentr...)
(dif) ← Versiunea anterioară | Versiunea curentă (dif) | Versiunea următoare → (dif)
Jump to navigationJump to search

Obiective

În urma parcurgerii acestui laborator veți fi capabili:

  • să compilați un program C în linie de comandă;
  • să utilizați fișiere Makefile pentru automatizarea procesului de compilare.

Scrierea, compilarea și executarea unui program în C

Editarea fișierului sursă

În cadrul laboratorului de PC, vom edita programele C într-un editor simplu de text, pentru a ne obișnui cu procesul de compilare și depanare a programelor. Editoarele de text care pot fi folosite sunt:

  • în mod text (din consolă): nano, mcedit, vim
  • în mod grafic: kate, gedit, notepad++

Fișierele C sunt fișiere text în care se scrie programul.

Generarea fișierului executabil

Din fișierul sursă, se genereaza fișierul executabil în mai multe etape, conform schemei de mai jos:

C executable generation.svg

Compilatorul de C pe care îl vom folosi se numește GNU Compiler Collection (sau scurt, gcc). Acesta, de fapt, nu este doar compilator, ci este un toolchain complet pentru generarea de fișiere executabile din fișiere surse C. Acesta poate face doar o parte din, sau toate etapele din schema de mai sus, în funcție de fanioanele utilizate. O parte din fanioanele posibile pentru gcc sunt (pentru lista completă studiaţi pagina de manual pentru GCC - man gcc):

Opțiune Efect
-o nume_fișier Numele fișierului de ieşire va fi nume_fişier. În cazul în care această opțiune nu este setată, se va folosi numele implicit (pentru fișiere executabile: a.out - pentru Linux).
-I cale_către_fișiere_antet Caută fișiere antet și în calea specificată.
-L cale_către_biblioteci Caută fișiere bibliotecă și în calea specificată.
-l nume_bibliotecă Link-editează biblioteca nume_bibliotecă. Atenție!!! nume_bibliotecă nu este întotdeauna același cu numele fișierului antet prin care se include această bibliotecă. Spre exemplu, pentru includerea bibliotecii de funcții matematice, fișierul antet este math.h, iar biblioteca este m.
-W tip_warning Afișează tipurile de avertismente specificate (Pentru mai multe detalii man gcc sau gcc –help). Cel mai folosit tip este all. Este indicat ca la compilarea cu -Wall să nu apară nici un fel de avertismente.
-c Compilează și asamblează, dar nu link-editează. Generează fișiere obiect, cu extensia .o.
-S Se oprește după faza de compilare, fară să asambleze. Rezultă cod assembler în fișiere cu extensia .s.
-E Se oprește după faza de preprocesare, fară să compileze. Rezultă cod C în fișiere cu extensia .c.
-O [0-3] Setează nivelul de optimizare, o valoare numerică între 0 și 3, 0 fiind fără optimizare (viteză de execuție minimă, dimensiune mare a executabilului, timp de compilare mic, cod ușor de depanat), iar 3 fiind optimizare maximă (viteză de execuție maximă, dimensiune mare a codului, mai mare decât la nivelul 2 datorită operațiunii de loop unrolling, timp de compilare mare, cod greu de depanat).
-g Adaugă simboluri de debug în executabil, fără de care depanarea nu este posibilă. Mai multe despre instrumentele de debug în laboratoarele următoare.


În continuare vom prezenta cateva exemple:

  • generarea directă a unui executabil numit hello, dintr-un singur fișier sursă, hello.c, utilizând biblioteca standard:
  student@pracsis01 ~/Desktop $ gcc hello.c -o hello
  • generarea directă a unui executabil numit hello, dintr-un singur fișier sursă, hello.c, utilizând biblioteca standard și biblioteca math:
  student@pracsis01 ~/Desktop $ gcc hello.c -o hello -lm
  • generarea directă a unui executabil numit hello, din două fișiere sursă, hello1.c și hello2.c, utilizând biblioteca standard:
  student@pracsis01 ~/Desktop $ gcc hello1.c hello2.c -o hello
  • generarea unui fișier obiect, numit hello.o, dintr-un fișier sursă numit hello.c:
  student@pracsis01 ~/Desktop $ gcc -c hello.c -o hello.o
  • generarea unui fișier obiect, numit hello.o, din două fișiere sursă, hello1.c și hello2.c:
  student@pracsis01 ~/Desktop $ gcc -c hello1.c hello2.c -o hello.o
  • generarea unui executabil numit hello, din două fișiere obiect, hello1.o și hello2.o, utilizând biblioteca standard și biblioteca math:
  student@pracsis01 ~/Desktop $ gcc hello1.o hello2.o -o hello -lm
  • generarea unui fișier cu cod de asamblare, numit hello.s, dintr-un fișier sursă numit hello.c:
  student@pracsis01 ~/Desktop $ gcc -S hello.c -o hello.s
  • generarea unui fișier sursă preprocesat, numit hello_pre.c, dintr-un fișier sursă numit hello.c:
  student@pracsis01 ~/Desktop $ gcc -E hello.c -o hello_pre.c

Executarea programului

Pentru a executa un program sau script în consola Linux, numele fișierului se precede cu caracterele ./ :

  student@pracsis01 ~/Desktop $ gcc hello.c -o hello
  student@pracsis01 ~/Desktop $ ./hello

Compilarea automată a multiple surse - Makefiles

Dezvoltarea unei aplicații în limbajul C implică multe iterații de tip design - dezvoltare - depanare - integrare. La fiecare modificare a sursei, pentru a putea rula aplicația, codul trebuie recompilat. Odată cu creșterea numărului de linii de cod și a fișierelor sursă, timpul necesar operației de generare a executabilului (build) crește și el, până în punctul în care durează zeci de minute. Aceasta este în special o problemă când dezvoltarea se face pe microcontroler. Se evidențiază următoarele probleme:

  • Dacă o sursă este modificată, ea trebuie recompilată prin rescrierea în consolă a comenzii de compilare. Această comandă poate fi foarte lungă, datorită multiplelor biblioteci și fanioane folosite, prin urmare este incomod de scris de fiecare dată. Este deci nevoie de o soluție pentru a predefini comanda de compilare și a o rula fără efort de fiecare dată.
  • Dacă o aplicație este formată din multe fișiere sursă, compilarea tuturor acestor fișiere durează. În practică, se cere doar compilarea surselor care s-au modificat și apoi rularea operației de link-editare a fișierelor obiect în fișierul executabil (compilarea este operația care consumă cel mai mult timp din tot procesul de build). Este deci nevoie de un sistem care să detecteze care surse s-au modificat, apoi să le compileze strict pe acelea.

Exisă mai multe sisteme de build care rezolvă aceste probleme, dar de departe cel mai folosit în practică este utilizarea aplicației make, și configurarea acesteia prin Makefiles.

Anatomia unui Makefile - rețete de fișiere

Scrierea unui Makefile este extrem de simplă și se bazează pe ideea de rețetă: pentru a genera un fișier destination, este nevoie de unul sau mai multe fișiere sursă sourceA, sourceB, sourceC, ... și de o comandă command care să genereze fișierul A din B, C, D, ... Această rețetă se scrie într-un fișier cu numele Makefile în felul următor:

destination: sourceA sourceB sourceC ...
  [tab]  command


Atenție: Fișierele Makefile sunt sensibile la spații! Nu aveți voie să puneți linii goale decât între rețete, și nu aveți voie să plasați spații la începutul liniei. În plus, comenzile care trebuie rulate pentru generarea fișierului trebuie obligatoriu precedate de un caracter TAB, acesta nu se poate înlocui cu spații.

De exemplu, o rețetă care să genereze un executabil numit hello dintr-o sursă C numită hello.c se scrie în felul următor:

hello: hello.c
	gcc hello.c -o hello

Pentru a face build-ul, se rulează simplu comanda make, care va rula automat prima rețetă din fișier. Dacă vrem să specificăm care rețetă este rulată, atunci numele ei urmează comenzii make: make hello.

Mai departe, dorim să mai definim o rețetă care să șteargă fișierele generate (să facă curat). Vom numi această rețetă clean. Această rețetă nu generează un fișier (nu vrem să creăm un fișier numit clean), astfel această rețetă este una falsă, care se declară PHONY:

hello: hello.c
	gcc hello.c -o hello

.PHONY: clean
clean:
	rm -f hello

Putem acum oricând să ștergem fișierul generat rulând comanda make clean.

Regulile după care se execută o rețetă sunt următoarele:

  1. Dacă numele rețetei NU este un fișier generat de către comandă, atunci acel target e PHONY, cum este clean. În cazul acesta, comanda se rulează de fiecare data cand se face rețeta.
  2. Dacă numele rețetei ESTE un fișier dar lista de surse lipseste, atunci comanda se ruleaza doar daca fișierul nu există.
  3. Dacă numele rețetei ESTE un fișier și lista de surse există, comanda se rulează doar dacă fișierul nu există SAU dacă oricare din surse e mai nouă decat fișierul.
  4. Dacă pentru oricare din sursele unei rețete există o altă rețetă care îl generează, atunci se rulează automat întâi acea rețetă, indiferent dacă sursa există sau nu. Dacă fișierul sursă NU există și nu există nici o rețetă care să îl genereze, atunci aplicația make va genera o eroare: No rule to make target.

Optimizări

Când se scrie o rețetă, numele fișierelor sursă și/ sau a fișierui destinație se poate schimba în timpul dezvoltării aplicației. Dacă numele acestora apare și în comenzile din rețetă, trebuie avut grija ca noile nume să fie modificate peste tot unde apar. De exemplu, dacă Makefile-ul de mai sus, dorim să modificăm numele sursei din hello.c în byebye.c, trebuie modificat în două locuri:

hello: byebye.c
	gcc byebye.c -o hello

.PHONY: clean
clean:
	rm -f hello

Astfel, se poate face o optimizare. Există două scurtături care pot fi folosite pentru numele fișierului destinație ($@) și pentru lista de fișiere sursă ($^):

hello: byebye.c
	gcc $^ -o $@

.PHONY: clean
clean:
	rm -f hello

Un alt avantaj este acela că dacă executabilul hello va trebui generat din două fișiere sursă, noul fișier nu trebuie adăugat decât în lista de surse, nu și în comandă, lista propagându-se automat datorită lui $^:

hello: byebye.c hello.c
	gcc $^ -o $@

.PHONY: clean
clean:
	rm -f hello


GCC

  1. Deschideți un editor de text.
  2. Scrieți într-un fișier nou următorul cod C
    #include <stdio.h>
    
    int main(){
        printf("Hello PC Lab!\n");
        return 0;
    }
    
  3. Salvați fișierul în directorul /home/student/work/PC/seriaF/grupaN/nume/lab1 cu numele source.c (directorul va trebui refăcut)
  4. Navigați în consolă în directorul /home/student/work/PC/seriaF/grupaN/nume/lab1
  5. Compilați fișierul sursă într-un executabil numit main
  6. Executați programul.
  7. Ștergeți fișierul main.
  8. Generați un fișier obiect, din fișierul source.c, numit source.o
  9. Generați un executabil numit main, din fișierul obiect.
  10. Executați programul.
  11. Generați un fișier cu cod asamblare, numit source.s, din fișierul sursă.
  12. Afișați pe ecran conținutul fișierului source.s
  13. Generați un fișier cu cod C preprocesat, numit source_pre.c, din fișierul sursă.
  14. Afișați pe ecran conținutul fișierului source_pre.c
  15. Afișați pe ecran conținutul fișierului source.o. Explicați comportamentul.

Makefiles

  1. Scrieți un Makefile cu două rețete:
    • una care să genereze executabilul main din sursa source.c
    • una care să șteargă fișierul generat.
  2. Rulați comanda make clean și verificați că executabilul a fost șters.
  3. Rulați comanda make, verificați că programul a fost compilat și rulați-l.
  4. Rulați comanda make din nou. Ce observați?
  5. Modificați fișierul sursă ca în loc de Hello PC Lab! să afișeze pe ecran Makefiles are awesome! (nu uitați să salvați fișierul după modificare).
  6. Rulați din nou comanda make și apoi executați programul.
  7. Modificați Makefile astfel încât să genereze executabilul dintr-un fișier obiect, iar fișierul obiect să fie generat din fișierul sursă.