1*4882a593Smuzhiyun.. SPDX-License-Identifier: GPL-2.0 2*4882a593Smuzhiyun 3*4882a593Smuzhiyun.. include:: ../disclaimer-ita.rst 4*4882a593Smuzhiyun 5*4882a593Smuzhiyun:Original: :ref:`Documentation/process/deprecated.rst <deprecated>` 6*4882a593Smuzhiyun:Translator: Federico Vaga <federico.vaga@vaga.pv.it> 7*4882a593Smuzhiyun 8*4882a593Smuzhiyun.. _it_deprecated: 9*4882a593Smuzhiyun 10*4882a593Smuzhiyun============================================================================== 11*4882a593SmuzhiyunInterfacce deprecate, caratteristiche del linguaggio, attributi, e convenzioni 12*4882a593Smuzhiyun============================================================================== 13*4882a593Smuzhiyun 14*4882a593SmuzhiyunIn un mondo perfetto, sarebbe possibile prendere tutti gli usi di 15*4882a593Smuzhiyunun'interfaccia deprecata e convertirli in quella nuova, e così sarebbe 16*4882a593Smuzhiyunpossibile rimuovere la vecchia interfaccia in un singolo ciclo di sviluppo. 17*4882a593SmuzhiyunTuttavia, per via delle dimensioni del kernel, la gerarchia dei manutentori e 18*4882a593Smuzhiyunle tempistiche, non è sempre possibile fare questo tipo di conversione tutta 19*4882a593Smuzhiyunin una volta. Questo significa che nuove istanze di una vecchia interfaccia 20*4882a593Smuzhiyunpotrebbero aggiungersi al kernel proprio quando si sta cercando di rimuoverle, 21*4882a593Smuzhiyunaumentando così il carico di lavoro. Al fine di istruire gli sviluppatori su 22*4882a593Smuzhiyuncosa è considerato deprecato (e perché), è stata create la seguente lista a cui 23*4882a593Smuzhiyunfare riferimento quando qualcuno propone modifiche che usano cose deprecate. 24*4882a593Smuzhiyun 25*4882a593Smuzhiyun__deprecated 26*4882a593Smuzhiyun------------ 27*4882a593SmuzhiyunNonostante questo attributo marchi visibilmente un interfaccia come deprecata, 28*4882a593Smuzhiyun`non produce più alcun avviso durante la compilazione 29*4882a593Smuzhiyun<https://git.kernel.org/linus/771c035372a036f83353eef46dbb829780330234>`_ 30*4882a593Smuzhiyunperché uno degli obiettivi del kernel è quello di compilare senza avvisi; 31*4882a593Smuzhiyuninoltre, nessuno stava agendo per rimuovere queste interfacce. Nonostante l'uso 32*4882a593Smuzhiyundi `__deprecated` in un file d'intestazione sia opportuno per segnare una 33*4882a593Smuzhiyuninterfaccia come 'vecchia', questa non è una soluzione completa. L'interfaccia 34*4882a593Smuzhiyundeve essere rimossa dal kernel, o aggiunta a questo documento per scoraggiarne 35*4882a593Smuzhiyunl'uso. 36*4882a593Smuzhiyun 37*4882a593SmuzhiyunBUG() e BUG_ON() 38*4882a593Smuzhiyun---------------- 39*4882a593SmuzhiyunAl loro posto usate WARN() e WARN_ON() per gestire le 40*4882a593Smuzhiyuncondizioni "impossibili" e gestitele come se fosse possibile farlo. 41*4882a593SmuzhiyunNonostante le funzioni della famiglia BUG() siano state progettate 42*4882a593Smuzhiyunper asserire "situazioni impossibili" e interrompere in sicurezza un 43*4882a593Smuzhiyunthread del kernel, queste si sono rivelate essere troppo rischiose 44*4882a593Smuzhiyun(per esempio, in quale ordine rilasciare i *lock*? Ci sono stati che 45*4882a593Smuzhiyunsono stati ripristinati?). Molto spesso l'uso di BUG() 46*4882a593Smuzhiyundestabilizza il sistema o lo corrompe del tutto, il che rende 47*4882a593Smuzhiyunimpossibile un'attività di debug o anche solo leggere un rapporto 48*4882a593Smuzhiyuncirca l'errore. Linus ha un'opinione molto critica al riguardo: 49*4882a593Smuzhiyun`email 1 50*4882a593Smuzhiyun<https://lore.kernel.org/lkml/CA+55aFy6jNLsywVYdGp83AMrXBo_P-pkjkphPGrO=82SPKCpLQ@mail.gmail.com/>`_, 51*4882a593Smuzhiyun`email 2 52*4882a593Smuzhiyun<https://lore.kernel.org/lkml/CAHk-=whDHsbK3HTOpTF=ue_o04onRwTEaK_ZoJp_fjbqq4+=Jw@mail.gmail.com/>`_ 53*4882a593Smuzhiyun 54*4882a593SmuzhiyunTenete presente che la famiglia di funzioni WARN() dovrebbe essere 55*4882a593Smuzhiyunusato solo per situazioni che si suppone siano "impossibili". Se 56*4882a593Smuzhiyunvolete avvisare gli utenti riguardo a qualcosa di possibile anche se 57*4882a593Smuzhiyunindesiderato, usare le funzioni della famiglia pr_warn(). Chi 58*4882a593Smuzhiyunamministra il sistema potrebbe aver attivato l'opzione sysctl 59*4882a593Smuzhiyun*panic_on_warn* per essere sicuri che il sistema smetta di funzionare 60*4882a593Smuzhiyunin caso si verifichino delle condizioni "inaspettate". (per esempio, 61*4882a593Smuzhiyundate un'occhiata al questo `commit 62*4882a593Smuzhiyun<https://git.kernel.org/linus/d4689846881d160a4d12a514e991a740bcb5d65a>`_) 63*4882a593Smuzhiyun 64*4882a593SmuzhiyunCalcoli codificati negli argomenti di un allocatore 65*4882a593Smuzhiyun---------------------------------------------------- 66*4882a593SmuzhiyunIl calcolo dinamico delle dimensioni (specialmente le moltiplicazioni) non 67*4882a593Smuzhiyundovrebbero essere fatto negli argomenti di funzioni di allocazione di memoria 68*4882a593Smuzhiyun(o simili) per via del rischio di overflow. Questo può portare a valori più 69*4882a593Smuzhiyunpiccoli di quelli che il chiamante si aspettava. L'uso di questo modo di 70*4882a593Smuzhiyunallocare può portare ad un overflow della memoria di heap e altri 71*4882a593Smuzhiyunmalfunzionamenti. (Si fa eccezione per valori numerici per i quali il 72*4882a593Smuzhiyuncompilatore può generare avvisi circa un potenziale overflow. Tuttavia usare 73*4882a593Smuzhiyuni valori numerici come suggerito di seguito è innocuo). 74*4882a593Smuzhiyun 75*4882a593SmuzhiyunPer esempio, non usate ``count * size`` come argomento:: 76*4882a593Smuzhiyun 77*4882a593Smuzhiyun foo = kmalloc(count * size, GFP_KERNEL); 78*4882a593Smuzhiyun 79*4882a593SmuzhiyunAl suo posto, si dovrebbe usare l'allocatore a due argomenti:: 80*4882a593Smuzhiyun 81*4882a593Smuzhiyun foo = kmalloc_array(count, size, GFP_KERNEL); 82*4882a593Smuzhiyun 83*4882a593SmuzhiyunSe questo tipo di allocatore non è disponibile, allora dovrebbero essere usate 84*4882a593Smuzhiyunle funzioni del tipo *saturate-on-overflow*:: 85*4882a593Smuzhiyun 86*4882a593Smuzhiyun bar = vmalloc(array_size(count, size)); 87*4882a593Smuzhiyun 88*4882a593SmuzhiyunUn altro tipico caso da evitare è quello di calcolare la dimensione di una 89*4882a593Smuzhiyunstruttura seguita da un vettore di altre strutture, come nel seguente caso:: 90*4882a593Smuzhiyun 91*4882a593Smuzhiyun header = kzalloc(sizeof(*header) + count * sizeof(*header->item), 92*4882a593Smuzhiyun GFP_KERNEL); 93*4882a593Smuzhiyun 94*4882a593SmuzhiyunInvece, usate la seguente funzione:: 95*4882a593Smuzhiyun 96*4882a593Smuzhiyun header = kzalloc(struct_size(header, item, count), GFP_KERNEL); 97*4882a593Smuzhiyun 98*4882a593SmuzhiyunPer maggiori dettagli fate riferimento a array_size(), 99*4882a593Smuzhiyunarray3_size(), e struct_size(), così come la famiglia di 100*4882a593Smuzhiyunfunzioni check_add_overflow() e check_mul_overflow(). 101*4882a593Smuzhiyun 102*4882a593Smuzhiyunsimple_strtol(), simple_strtoll(), simple_strtoul(), simple_strtoull() 103*4882a593Smuzhiyun---------------------------------------------------------------------- 104*4882a593SmuzhiyunLe funzioni simple_strtol(), simple_strtoll(), 105*4882a593Smuzhiyunsimple_strtoul(), e simple_strtoull() ignorano volutamente 106*4882a593Smuzhiyuni possibili overflow, e questo può portare il chiamante a generare risultati 107*4882a593Smuzhiyuninaspettati. Le rispettive funzioni kstrtol(), kstrtoll(), 108*4882a593Smuzhiyunkstrtoul(), e kstrtoull() sono da considerarsi le corrette 109*4882a593Smuzhiyunsostitute; tuttavia va notato che queste richiedono che la stringa sia 110*4882a593Smuzhiyunterminata con il carattere NUL o quello di nuova riga. 111*4882a593Smuzhiyun 112*4882a593Smuzhiyunstrcpy() 113*4882a593Smuzhiyun-------- 114*4882a593SmuzhiyunLa funzione strcpy() non fa controlli agli estremi del buffer 115*4882a593Smuzhiyundi destinazione. Questo può portare ad un overflow oltre i limiti del 116*4882a593Smuzhiyunbuffer e generare svariati tipi di malfunzionamenti. Nonostante l'opzione 117*4882a593Smuzhiyun`CONFIG_FORTIFY_SOURCE=y` e svariate opzioni del compilatore aiutano 118*4882a593Smuzhiyuna ridurne il rischio, non c'è alcuna buona ragione per continuare ad usare 119*4882a593Smuzhiyunquesta funzione. La versione sicura da usare è strscpy(). 120*4882a593Smuzhiyun 121*4882a593Smuzhiyunstrncpy() su stringe terminate con NUL 122*4882a593Smuzhiyun-------------------------------------- 123*4882a593SmuzhiyunL'utilizzo di strncpy() non fornisce alcuna garanzia sul fatto che 124*4882a593Smuzhiyunil buffer di destinazione verrà terminato con il carattere NUL. Questo 125*4882a593Smuzhiyunpotrebbe portare a diversi overflow di lettura o altri malfunzionamenti 126*4882a593Smuzhiyuncausati, appunto, dalla mancanza del terminatore. Questa estende la 127*4882a593Smuzhiyunterminazione nel buffer di destinazione quando la stringa d'origine è più 128*4882a593Smuzhiyuncorta; questo potrebbe portare ad una penalizzazione delle prestazioni per 129*4882a593Smuzhiyunchi usa solo stringe terminate. La versione sicura da usare è 130*4882a593Smuzhiyunstrscpy(). (chi usa strscpy() e necessita di estendere la 131*4882a593Smuzhiyunterminazione con NUL deve aggiungere una chiamata a memset()) 132*4882a593Smuzhiyun 133*4882a593SmuzhiyunSe il chiamate no usa stringhe terminate con NUL, allore strncpy() 134*4882a593Smuzhiyunpuò continuare ad essere usata, ma i buffer di destinazione devono essere 135*4882a593Smuzhiyunmarchiati con l'attributo `__nonstring <https://gcc.gnu.org/onlinedocs/gcc/Common-Variable-Attributes.html>`_ 136*4882a593Smuzhiyunper evitare avvisi durante la compilazione. 137*4882a593Smuzhiyun 138*4882a593Smuzhiyunstrlcpy() 139*4882a593Smuzhiyun--------- 140*4882a593SmuzhiyunLa funzione strlcpy(), per prima cosa, legge interamente il buffer di 141*4882a593Smuzhiyunorigine, magari leggendo più di quanto verrà effettivamente copiato. Questo 142*4882a593Smuzhiyunè inefficiente e può portare a overflow di lettura quando la stringa non è 143*4882a593Smuzhiyunterminata con NUL. La versione sicura da usare è strscpy(). 144*4882a593Smuzhiyun 145*4882a593SmuzhiyunSegnaposto %p nella stringa di formato 146*4882a593Smuzhiyun-------------------------------------- 147*4882a593Smuzhiyun 148*4882a593SmuzhiyunTradizionalmente, l'uso del segnaposto "%p" nella stringa di formato 149*4882a593Smuzhiyunesponne un indirizzo di memoria in dmesg, proc, sysfs, eccetera. Per 150*4882a593Smuzhiyunevitare che questi indirizzi vengano sfruttati da malintenzionati, 151*4882a593Smuzhiyuntutto gli usi di "%p" nel kernel rappresentano l'hash dell'indirizzo, 152*4882a593Smuzhiyunrendendolo di fatto inutilizzabile. Nuovi usi di "%p" non dovrebbero 153*4882a593Smuzhiyunessere aggiunti al kernel. Per una rappresentazione testuale di un 154*4882a593Smuzhiyunindirizzo usate "%pS", l'output è migliore perché mostrerà il nome del 155*4882a593Smuzhiyunsimbolo. Per tutto il resto, semplicemente non usate "%p". 156*4882a593Smuzhiyun 157*4882a593SmuzhiyunParafrasando la `guida 158*4882a593Smuzhiyun<https://lore.kernel.org/lkml/CA+55aFwQEd_d40g4mUCSsVRZzrFPUJt74vc6PPpb675hYNXcKw@mail.gmail.com/>`_ 159*4882a593Smuzhiyundi Linus: 160*4882a593Smuzhiyun 161*4882a593Smuzhiyun- Se il valore hash di "%p" è inutile, chiediti se il puntatore stesso 162*4882a593Smuzhiyun è importante. Forse dovrebbe essere rimosso del tutto? 163*4882a593Smuzhiyun- Se credi davvero che il vero valore del puntatore sia importante, 164*4882a593Smuzhiyun perché alcuni stati del sistema o i livelli di privilegi di un 165*4882a593Smuzhiyun utente sono considerati "special"? Se pensi di poterlo giustificare 166*4882a593Smuzhiyun (in un commento e nel messaggio del commit) abbastanza bene da 167*4882a593Smuzhiyun affrontare il giudizio di Linus, allora forse potrai usare "%px", 168*4882a593Smuzhiyun assicurandosi anche di averne il permesso. 169*4882a593Smuzhiyun 170*4882a593SmuzhiyunInfine, sappi che un cambio in favore di "%p" con hash `non verrà 171*4882a593Smuzhiyunaccettato 172*4882a593Smuzhiyun<https://lore.kernel.org/lkml/CA+55aFwieC1-nAs+NFq9RTwaR8ef9hWa4MjNBWL41F-8wM49eA@mail.gmail.com/>`_. 173*4882a593Smuzhiyun 174*4882a593SmuzhiyunVettori a dimensione variabile (VLA) 175*4882a593Smuzhiyun------------------------------------ 176*4882a593Smuzhiyun 177*4882a593SmuzhiyunUsare VLA sullo stack produce codice molto peggiore rispetto a quando si usano 178*4882a593Smuzhiyunvettori a dimensione fissa. Questi `problemi di prestazioni <https://git.kernel.org/linus/02361bc77888>`_, 179*4882a593Smuzhiyuntutt'altro che banali, sono già un motivo valido per eliminare i VLA; in 180*4882a593Smuzhiyunaggiunta sono anche un problema per la sicurezza. La crescita dinamica di un 181*4882a593Smuzhiyunvettore nello stack potrebbe eccedere la memoria rimanente in tale segmento. 182*4882a593SmuzhiyunQuesto può portare a dei malfunzionamenti, potrebbe sovrascrivere 183*4882a593Smuzhiyundati importanti alla fine dello stack (quando il kernel è compilato senza 184*4882a593Smuzhiyun`CONFIG_THREAD_INFO_IN_TASK=y`), o sovrascrivere un pezzo di memoria adiacente 185*4882a593Smuzhiyunallo stack (quando il kernel è compilato senza `CONFIG_VMAP_STACK=y`). 186*4882a593Smuzhiyun 187*4882a593SmuzhiyunSalto implicito nell'istruzione switch-case 188*4882a593Smuzhiyun------------------------------------------- 189*4882a593Smuzhiyun 190*4882a593SmuzhiyunIl linguaggio C permette ai casi di un'istruzione `switch` di saltare al 191*4882a593Smuzhiyunprossimo caso quando l'istruzione "break" viene omessa alla fine del caso 192*4882a593Smuzhiyuncorrente. Tuttavia questo rende il codice ambiguo perché non è sempre ovvio se 193*4882a593Smuzhiyunl'istruzione "break" viene omessa intenzionalmente o è un baco. Per esempio, 194*4882a593Smuzhiyunosservando il seguente pezzo di codice non è chiaro se lo stato 195*4882a593Smuzhiyun`STATE_ONE` è stato progettato apposta per eseguire anche `STATE_TWO`:: 196*4882a593Smuzhiyun 197*4882a593Smuzhiyun switch (value) { 198*4882a593Smuzhiyun case STATE_ONE: 199*4882a593Smuzhiyun do_something(); 200*4882a593Smuzhiyun case STATE_TWO: 201*4882a593Smuzhiyun do_other(); 202*4882a593Smuzhiyun break; 203*4882a593Smuzhiyun default: 204*4882a593Smuzhiyun WARN("unknown state"); 205*4882a593Smuzhiyun } 206*4882a593Smuzhiyun 207*4882a593SmuzhiyunDato che c'è stata una lunga lista di problemi `dovuti alla mancanza dell'istruzione 208*4882a593Smuzhiyun"break" <https://cwe.mitre.org/data/definitions/484.html>`_, oggigiorno non 209*4882a593Smuzhiyunpermettiamo più che vi sia un "salto implicito" (*fall-through*). Per 210*4882a593Smuzhiyunidentificare un salto implicito intenzionale abbiamo adottato la pseudo 211*4882a593Smuzhiyunparola chiave 'fallthrough' che viene espansa nell'estensione di gcc 212*4882a593Smuzhiyun`__attribute__((fallthrough))` `Statement Attributes 213*4882a593Smuzhiyun<https://gcc.gnu.org/onlinedocs/gcc/Statement-Attributes.html>`_. 214*4882a593Smuzhiyun(Quando la sintassi C17/C18 `[[fallthrough]]` sarà più comunemente 215*4882a593Smuzhiyunsupportata dai compilatori C, analizzatori statici, e dagli IDE, 216*4882a593Smuzhiyunallora potremo usare quella sintassi per la pseudo parola chiave) 217*4882a593Smuzhiyun 218*4882a593SmuzhiyunQuando la sintassi [[fallthrough]] sarà più comunemente supportata dai 219*4882a593Smuzhiyuncompilatori, analizzatori statici, e ambienti di sviluppo IDE, 220*4882a593Smuzhiyunallora potremo usarla anche noi. 221*4882a593Smuzhiyun 222*4882a593SmuzhiyunNe consegue che tutti i blocchi switch/case devono finire in uno dei seguenti 223*4882a593Smuzhiyunmodi: 224*4882a593Smuzhiyun 225*4882a593Smuzhiyun* ``break;`` 226*4882a593Smuzhiyun* `fallthrough;`` 227*4882a593Smuzhiyun* ``continue;`` 228*4882a593Smuzhiyun* ``goto <label>;`` 229*4882a593Smuzhiyun* ``return [expression];`` 230