Scrivere Software è un piacere. Un programma non solo deve funzionare bene ed essere efficiente (questo si dà per scontato), ma deve essere anche bello ed elegante da leggere, comprensibile e facile da manutenere, sia per l'autore che per eventuali lettori futuri. Programmare bene in C è un'arte.
Nei titoli e nei testi troverete qualche rimando cinematografico (ebbene si, sono un cinefilo). Se non vi interessano fate finta di non vederli, già che non sono fondamentali per la comprensione dei post...
Di questo blog ho mandato avanti, fino a Settembre 2018, anche una versione in Spagnolo. Potete trovarla su El arte de la programación en C. Buona lettura.
sabato 21 dicembre 2024
lunedì 16 dicembre 2024
Thread Runner come usare i thread in C - pt.2
Eldon Tyrell: Quale sarebbe il tuo problema?
Roy Batty: La morte.
Eldon Tyrell: La morte... beh questo temo sia un po' fuori della mia giurisdizione.
Roy Batty: Io voglio più vita, padre!
(...una premessa: questo post è un remake di un mio vecchio post (parte 2 di 3). Ma, anche se tratta lo stesso argomento, amplia e perfeziona un po' il discorso è mi è sembrato il caso di riproporlo. Leggete e mi direte...)
...ti spiego: io sono un POSIX thread e tu un C11 thread... |
Ma torniamo al nostro amato C: il committee ISO del C non poteva rimanere indietro, e quindi hanno pensato di fare la stessa cosa con il C11, ovvero far si che i thread siano direttamente una parte del C... e ci sono riusciti? Prima di rispondere vi anticipo una considerazione: ho una stima del committee ISO del C maggiore di quella che ho di quello del del C++ (e non ci voleva molto...), ma in questo caso devo proprio dire che non ci siamo: a seguire vedremo il perché.
Come sono stati pensati i nuovi C11 Threads? Allora, hanno preso tutte le funzioni e variabili che compongono i POSIX Threads e gli hanno cambiato il nome (e devo ammettere che quelli nuovi sono più semplici e immediati); inoltre, in alcuni casi (pochi, per fortuna), hanno cambiato i tipi dei codici di ritorno e degli argomenti delle funzioni. Punto. Geniale? Non proprio direi, e niente a che vedere con la soluzione brillante usata nel C++11. Motivi per usare questa nuova versione? Zero, direi, e non vi ho ancora esposto l problemi principali...
Comunque, per farla breve, ho riscritto l'esempio dello scorso post usando i C11 Threads. Vai col codice!
#include <stdio.h>#include <string.h>#include <unistd.h>#include <stdbool.h>#include <threads.h>#define NUMTHREADS 2 // numero di thread da trattare, in questo caso 2// struttura per i dati condivisi dai threadtypedef struct {mtx_t mutex; // mutex di sincronizzazione comune ai threadbool stop; // flag per stop threadunsigned long cnt; // counter condiviso dai thread} Thdata;// prototipi localiint threadFunc(void *arg);// funzione main()int main(int argc, char* argv[]){char errmsg_buf[256]; // buffer per strerror_r(3)// init dei dati condivisi dai threadThdata thdata;thdata.stop = false;thdata.cnt = 0;// init del mutex di Thdataint error;if ((error = mtx_init(&thdata.mutex, mtx_plain)) != thrd_success) {// errore!printf("%s: non posso creare il mutex (%d)\n", argv[0], error);return 1;}// loop di avvio dei threadthrd_t tid[2];for (int i = 0; i < NUMTHREADS; i++) {// creo un threadif ((error = thrd_create(&tid[i], threadFunc, &thdata)) != thrd_success) {// errore!printf("%s: non posso creare il thread %d (%d)\n", argv[0], i, error);return 1;}}// dopo 10 secondi fermo tutti i threadsleep(10);thdata.stop = true;// loop di join dei threadfor (int i = 0; i < NUMTHREADS; i++) {if ((error = thrd_join(tid[i], NULL)) != thrd_success) {// errore!printf("%s: non posso attendere il thread %d (%d)\n", argv[0], i, error);return 1;}}// cancello il mutex di sincronizzazionemtx_destroy(&thdata.mutex);// escoprintf("%s: thread terminati: thdata.cnt=%lu\n", argv[0], thdata.cnt);return 0;}// threadFunc() - funzione per i threadint threadFunc(void *arg){// ottengo i dati del thread con un cast (Thdata *) di (void *) argThdata *thdata = (Thdata *)arg;// loop del threadprintf("thread partito\n");unsigned long loc_cnt = 0;for (;;) {// lock del mutexmtx_lock(&thdata->mutex);// incremento il counter locale e quello condivisoloc_cnt++;thdata->cnt++;// unlock del mutexmtx_unlock(&thdata->mutex);// test dello stop flagif (thdata->stop) {// il thread esceprintf("thread terminato dal main: loc_cnt=%lu\n", loc_cnt);thrd_exit(0);}// sleep del thread (uso usleep solo per comodità invece della nanosleep(2))usleep(1000);}// il thread esce per altro motivo diverso dallo stop flagprintf("thread terminato localmente: loc_cnt=%lu\n", loc_cnt);thrd_exit(0);}
Come vedete il codice nuovo è praticamente identico al vecchio, mi sono limitato a usare le nuove funzioni al posto di quelle vecchie (per esempio thrd_create() invece di pthread_create(3)) , ho usato i nuovi tipi (per esempio mtx_t invece di pthread_mutex_t) e ho leggermente modificato il test dei valori di ritorno: poche differenze, devo dire, e, in alcuni casi, in peggio: ad esempio è sparito il parametro attr di pthread_create(3), che (per semplicità) nello scorso esempio avevo lasciato a NULL, ma che a volte può risultare utile (leggere il manuale della funzione per rendersene conto). Comunque si potrebbe dire (senza fare troppo gli schizzinosi) che la nuova interfaccia non ci offre nessun vantaggio sostanziale, ma neanche un peggioramento decisivo, quindi si potrebbe anche usare (de gustibus).
Ma c'è un problema: pare che i C11 Threads non siano stati considerati una priorità per chi scrive i compilatori e le varie libc, quindi per molti anni dall'introduzione è stato difficile compilare/eseguire un programma come quello che ho mostrato. Perfino il nostro amato GCC (che di solito è il primo a fornire supporto per le ultime novità) ci ha messo una vita a supportare i nuovi thread (in realtà a causa della mancata integrazione nella glibc). Questo è un brutto segno, per niente incoraggiante: se il mondo che gira intorno ai linguaggi e ai Sistemi Operativi non approva rapidamente vuol dire che ci sono delle perplessità.
Comunque dopo tanto tempo qualcosa è arrivato, e adesso è possibile compilare i C11 Threads su Linux (con versioni recenti di GCC e con una glibc dalla v.2.28 in su), sui vari sistemi BSD, su Windows, ma non ancora su macOS (se non mi sbaglio). Aggiungo un dettaglio illuminante sulla coppia GCC/glibc: in attesa di un supporto veramente nativo, la compilazione era basata su una implementazione che era, praticamente, un wrapper che simulava i C11 Threads usando i POSIX Threads (infatti bisognava linkare la libpthread): un po' assurdo, no?
E anche se ora si può usare direttamente GCC con una glibc recente (la v.2.28 è del 2018), io ho deciso di seguire con la stessa soluzione che è stata disponibile quasi da subito (già nel 2014) e ho già usato in passato: ho usato la musl libc che è una alternativa alla glibc, ed è dotata di un wrapper per GCC (musl-gcc). musl fornisce (su Linux) il supporto completo al C11, ed è stata (credo) la prima in assoluto a fornire il supporto nativo ai C11 Thread. E come funziona tutto questo? Una volta compilato il programma si comporta correttamente, come potete vedere qui sotto:
aldo@Linux $ musl-gcc c11threads.c -o c11threadsaldo@Linux $ ./c11threadsthread partitothread partitothread terminato dal main: loc_cnt=8597thread terminato dal main: loc_cnt=8597./c11threads: thread terminati: thdata.cnt=17194
Ma il gioco vale la candela? No, per quel che mi riguarda continuerò ad usare i POSIX Threads, che uso da molti anni e rimangono il riferimento d'eccellenza. I C11 Threads sono uno strano ibrido che non rispetta la classica sintassi della glibc (ad esempio in caso di errore non ritorna -1 ma un codice di errore e, oltretutto, non aggiorna errno), e non rispetta pienamente neanche lo standard POSIX (i codici di errore non sono inseriti in una lista analizzabile con strerror(3)). E poi, come già detto sopra, le nuove funzioni sono equivalenti alle vecchie ma non abbastanza, alcune funzionalità non sono più disponibili e alcune cose semplici (come stampare il thread Id interno della libreria o le stringhe degli errori) sono molto più complicate... e, dulcis in fundo, nelle classiche pagine dei manuali Linux e POSIX non c'è ancora traccia delle nuove funzioni. Son problemi, no?
Ok, per oggi chiudo qui. Nel prossimo e ultimo episodio di Thread Runner vedremo un esempio pratico. Rimanete sintonizzati, eh!
Ciao e al prossimo post!