Synkronointi (ohjelmointi)
Synkronointi (engl. synchronization) tietokoneen ohjelmoinnissa tarkoittaa hallittua synkronointimekanismia rinnakkaisten ohjelmien välisen jaetun tiedon käsittelyyn.[1]
Synkronointia tarvitaan, jotta kahden rinnakkain samanaikaisesti tapahtuvan muutoksen välillä ei synny kilpatilannetta (engl. race condition), jolloin ohjelmat voivat päivittää yhtä aikaa samaa muuttujaa tai tietorakennetta, mikä yleensä johtaa virheeseen. Tämän takia käyttöjärjestelmä tarjoaa mekanismeja resurssien hallittuun synkronointiin (engl. synchronization).
Kilpatilanne voi syntyä myös yhden suorittimen järjestelmässä riippuen säikeiden ja prosessien vuoronnuksesta (context switch) sekä laitteistokeskeytyksien käsittelystä.[2][3] Synkronoinnilla pyritään välttämään kilpatilanteita, mutta virheellinen synkronointi voi aiheuttaa nälkiintymisen tai lukkiutumisen. Ohjelmakoodin uudelleen käytettävyyttä voi haitata synkronoinnin ja muun ohjelmakoodin välinen keskinäinen riippuvuussuhde.[4]
Nykyisin käytettävät lukottomat synkronointimekanismit välttävät suuren osan aiemmin tunnetuista ongelmista. Lukottomia menetelmiä ovat atomiset operaatiot sekä Read-Copy-Update (RCU) mekanismi.[5][6]
Taustaa
[muokkaa | muokkaa wikitekstiä]Rinnakkaisuuden käsittely on perustavanlaatuinen ongelma, jossa objektien tilan on pysyttävä aina eheänä ja pääsy objekteihin on synkronoitava. Rinnakkaisessa käsittelyssä ei toimita enää sekventiaalisessa ohjelmoinnissa ja sarjamuotoisista ohjelmointimenetelmistä ei ole apua.[7] Menetelmiä synkronointiin on muun muassa lukituspohjainen synkronointi, joka tunnetaan myös poissulkemisena.[7] Atomisuus (linearisointi) on tärkeä konsepti yhtäaikaisesti ja hajautetusti toimiville ohjelmille (erona sarjamuotoisesta eli serialisoinnista).[7] Lukottomuudella tarkoitetaan menetelmiä, joissa kriittistä aluetta ei voi käyttää (mutex-free).[7]
Synkronointimekanismit
[muokkaa | muokkaa wikitekstiä]Synkronointimekanismit tai synkronointiprimitiivit[8][9][10][11][12][13][14][15][16][17][18] ovat sopimuspohjaisia siten, että ohjelmakoodia tehtäessä on tunnistettava mitä muuttujia ei saa muokata lukituksen ollessa voimassa. Lukituksen käsittely tarvitaan sekä lukevalle että kirjoittavalle puolelle jotta molemmilla on käsitys milloin lukitus on voimassa.
Synkronointiin käytettäviä mekanismeja ovat spinlock, barrier, semaforit, (engl. semaphore) poissulkeminen (engl. mutual exclusion) ja kriittinen alue (engl. critical section, critical region), joilla varmistetaan, että vain yksi säie tai prosessi voi kerrallaan käsitellä yhteisiä muuttujia.
Eri mekanismeilla voi olla suorituskyvyn kannalta merkitystä lukituksen haastamisen (engl. contention) kannalta.[19]
Spinlock
[muokkaa | muokkaa wikitekstiä]Spinlock nimensä mukaisesti tarkastelee lukitusmuuttujaa kunnes se voi varata lukituksen (aktiivinen odotus). Tästä voi aiheutua busy-wait tilanne, mutta menetelmää yhä käytetään sen yksinkertaisuuden vuoksi.[20] Ratkaisu riippuu arkkitehtuurikohtaisesta atomisesta tarkista-ja-aseta -toiminnosta.[21] Tietokoneen suoritin voi tukea toimintoa ilman että laitteisto muuttaa tilaa kesken toiminnon.[22] Sekventiaaliset lukitukset (Seqlocks) ovat spinlockien variaatio, joka huomioi erikseen kirjoituksen ja lukituksen antaen kirjoitukselle korkeamman prioriteetin.[23] Lippupohjaiselle (ticket) ratkaisulle on vaihtoehtona kehitetty jonopohjainen (queue) suurissa NUMA-järjestelmissä käytettäväksi.[24]
Barrier
[muokkaa | muokkaa wikitekstiä]Barrier on esto, joka rajaa mm. kääntäjän tai suorittimen tekemän uudelleenjärjestelyn[25] varmistaen että tietty ohjelmakoodi suoritetaan sitä ennen.[25][26] Tähän voidaan liittää myös välimuistin tyhjentäminen keskusmuistiin (ks. välimuistin yhtenäisyys) ja muita toimintoja. Etenkin moniprosessointi voi vaatia mekanismia.[27]
Semafori
[muokkaa | muokkaa wikitekstiä]Semafori on kahden "suunnan" menettely, jossa lasketaan lukituksen ottaneet ja lukituksen vapauttaneet: kun lukituksen ottaneita on enemmän kuin lukituksen vapauttaneita muutoksia ei saa tehdä synkronoitavaan tietoon.[26] Linuxin toteutus laskee sekä lukituksen ottaneet että odottajat.[26][23]
Luku-kirjoitus-semaforit sallivat jaetun lukemisen, mutta vaativat yksityisen pääsyn kirjoittamista varten: menetelmä parantaa rinnakkaisuutta ja järjestelmän kokonaissuorituskykyä.[28] Semaforipohjainen ratkaisu on suunnattu "luku-kirjoitus-ongelman" (engl. reader-writer-problem) ratkaisemiseen.[29]
Tyypillinen semaforimekanismi ei ole sopiva reaaliaikaisten järjestelmien toteutukseen johtuen prioriteetti-inversion mahdollisuudesta.[30]
Edsger Dijkstra mainitsee alankomaalaisen fyysikon C. S. Scholtenin osoittaneen semaforien soveltamista.[31]
Turnstile
[muokkaa | muokkaa wikitekstiä]Turnstile tai kääntöportti on Solaris-käyttöjärjestelmän toteuttama synkronointimenetelmä.[32] Toteutus sisältää odotusjonot ja prioriteetin perintämekanismin.[32]
Fence
[muokkaa | muokkaa wikitekstiä]Fence (aita) on käytössä muun muassa grafiikkaprosessorin (GPU) ja suorittimen (CPU) välisessä synkronoinnissa.[33] Fence signaloidaan kun käsittely on valmis (GPU), jolla välin suoritin (CPU) voi tehdä työtä etukäteen.[34] Periaatteena on käsky, joka estää ohjelmointikielen kääntäjää ja laitteistoa järjestelemästä muistikäsittelyjä uudelleen: 1) fence valmistuu kun, sitä ennen annetut käsittelyt ovat valmiita; 2) fence on oltava valmis ennen kuin sen jälkeiset käsittelyt ovat näkyvissä muille prosessoreille.[35]
Poissulkeminen
[muokkaa | muokkaa wikitekstiä]Poissulkeminen on yleinen prosessien välillä käytetty synkronointi. Monet menetelmät toimivat vain säikeiden välillä nopeusedun vuoksi, mutta yleiset tietokoneresurssit kuten sarjaportti voivat vaatia varmistuksen että muiden prosessien säikeet eivät myöskään pääse käsittelemään samaa resurssia samaan aikaan.[2] Linux toteuttaa mutex tyypin lisäksi futex poissulkemisen, joka on nopeampi mutta kernelin sisäisiin toimintoihin rajoitettu.[26] Poissulkemisella voidaan tarkoittaa myös yleistä termiä akateemisissa ja teoreettisissa teoksissa.[36]
Lukkiutumisen välttämiseen on kehitetty vaihtoehtona muun muassa wait/wound mutex (ww-mutex): tämä ratkaisee päinvastaisessa järjestyksessä varauksen "vahingoittamalla" myöhemmin tullutta vapauttamaan omansa, jotta ensin ehtinyt voi jatkaa.[37][38] Perinteisesti vastakkaisia järjestyksiä on vain vältettävä.[37] Wait/wound-mekanismia käytetään grafiikkaprosessorien kanssa (GPU), joissa on useita puskureita joiden järjestystä ei voi määrittää etukäteen: nämä puskurit voivat olla vuoroin GPU:n, laiteajurin, käyttäjäavaruuden tai mahdollisesti toisen ajurin käytössä (videokuvan kaappaus).[37] Relaatiotietokantoja käsittelevässä kirjallisuudessa transaktioiden käsittelyssä vastaavasta lukituksesta käytetään nimeä wait-die.[38]
Leslie Lamport on tutkinut ja todennut poissulkemisessa olevan useita ongelmia.[39] Lamport myös esitti ensimmäisen hajautettuihin järjestelmiin sopivan ratkaisun poissulkemisongelmaan.[39] Lamportin mukaan hajautetuissa järjestelmissä ja moniprosessointia käyttävissä järjestelmissä on samanlaisia ongelmia, joissa tapahtumien järjestys ei ole ennakoitavissa.[40] Petersonin algoritmia sanotaan standardinomaiseksi alhaisen tason algoritmiksi kahden säikeen väliseen poissulkemiseen.[41]
Poissulkeminen voi olla verrattain raskas ja hidas operaatio ja ohjelman sisäiseen synkronointiin käytetään tyypillisesti muita menetelmiä kuten kriittisiä alueita.[42][43]
Kiireinen odotus (engl. busy wait) -tekniikoita käytetään paljon poissulkemiseen, mutta tyypilliset toteutukset vaativat paljon muistia sekä kytkentäverkon kilpatilanteita, jotka aiheuttavat pullonkauloja suorituskyvylle.[44]
Kriittinen alue
[muokkaa | muokkaa wikitekstiä]Kriittinen alue (engl. Critical Section) on ohjelmalohko, joka on suoritettava sen loppuun asti ennen luovuttamista toiselle suorittajalle.[23][45] Termiä käytetään synkronointiprimitiivin (Critical Section)[45] lisäksi myös synkronoinnilla suojatusta ohjelmalohkosta.[46] Synkronointiprimitiivi Critical Section Windowsissa eroaa poissulkemisesta muun muassa siten, että se toimii vain saman prosessin säikeiden välillä eikä voida jakaa eri prosessien kesken.[45]
Prosessin sisäisen suorituksen synkronointiin kriittinen alue voi olla tehokkaampi vaihtoehto kuin poissulkeva mutex.[43][47]
Atomisuus
[muokkaa | muokkaa wikitekstiä]Atomisilla operaatioilla voidaan tehdä ohjelmointikielen (ja laitteiston) tukemana automaattisesti suojattuja (synkronoituja) tai lukottomia muuttujia, joilla voidaan välttää kilpatilanteita.[26][48][2]
Suorittimet sisältävät laitteistotasolla erilaisia atomisia primitiivejä, joita voidaan käyttää "rakennuspalikoina" muille synkronointimuunnelmille.[49] Laitteiston tukemia primitiivejä ovat esimerkiksi: test and set, compare and swap, fetch and increment ja load-link / store-conditional.[49]
Muun muassa C++11 lisää ohjelmointikieleen tuen atomisille operaatioille (ISO/IEC 14882:2011). Atomisilla operaatioilla voidaan toteuttaa korkeamman tason synkronointitoimintoja.[50]
Atomiset operaatiot voivat parantaa suorituskykyä huomattavasti verrattuna perinteisiin synkronointimenetelmiin.[51]
Microsoft on käyttänyt atomisista toiminnoista nimitystä lomittuvat toiminnot (engl. interlocked operations).[52]
Atomisuutta käytetään synkronointiprimitiivien lisäksi myös muualla kuten tietokantojen ACID-periaatteessa sekä tiedostojärjestelmän toimintaperiaatteiden kuvauksissa, jotka ovat eri tason käsitteitä.
ReaderWriterLock
[muokkaa | muokkaa wikitekstiä]ReaderWriterLock on Microsoftin synkronointiprivimitiivi, joka sallii useamman lukevan säikeen mutta vaatii kirjoittavan säikeen odottavan poissulkevaa lukitusta.[8][53]
Read-Copy-Update
[muokkaa | muokkaa wikitekstiä]Read-Copy-Update (RCU) mekanismi on Linuxissa käytetty skaalautuva mekanismi.[54] RCU:n etu on äärimmäisen nopea lukeminen harvoin päivitettäville tiedoille.[54][55]
RCU on immuuni deadlock tilanteelle ja se on lähtöisin Sequentin DYNIX/ptx ytimestä, jossa RCU:n käytöllä vähennettiin huomattavasti deadlockin välttämiseen tarkoitettua ohjelmakoodia.[56] Mekanismia on käytetty myös IBM:n tutkimusalustassa K42.[57]
Mekanismi käyttää elävän (luettavissa olevan) ja kuolleen (muokattavissa olevan) muuttujan periaatetta.[58]
Koska menetelmä käyttää eri operaatioita lukevalle ja kirjoittavalle puolelle tähän viitataan myös reader-writer locking konseptina, mutta sillä korvataan Linuxissa käytettyä read-write lock mekanismia (rwlock).[59]
RCU:sta on tehty esitys C++:n ISO-standardointiin liittyen.[60][61]
Myös käyttäjäavaruudessa käytettävää muunnosta on kehitetty.[62]
Compare-and-Swap
[muokkaa | muokkaa wikitekstiä]Lukottomalle Compare-and-Swap -mekanismille perustuen on toteutettu moniprosessointia tukeva Synthesis-ydin Sony NEWS -työasemalle.[63]
Keskeytyskäsittely
[muokkaa | muokkaa wikitekstiä]Laitteistokeskeytyksien pysäyttäminen kriittisen alueen ajaksi on yksi reaaliaikajärjestelmien käyttämä menetelmä.[1] Keskeytyksien ja moniajon pysäyttäminen ei ole lukitus, mutta se voi olla alhaisen tason koodin käyttämä tiedon serialisointiin suoritinkohtaisesti.[64]
Lukottomat algoritmit
[muokkaa | muokkaa wikitekstiä]Lukoton algoritmi voi vaihtaa (esimerkiksi linkitetyn listan kanssa) käytössä olevan alkion toiseen kun se on valmis häiritsemättä olemassa olevaa käyttöä.[5] Ohjelmointikielen kääntäjä on usein vapaa muuttamaan toimintojen järjestystä, joka voi aiheuttaa ongelmia.[65][5][66] Lisäksi suorittimet voivat käyttää myös spekuloivaa suoritusta.[54] Välimuistin yhtenäisyyden vuoksi suorittimet vaihtavat viestejä (QuickPath, HyperTransport) varmistaakseen järjestyksen.[6] Assembly-tasolla ohjelmoija näkee operaatioita kuten load-store-käskyjä viestivälityksen sijaan.[67]
Hazardiosoitin on eräs menetelmä, joka on kehitetty dynaamisien olioiden lukottomaan käsittelyyn.[68][69][70]
Synkronointivirheet
[muokkaa | muokkaa wikitekstiä]Rinnakkaiset ohjelmat voivat joutua myös odottamaan toisiaan jolloin säikeet estävät toisiaan (engl. block).
Väärin tehty ohjelma voi lukkiutua (engl. deadlock). Lukkiutumistilanteessa prosessit odottavat tosiaan, eikä yksikään niistä pääse etenemään.[2]
Nälkiintyminen tai nääntyminen (engl. starvation) on tilanne, jossa ainakin yksi säie jatkuvasti jää ilman riittävää suoritinaikaa, vaikka varsinaista lukkiutumista ei esiinnykään – järjestelmä siis ainakin osittain toimii kuten pitääkin.[71]
Livelock on deadlockin ja nälkiintymisen kaltainen tilanne, jossa säikeet jatkavat toimintaansa mutta eivät pääse etenemään: säikeet eivät ole pysäytettyjä kuten deadlockissa.[71] Käsitteen nimen esitti E. A. Ashcroft vuonna 1973 artikkelissaan Proving Assertions about Parallel Program.[72]
Priority inversion tilanne on ongelmallinen vuoronnetuissa järjestelmissä, joissa korkean prioriteetin säie voi joutua odottamaan matalan prioriteetin säiettä vapauttamaan varatun resurssin, mutta matalan prioriteetin säie ei saa riittävästi ajoaikaa.[1] Ongelman välttämiseksi Real-Time Linux tukee prioriteettien perintämekanismia.[73][74]
Käytetystä synkronointimenetelmästä riippuen järjestelmä voi olla myös herkkä busy-wait tilanteelle, jossa suoritinaikaa käytetään paljon odotettavan lukituksen tilan tarkisteluun. Etenkin spinlock synkronointi voi aiheuttaa tämän tilanteen.[20]
Synkronointituki
[muokkaa | muokkaa wikitekstiä]Itse alustakohtaisen tuen lisäksi eräissä ohjelmointikielissä on niiden standardikirjastossa toteutus alustakohtaiselle synkronointimekanismille, joka käyttää samaa rajapintaa eri alustoilla.
Esimerkiksi C++11 lisäsi std::mutex
toteutuksen kielen standardikirjastoon. C11 lisäsi säikeistyksen tuen mukaan lukien synkronointituen.
Käyttö
[muokkaa | muokkaa wikitekstiä]Esimerkki synkronoinnin käyttötapauksesta, jossa write_value()
kirjoittaa arvon ja read_value()
lukee arvon:
Sync valuelock;
int myValue;
void write_value(int scalarvalue)
{
valuelock->lock();
myValue = scalarvalue;
valuelock->release();
}
int read_value()
{
int current_value;
valuelock->lock();
current_value = myValue;
valuelock->release();
return current_value;
}
Huomaa, että tämä esimerkki olettaa synkronointiprimitiivin odottavan kunnes lukitus onnistuu ja voi pysäyttää säikeen suorituksen pitkäksi aikaa.
Esimerkki demonstroi miten lukittavan arvon käsittely on oltava sekä luku- että kirjoituspuolella sopimuskäytännön mukaisesti.
Katso myös
[muokkaa | muokkaa wikitekstiä]- Aterioivat filosofit
- Producer–consumer problem (englanniksi)
- Readers–writers problem (englanniksi)
- Fork–join model (englanniksi)
Lähteet
[muokkaa | muokkaa wikitekstiä]- ↑ a b c Khushu, Sanjeev & Simmons, Johnathan: Scheduling and Synchronization in Embedded Real-Time Operating Systems (PDF) cseweb.ucsd.edu. Viitattu 20.2.2017.
- ↑ a b c d Jones, M. Tim: Anatomy of Linux synchronization methods IBM. Viitattu 21.2.2017.
- ↑ Kernel Synchronization cs.utexas.edu. Viitattu 21.2.2017.
- ↑ Huang, Yi: Contract-based synchronization of multi-threaded Java programs d.lib.msu.edu. 2011. doi:10.25335/58x3-vr49 ISBN 9781267057402 Viitattu 5.7.2024. (englanniksi)
- ↑ a b c Jonathan Corbet: Lockless algorithms for mere mortals lwn.net. 28.7.2020. Viitattu 1.11.2021. (englanniksi)
- ↑ a b Paolo Bonzini: An introduction to lockless algorithms lwn.net. 19.2.2021. Viitattu 1.11.2021. (englanniksi)
- ↑ a b c d Michel Raynal: Concurrent Programming: Algorithms, Principles, and Foundations, s. vi, 2, 112, 134. Springer, 2013. doi:10.1007/978-3-642-32027-9 ISBN 978-3-642-32027-9 (englanniksi)
- ↑ a b Overview of Synchronization Primitives Microsoft. Viitattu 2.3.2017.
- ↑ Schulzrinne, Henning: Synchronization primitives cs.columbia.edu. Viitattu 2.3.2017.
- ↑ Synchronization primitives in the Linux kernel. 0xax.gitbooks.io. Viitattu 2.3.2017.
- ↑ Synchronization Primitives: Implementation and Examples inst.eecs.berkeley.edu. Viitattu 2.3.2017.
- ↑ Lecture 4: Synchronization Primitives cs.brown.edu. Viitattu 2.3.2017.
- ↑ 18.5.7. Synchronization primitives docs.python.org. Viitattu 2.3.2017.
- ↑ Choosing Between Synchronization Primitives Intel. Viitattu 2.3.2017.
- ↑ Synchronization primitives qnx.com. Viitattu 2.3.2017.
- ↑ Synchronization Primitives Apple. Viitattu 2.3.2017.
- ↑ ARM® Synchronization Primitives Development Article ARM. Viitattu 2.3.2017.
- ↑ Synchronization primitives IBM. Viitattu 2.3.2017.
- ↑ Dillon, Matt: Multi-Processing Basics for a *real* multi-processing system. groups.google.com. Viitattu 2.3.2017.
- ↑ a b Blieberger, Johann & Burgstaller, Bernd & Scholz, Bernhard: Busy Wait Analysis complang.tuwien.ac.at. Viitattu 21.2.2017.
- ↑ Chapter 6. Synchronization helenos.org. Viitattu 1.11.2021. (englanniksi)
- ↑ Lecture 6: architectural support for synchronization cs.cornell.edu. Viitattu 1.11.2021. (englanniksi)
- ↑ a b c Chapter 5. Kernel Synchronization linux.linti.unlp.edu.ar. Viitattu 21.2.2017.
- ↑ Waiman Long: qspinlock: Introducing a 4-byte queue spinlock implementation lwn.net. 31.7.2013. Viitattu 1.11.2021. (englanniksi)
- ↑ a b McKenney, Paul E.: Memory Ordering in Modern Microprocessors rdrop.com. Viitattu 21.2.2017.
- ↑ a b c d e Yang, Junfeng: W4118 Operating Systems cs.columbia.edu. Viitattu 21.2.2017.
- ↑ McKenney, Paul E.: Memory Barriers: a Hardware View for Software Hackers 7.6.2010. Viitattu 2.3.2017.
- ↑ Read Write Semaphores halolinux.us. 1.3.2021. Viitattu 2.9.2021. (englanniksi)
- ↑ H. Ballhausen: Fair Solution to the Reader-Writer-Problem with Semaphores only (PDF) arxiv.org. Viitattu 2.9.2021. (englanniksi)
- ↑ Buttazzo, Giorgio C.: Hard Real-Time Computing Systems, s. 19. (Third Edition) Springer, 2011. ISBN 978-1-4614-0675-4
- ↑ Edsger Dijkstra: 3.2 The Synchronizing Primitives (PDF) (sivu 28) cs.utexas.edu. Viitattu 1.11.2021. (englanniksi)
- ↑ a b Turnstiles and priority inheritance sunsite.uakom.sk. Viitattu 21.2.2017.
- ↑ Understanding Vulkan Synchronization khronos.org. Viitattu 27.11.2021. (englanniksi)
- ↑ Synchronization Framework source.android.com. Viitattu 27.11.2021. (englanniksi)
- ↑ Asymmetric Memory Fences: Optimizing Both Performance and Implementability (PDF) iacoma.cs.uiuc.edu. Viitattu 27.11.2021. (englanniksi)
- ↑ Generic Mutex Subsystem kernel.org. Viitattu 25.11.2021. (englanniksi)
- ↑ a b c Jonathan Corbet: Wait/wound mutexes lwn.net. 1.5.2013. Viitattu 1.11.2021. (englanniksi)
- ↑ a b Wound/Wait Deadlock-Proof Mutex Design kernel.org. Viitattu 1.11.2021. (englanniksi)
- ↑ a b James H. Anderson: Lamport on Mutual Exclusion: 27 Years of Planting Seeds (PDF) cs.unc.edu. Viitattu 25.11.2021. (englanniksi)
- ↑ Leslie Lamport: Time, Clocks, and the Ordering of Events in a Distributed System (PDF) lamport.azurewebsites.net. heinäkuu 1978. Viitattu 25.11.2021. (englanniksi)
- ↑ Peterson's Algorithm cs.wcupa.edu. Arkistoitu 27.9.2022. Viitattu 8.6.2022. (englanniksi)
- ↑ MSVC mutex is slower than you might expect stoyannk.wordpress.com. Viitattu 2.3.2017.
- ↑ a b Always Use a Lightweight Mutex preshing.com. Viitattu 2.3.2017.
- ↑ John M. Mellor-Crummey & Michael L. Scott: Algorithms for scalable synchronization on shared-memory multiprocessors (PDF) cs.rochester.edu. helmikuu 1991. Viitattu 25.11.2021. (englanniksi)
- ↑ a b c Critical Section Objects Microsoft. Viitattu 2.3.2017.
- ↑ Arpaci-Dusseau, Remzi H. & Arpaci-Dusseau, Andrea C.: Concurrency: An Introduction pages.cs.wisc.edu. Viitattu 2.3.2017.
- ↑ William Stallings: Operating System Internals and Design Principles, s. 296. (Seventh edition) Prentice Hall, 2012. ISBN 978-0-13-230998-1 (englanniksi)
- ↑ Atomic Operations Intel. Viitattu 2.3.2017.
- ↑ a b Levi Ego: Memory Synchronization Techniques (PDF) es.cs.uni-kl.de. Viitattu 27.11.2021. (englanniksi)
- ↑ *Stroustrup, Bjarne: The C++ Programming Language, 4th ed.. Addison-Wesley, 2015. ISBN 0-321-56384-0
- ↑ Comparison: Lockless programming with atomics in C++ 11 vs. mutex and RW-locks arangodb.com. Viitattu 9.2.2017.
- ↑ Interlocked Operations Microsoft. Viitattu 2.3.2017.
- ↑ Reader-Writer Locks Microsoft. Viitattu 2.3.2017.
- ↑ a b c McKenney, Paul: What is RCU, Fundamentally? LWN. Viitattu 21.2.2017. (englanniksi)
- ↑ Paul E. McKenney: Structured Deferral: Synchronization via Procrastination (PDF) dl.acm.org. 2013. doi:10.1145/2488364.2488549 Viitattu 5.7.2024. (englanniksi)
- ↑ McKenney, Paul E. & Boyd-Wickizer, Silas & Walpole, Jonathan: RCU Usage In the Linux Kernel: One Decade Later eecg.toronto.edu. Viitattu 2.3.2017.
- ↑ Notes on Read-Copy Update read.seas.harvard.edu. Viitattu 2.3.2017.
- ↑ McKenney, Paul E. & Sarma, Dipankar & Arcangeli, Andrea & Kleen, Andi & Krieger, Orran & Russell, Rusty: Read Copy Update kernel.org. Viitattu 2.3.2017.
- ↑ McKenney, Paul: What is RCU? Part 2: Usage LWN. Viitattu 2.3.2017.
- ↑ McKenney, Paul E.: Read-Copy Update (RCU) for C++ open-std.org. Viitattu 2.3.2017.
- ↑ Read-Copy Update (RCU) (PDF) open-std.org. 8.3.2023. Viitattu 17.7.2024. (englanniksi)
- ↑ Userspace RCU Implementation github.com. Viitattu 14.4.2022. (englanniksi)
- ↑ Massalin, Henry & Pu, Calton: A Lock-Free Multiprocessor OS Kernel (PDF) web.cecs.pdx.edu. 19.6.1991. Arkistoitu 31.3.2020. Viitattu 26.10.2019. (englanniksi)
- ↑ Marta Rybczyńska: Local locks in the kernel lwn.net. 11.8.2020. Viitattu 27.11.2021. (englanniksi)
- ↑ Calibrating your fear of big bad optimizing compilers lwn.net. 11.10.2019. Viitattu 25.11.2021. (englanniksi)
- ↑ Paolo Bonzini: Lockless patterns: an introduction to compare-and-swap lwn.net. 12.3.2021. Viitattu 25.11.2021. (englanniksi)
- ↑ Paolo Bonzini: Lockless patterns: full memory barriers lwn.net. 5.3.2021. Viitattu 25.11.2021. (englanniksi)
- ↑ Maged M. Michael: Hazard Pointers: Safe Memory Reclamation for Lock-Free Objects (PDF) kesäkuu 2004. Arkistoitu Viitattu 5.7.2024. (englanniksi)
- ↑ Andrei Alexandrescu & Maged Michael: Lock-Free Data Structures with Hazard Pointers drdobbs.com. 1.12.2004. Viitattu 5.7.2024. (englanniksi)
- ↑ Proposed Wording for Concurrent Data Structures: Hazard Pointer and Read-Copy-Update (RCU) (PDF) open-std.org. Viitattu 5.7.2024. (englanniksi)
- ↑ a b Starvation and Livelock docs.oracle.com. Viitattu 23.11.2019. (englanniksi)
- ↑ E. A Ashcroft: Proving Assertions about Parallel Program (PDF) core.ac.uk. 19.1.1973. Viitattu 23.11.2019. (englanniksi)
- ↑ Steven Rostedt: Real-Time is coming to Linux (PDF) events19.linuxfoundation.org. 24.10.2018. Viitattu 26.10.2019. (englanniksi)
- ↑ RT-mutex subsystem with PI support kernel.org. Viitattu 27.11.2021. (englanniksi)
Aiheesta muualla
[muokkaa | muokkaa wikitekstiä]- Paul McKenneyn RCU sivut (englanniksi)