En semafor er en synkroniseringsprimitiv [1] af arbejdet med processer og tråde , som er baseret på en tæller, på hvilken der kan udføres to atomariske operationer : øg og sænk værdien med én, mens reduktionsoperationen for nulværdien af tælleren blokerer [2] . Tjener til at bygge mere komplekse synkroniseringsmekanismer [1] og bruges til at synkronisere opgaver, der kører parallelt, til at beskytte dataoverførsel gennem delt hukommelse , til at beskytte kritiske sektioner og også til at kontrollere adgang til hardware.
Computational semaforer bruges til at kontrollere begrænsede ressourcer [3] . Binære semaforer giver gensidig udelukkelse af udførelse af kritiske sektioner [4] , og deres forenklede implementering er mutexes , som er mere begrænset i brug [5] . Ud over gensidig udelukkelse i det generelle tilfælde kan semaforer og mutexes bruges i mange andre typiske algoritmer, herunder signalering til andre opgaver [6] , hvilket kun tillader én opgave at passere bestemte kontrolpunkter ad gangen, analogt med en tællekors [7 ] , producent- og forbrugerproblemet, som indebærer overførsel af data fra én opgave til en anden [8] , barrierer, der tillader grupper af opgaver at blive synkroniseret ved bestemte kontrolpunkter [9] , betingelsesvariable til at underrette andre opgaver om eventuelle hændelser [3] og læse- og skrivelåse, der tillader samtidig læsning af data, men forbyder deres samtidige ændring [10] .
Typiske problemer ved at bruge semaforer er samtidig blokering af to opgaver, mens man venter på hinanden [11] og ressourcesult, som følge af, at en ressource periodisk kan være utilgængelig til nogle opgaver på grund af dens brug af andre opgaver [12] . Når det bruges i realtidsprocesser, kan der forekomme prioritetsinversion, hvilket kan forårsage, at en højere prioritetsproces blokerer på ubestemt tid på grund af den lavere prioritetsproces, der erhverver semaforen, mens CPU-tid gives til medium prioritetsprocessen [13] , løsningen til som er prioriteret arv [14] .
Begrebet semafor blev introduceret i 1965 af den hollandske videnskabsmand Edsger Dijkstra [15] , og i 1968 foreslog han at bruge to semaforer til at løse problemet med producent og forbruger [8] .
En semafor er en tæller, hvorpå der kan udføres to operationer: øg med 1 ( engelsk op ) og mindsk med 1 ( engelsk ned ). Når du forsøger at reducere en semafor, hvis værdi er nul, skal opgaven, der anmodede om denne handling, blokere, indtil det bliver muligt at reducere værdien af semaforen til en ikke-negativ værdi, dvs. indtil en anden proces øger værdien af semaforen [ 16] . Blokering af en opgave forstås som en ændring i tilstanden af en proces eller tråd af opgaveplanlæggeren til sådan, at opgaven vil suspendere dens udførelse [17] .
Operationerne med at formindske og øge værdien af en semafor blev oprindeligt betegnet med bogstaverne P (fra hollandsk proberen - at prøve) og V (fra hollandsk raise - at hæve højere). Dijkstra gav disse notationer til operationer på semaforer, men da de ikke forstås af folk, der taler andre sprog, bruges andre notationer normalt i praksis. Betegnelserne upog downblev først brugt i sproget Algol 68 [18] .
Increment- og decrementoperationerne for en semafor skal sammen med alle kontroller være atomare . Hvis der i det øjeblik, hvor værdien af semaforen øges, er mere end én proces blokeret på denne semafor, så vælger operativsystemet en af dem og tillader det at fuldføre operationen med at reducere værdien af semaforen [16] .
Det er generelt accepteret, at værdien af en semafor er ikke-negativ, men der er en anden tilgang til at definere en semafor, hvor en negativ værdi forstås som antallet af blokerede opgaver med et negativt fortegn. Med denne tilgang blokerer faldet af semaforen, hvis resultatet af operationen blev negativ [17] .
Hovedformålet med semaforen er at tillade eller midlertidigt forbyde udførelsen af enhver handling, så hvis værdien af semafortælleren er større end nul, så siger de, at den er i en signaltilstand, hvis værdien er nul - i en ikke-signaltilstand [19] . Reduktion af værdien af en semafor kaldes også nogle gange en anskaffelse ( eng. erhverve [20] ), og at øge værdien - frigive eller frigive ( eng. release [20] ) [21] , hvilket gør det muligt at lave beskrivelsen af driften af en semafor mere forståelig i sammenhæng med at kontrollere brugen af en ressource eller når den bruges i kritiske sektioner.
Generelt kan en semafor repræsenteres som et objekt bestående af [22] :
Konceptet med en semafor er velegnet til synkronisering af tråde, det kan bruges til synkronisering af processer, men det er fuldstændig uegnet til at synkronisere interaktionen mellem computere. En semafor er en synkroniseringsprimitiv på lavt niveau, så bortset fra at beskytte kritiske sektioner kan den være vanskelig at bruge alene [23] . En anden primitiv synkronisering på lavere niveau er futex . Det kan leveres af operativsystemet og er velegnet til implementering af semaforer på applikationsniveau, når der bruges atomoperationer på en delt tæller [24] .
Semaforer kan være binære og beregningsmæssige [3] . Databehandlingssemaforer kan tage ikke-negative heltalsværdier og bruges til at arbejde med ressourcer, hvis antal er begrænset [3] , eller deltage i synkroniseringen af parallelle eksekveringsopgaver. Binære semaforer kan kun tage værdierne 0 og 1 [3] og bruges til gensidigt at udelukke to eller flere processer fra at være i deres kritiske sektioner på samme tid [4] .
Mutex semaforer [3] ( mutexes ) er en forenklet implementering af semaforer, der ligner binære semaforer med den forskel, at mutexer skal frigives af den samme proces eller tråd, der erhverver dem [25] , men afhængigt af typen og implementeringen, en forsøg på at frigive af en anden tråd kan hvordan man frigiver mutex og returnerer en fejl [26] . Sammen med binære semaforer bruges de til at organisere kritiske kodesektioner [27] [28] . I modsætning til binære semaforer kan starttilstanden af en mutex ikke fanges [29] , og de kan understøtte prioritetsarv [30] .
Letvægts semaforer er semaforer, der bruger en aktiv venteløkke, før de udfører en lås. En aktiv venteløkke giver dig i nogle tilfælde mulighed for at reducere antallet af systemopkald [3] .
Signalering, også kaldet notifikation, er det grundlæggende formål med semaforer, det sikrer at et stykke kode i en opgave udføres efter at et stykke kode i en anden opgave er eksekveret [6] . Signalering af brugen af en semafor involverer normalt at sætte dens startværdi til 0, så opgaver, der venter på den signalerede tilstand, kan blokere, indtil hændelsen indtræffer. Signalering udføres ved at øge værdien af semaforen, og venting udføres ved at dekrementere værdien [29] .
hovedstrøm | |
---|---|
| |
Stream 1 | Stream 2 |
|
Tråd 2 fik CPU-tid først
|
Semaforer er velegnede til at signalere en eller flere opgaver, hvis antal er kendt på forhånd. Hvis antallet af opgaver, der venter på en signaltilstand, ikke kendes på forhånd, bruges betingelsesvariable normalt .
Gensidig udelukkelseI flertrådede applikationer er det ofte påkrævet, at separate sektioner af kode, kaldet kritiske sektioner , ikke kan køre parallelt, f.eks. når man får adgang til en ikke-delt ressource, eller når man ændrer delt hukommelsesplacering. For at beskytte sådanne områder kan du bruge en binær semafor eller en mutex [3] . En mutex er mere sikker at bruge, fordi den kun kan frigives af den proces eller tråd, der har erhvervet den [5] . Det kan også være mere effektivt at bruge en mutex i stedet for en semafor på grund af optimeringen for to værdier på assembly-kodeimplementeringsniveauet.
Startværdien af semaforen er sat til én, hvilket betyder, at den ikke er fanget - ingen er kommet ind i den kritiske sektion endnu. Indtastningen ( engelsk enter ) i den kritiske sektion er indfangningen af semaforen - dens værdi reduceres til 0, hvilket gør et gentaget forsøg på at komme ind i den kritiske sektionsblokering. Når man forlader ( eng. leave ) fra den kritiske sektion, frigives semaforen, og dens værdi bliver lig med 1, hvilket tillader genindtræden i den kritiske sektion, inklusive andre tråde eller processer .
For forskellige ressourcer kan der være forskellige semaforer, der er ansvarlige for kritiske sektioner. Således kan kritiske sektioner beskyttet af forskellige semaforer arbejde parallelt.
hovedstrøm | |
---|---|
| |
Stream 1 | Stream 2 |
Tråd 1 fik CPU-tid først
|
A fanget i stream 1
|
Ud over semaforer kan gensidig udelukkelse organiseres gennem andre synkroniseringsmetoder, for eksempel gennem monitorer , hvis de understøttes af det anvendte programmeringssprog. Monitorer giver dig mulighed for at beskytte et sæt data ved at skjule synkroniseringsdetaljer fra programmøren og give adgang til beskyttede data kun for at overvåge procedurer, og implementeringen af monitorer er overladt til compileren og er normalt baseret på en mutex eller en binær semafor. Sammenlignet med semaforer kan skærme reducere antallet af fejl i programmer, men på trods af den lette brug er antallet af sprog, der understøtter skærme, lille [31] .
TurnstileDet er ofte opgaven at tillade eller nægte en eller flere opgaver at passere gennem bestemte kontrolpunkter. For at løse dette problem bruges en algoritme baseret på to semaforer, som i sin funktion ligner en tællekors, da den tillader kun at springe én opgave over ad gangen. Drejekorset er baseret på en semafor, som fanges ved checkpoints og straks frigives. Hvis det er påkrævet at lukke drejekorset, skal semaforen beslaglægges, hvilket resulterer i, at alle opgaver, der passerer gennem drejekorset, vil blive blokeret. Hvis du vil tillade opgaver at passere gennem tælleren igen, så er det nok at frigive semaforen, hvorefter opgaverne fortsætter med at udføre på skift [7] .
Skiftevis at passere gennem drejekorset har en stor ulempe - for hver passage kan der forekomme et unødvendigt kontekstskift mellem opgaver, som et resultat af hvilket algoritmens ydeevne vil falde. I nogle tilfælde kan løsningen være at bruge en flersædet tæller, der ophæver blokeringen af flere opgaver på én gang, hvilket for eksempel kan udføres ved cyklisk at frigive semaforen, hvis den anvendte semaforimplementering ikke understøtter en stigning med et vilkårligt tal [ 32] .
Turnstile pseudokodeInitialisering | Turnstile | blokering | Lås op |
---|---|---|---|
drejekors = Semafor(1) | gribe (turstile) give slip (turstile) | gribe (turstile) | give slip (turstile) |
Semaforbaserede tællere kan fx bruges i barrieremekanismer [33] eller læse/skrivelåse [34] .
SkiftEn anden typisk semafor-baseret algoritme er switch-implementeringen. Opgaver kan tage fat i kontakten og slippe den. Den første opgave, der griber kontakten, er at tænde den. Og den sidste opgave, der frigiver den, slukker den. For denne algoritme kan vi tegne en analogi med en lyskontakt i et rum. Den første, der kommer ind i rummet, tænder lyset, og den sidste, der forlader rummet, slukker det [35] .
Algoritmen kan implementeres baseret på tælleren af opgaver, der fangede switchen og switch-semaforen, hvor operationer skal beskyttes af en mutex. Når kontakten er fanget, øges tælleren med 1, og hvis dens værdi er ændret fra nul til en, så fanges switch-semaforen, hvilket svarer til at tænde for kontakten. I dette tilfælde er inkrementering af tælleren sammen med kontrol og indfangning af semaforen en atomoperation beskyttet af en mutex. Når kontakten slippes, falder tælleren, og hvis dens værdi bliver nul, frigives switch-semaforen, det vil sige, at kontakten skiftes til slukket tilstand. At mindske tælleren sammen med at kontrollere den og frigive semaforen må også være en atomoperation [35] .
Pseudokode for afbryderens driftsalgoritmeDatatype | Initialisering | Brug |
---|---|---|
Kontakt: antal = 0 mutex = semafor(1) Kontakt, lås (mål-semafor): grab (mutex) mængde += 1 hvis tæller == 1: fange (mål-semafor) frigive (mutex) Kontakt, unlock (mål-semafor): grab (mutex) mængde -= 1 hvis tæller == 0: frigivelse (mål-semafor) frigive (mutex) | switch = Switch() semafor = semafor(1) | blok (switch, semafor) // Kritisk del af switchen, // semaforen er låst låse op (kontakt, semafor) |
Switch-algoritmen bruges i en mere kompleks mekanisme - læse- og skrivelåse [35] .
Problemet med producent og forbrugerForbrugerproducentopgaven indebærer produktion af nogle oplysninger fra én opgave og overførsel af disse oplysninger til en anden opgave til behandling. I flertrådede systemer kan samtidig produktion og forbrug føre til løbsforhold , hvilket kræver brug af kritiske sektioner eller andre måder at synkronisere på. Semaforen er den enkleste synkroniseringsprimitiv, der kan bruges til at løse problemet med producent og forbruger.
Sender data gennem en ringbufferRingbufferen er en buffer med et fast antal elementer, hvor data indtastes og behandles i en først-ind-først-ud ( FIFO ) basis. I en enkelt-trådet version er 4 hukommelsesceller nok til at organisere en sådan buffer:
I en multitasking-implementering kompliceres algoritmen af behovet for at synkronisere opgaver. For to opgaver (producent og forbruger) kan vi begrænse os til to hukommelsesceller og to semaforer [8] :
Startværdien af semaforen, der er ansvarlig for læsning, er sat til 0, fordi køen er tom. Og værdien af semaforen, der er ansvarlig for at skrive, er sat lig med bufferens samlede størrelse, det vil sige, at hele bufferen er tilgængelig til udfyldning. Før udfyldning af det næste element i bufferen, dekrementeres skrivesemaforen med 1, idet det næste element i køen reserveres til at skrive data, hvorefter skriveindekset ændres, og læsesemaforen øges med 1, hvilket gør det muligt at læse det tilføjede element til køen. Læseopgaven fanger tværtimod semaforen til læsning, hvorefter den læser det næste element fra bufferen og ændrer indekset for det næste element til læsning, og frigiver derefter semaforen til skrivning, så skriveopgaven kan skrive til det frigjorte element [8] .
Ringbuffer-pseudokodeInitialisering | Brug |
---|---|
bufferstørrelse = N skrivetilladelse = Semafor(bufferstørrelse) læsetilladelse = Semafor(0) pr. skrivning = 0 on-read = 0 buffer = matrix(bufferstørrelse) | // Skriveopgave produceret-element = producere-element() capture (skrivetilladelse) buffer[per-write] = produceret-vare pr. skrivning += 1 hvis pr. post >= bufferstørrelse: pr. skrivning = 0 frigivelse (læsetilladelse) // Læs opgave grab (læsetilladelse) element-read = buffer[per-read] pr. læsning += 1 hvis pr. læsning >= bufferstørrelse: on-read = 0 frigivelse (skrivetilladelse) proces (læse-element) |
Hvis en ringbuffer er implementeret for flere skribenter og læsere, tilføjes en mutex til implementeringen, der låser bufferen, når der skrives til eller læses fra den [36] .
Sender data gennem en vilkårlig bufferUdover at overføre data gennem en ringbuffer, er det også muligt at overføre via en vilkårlig, men i dette tilfælde skal skrive- og læsedata beskyttes af en mutex, og semaforen bruges til at underrette læseopgaven om tilstedeværelsen af det næste element i bufferen. Skriveopgaven tilføjer et element beskyttet af mutex'en til bufferen og signalerer derefter dets tilstedeværelse. Læseopgaven fanger semaforen og modtager derefter, under beskyttelse af mutexen, det næste element. Det er værd at nævne, at forsøg på at erhverve en mutex-beskyttet semafor kan føre til et dødvande, hvis et forsøg på at læse fra en tom buffer, og frigivelse af semaforen inde i en kritisk sektion kan forringe ydeevnen en smule. Denne algoritme, som i tilfældet med en ringbuffer beskyttet af en mutex, tillader flere opgaver at skrive og læse samtidigt [37] .
En barriere er en mekanisme til at synkronisere kritiske punkter for en gruppe af opgaver. Opgaver kan kun passere gennem barrieren på én gang. Før du indtaster et kritisk punkt, skal opgaver i en gruppe blokere, indtil den sidste opgave i gruppen når det kritiske punkt. Når alle opgaver er ved at gå ind i deres kritiske punkter, skal de fortsætte deres udførelse [9] .
Den enkleste løsning til at organisere en barriere i tilfælde af to opgaver er baseret på to binære semaforer A og B, initialiseret til nul. Ved det kritiske punkt i den første opgave skal semafor B signaleres, og derefter skal semafor A indfanges. Ved det kritiske punkt i den anden opgave skal semafor A først signaleres, og derefter skal B indfanges. vil signalere en anden opgave , der tillader dens udførelse. Når begge opgaver har nået deres kritiske punkter, vil deres semaforer blive signaleret, hvilket giver dem mulighed for at fortsætte deres udførelse [38] .
Simpel barriere pseudokodeInitialisering | Opgave ved hjælp af barrieren |
---|---|
målbeløb = N antal = 0 mutex = semafor(1) indgangsturnstile = Semafor(0) | // Første barrierefase grab (mutex) mængde += 1 hvis tælle == tælle-opgaver: frigivelse (indgang-turnstile) frigive (mutex) beslaglægge (indgang-turnstile) frigivelse (indgang-turnstile) // Kritisk punkt |
En sådan implementering er single-pass, da barrieren ikke vender tilbage til sin oprindelige tilstand, den har også lav ydeevne på grund af brugen af en enkelt-sædet tællekors, som kræver en kontekstkontakt for hver opgave, så denne løsning er af ringe brug i praksis [32] .
To-fase barriereEt kendetegn ved den tofasede barriere er, at hver opgave, når den bruges, stopper ved barrieren to gange - før det kritiske punkt og efter. To stop gør barrieren tilbagegående , da det andet stop tillader barrieren at vende tilbage til sin oprindelige tilstand [39] .
Den universelle reentrant-algoritme for den tofasede barrieremekanisme kan være baseret på brugen af en tæller af opgaver, der har nået det kritiske punkt, og to flersæders drejekors. Operationer på tælleren og styring af drejekors skal beskyttes af en mutex. I dette tilfælde skal det samlede antal opgaver kendes på forhånd. Det første drejekors tillader opgaver at passere til det kritiske punkt og skal først blokeres. Den anden springer opgaver over, der lige har passeret det kritiske punkt, og bør også blokeres i starten. Inden man nærmer sig det kritiske punkt, øges tælleren for nåede opgaver med 1, og så snart den når det samlede antal opgaver, låses den første tællekors op for alle opgaver, og sender dem til det kritiske punkt, hvilket sker atomisk gennem mutexen sammen med tællerstigningen og dens verifikation. Efter det kritiske punkt, men før det andet tællekors, falder tælleren for antallet af opgaver med 1. Når værdien når nul, låses den anden tællekors op for alle opgaver, mens operationer på den anden tæller også sker atomært, sammen med modnedsættelse og kontrol heraf. Som et resultat stopper alle opgaver først før det kritiske punkt og derefter efter. Efter at have passeret barrieren er tællerens og drejekorsernes tilstande i deres oprindelige værdier [32] .
Pseudokode for Reentrant Two-Phase Barrier AlgorithmInitialisering | Opgave ved hjælp af barrieren |
---|---|
mutex = semafor(1) antal = 0 indgangsturnstile = Semafor(0) exit-turnstile = Semafor(0) | // Første barrierefase grab (mutex) mængde += 1 hvis tælle == tælle-opgaver: frigivelse (indgang-turnstile, mængde) frigive (mutex) beslaglægge (indgang-turnstile) // Kritisk punkt // Anden barrierefase grab (mutex) mængde -= 1 hvis tæller == 0: frigivelse (udgangsturstile, mængde) frigive (mutex) beslaglægge (udgangsturstile) |
En betingelsesvariabel er en måde at give besked på afventende opgaver, når en hændelse opstår [3] . Den tilstandsvariable mekanisme på applikationsniveau er normalt baseret på en futex og giver funktioner til at vente på en begivenhed og sende et signal om dens forekomst, men separate dele af disse funktioner skal beskyttes af en mutex eller semafor, da futex, den betingelsesvariable mekanisme indeholder normalt yderligere delte data [40] . I simple implementeringer kan futex erstattes af en semafor, som, når den bliver underrettet, skal frigives lige så mange gange som antallet af opgaver, der abonneres på betingelsesvariablen, men med et stort antal abonnenter kan meddelelsen blive en flaskehals [41] .
Betingelsesvariabelmekanismen antager tilstedeværelsen af tre operationer: at vente på en hændelse, at signalere en hændelse til en opgave og at underrette alle opgaver om en hændelse. For at implementere en semafor-baseret algoritme skal du bruge: en mutex eller en binær semafor til at beskytte selve betingelsesvariablen, en tæller for antallet af ventende opgaver, en mutex til at beskytte tælleren, semafor A til at blokere ventende opgaver og en yderligere semafor B for at vække den næste venteopgave i tide [42] .
Når du abonnerer på begivenheder, øges tælleren for tilmeldte opgaver atomisk med 1, hvorefter den forudindfangede mutex af betingelsesvariablen frigives. Semafor A fanges derefter for at vente på, at begivenheden indtræffer. Ved forekomsten af en hændelse kontrollerer signaleringsopgaven atomisk tælleren for tilmeldte opgaver og underretter den næste opgave om hændelsens forekomst ved at frigive semafor A og derefter blokere på semafor B, mens man venter på bekræftelse af oplåsning. Den advarede opgave frigiver semafor B og genindhenter tilstandsvariablens mutex for at vende tilbage til dens oprindelige tilstand. Hvis der laves en udsendelsesmeddelelse om alle abonnerede opgaver, frigives semaforen A for blokerede opgaver i en cyklus i henhold til antallet af abonnerede opgaver i tælleren. I dette tilfælde sker meddelelsen atomisk under beskyttelse af tælleren mutex, således at tælleren ikke kan ændre sig under meddelelsen [42] .
Tilstandsvariabel PseudokodeTypeerklæring | Brug |
---|---|
betingelse-variabel(): antal = 0 mutex = semafor(1) wait-event = Semafor(0) modtage-event = Semafor(0) betinget-variabel, forventer(mål-mutex): grab (mutex) mængde += 1 frigive (mutex) release(target-mutex) grab (vente-begivenheder) release(get-events) grab(target-mutex) betinget-variabel, underrette(): grab (mutex) hvis mængde > 0: mængde -= 1 udgivelse (vente-begivenheder) grab(get-events) frigive (mutex) betinget-variabel, besøg-alle(): grab (mutex) hvis mængde > 0: frigivelse (vente-begivenheder, tæller) grab(få-begivenheder, tæl) antal = 0 frigive (mutex) | // initialisering hændelse = betingelsesvariabel() mutex = semafor(1) // Vent på en begivenhed grab (mutex) vente (begivenhed) // Kritisk del af begivenheden frigive (mutex) // Alarm én opgave underrette (begivenhed) // Giv besked til alle opgaver underrette-alle (begivenhed) |
Semaforløsningen har et væsentligt problem - to signalerende kontekstswitche, som i høj grad reducerer algoritmens ydeevne, så i det mindste på operativsystemniveau bruges den normalt ikke [42] .
Et interessant faktum er, at semaforen i sig selv let implementeres baseret på en betingelsesvariabel og en mutex [24] , mens implementeringen af en betingelsesvariabel baseret på semaforer er meget mere kompliceret [42] .
Læse- og skrivelåseEt af de klassiske problemer er synkroniseringen af adgangen til en ressource, der er tilgængelig for læsning og skrivning på samme tid. Læse- og skrivelåse er designet til at løse dette problem og giver dig mulighed for at organisere separate læse- og skrivelåse på en ressource, hvilket tillader samtidig læsning, men forbyder samtidig skrivning. Skrivningen blokerer også enhver læsning [10] . En effektiv mekanisme kan ikke bygges på basis af en futex alene, tælleren for antallet af læsere kan ændres uden at låse op for nogen opgaver [24] . Læse- og skrivelåse kan implementeres baseret på en kombination af mutexes og semaforer, eller mutexes og en betingelsesvariabel.
Den universelle algoritme, der er blottet for problemet med ressourcesult af skriveopgaver, inkluderer en binær semaforkontakt A til at organisere en kritisk sektion af læseopgaver og en drejekors til at blokere nye læseopgaver i nærværelse af ventende forfattere. Når den første opgave at læse ankommer, griber den semafor A med en switch, hvilket forhindrer skrivning. For forfattere beskytter semafor A den kritiske del af forfatteren, så hvis den er fanget af læserne, blokerer alle forfattere for at komme ind i deres kritiske del. Imidlertid er optagelsen af skribent-opgaver af semafor A og efterfølgende skrivning beskyttet af drejekorset semaforen. Derfor, hvis der opstår en blokering af en skriveopgave på grund af tilstedeværelsen af læsere, blokeres tællekorset sammen med nye læseopgaver. Så snart den sidste læser er færdig med sit job, frigives switch-semaforen, og blokeringen af den første forfatter i køen ophæves. I slutningen af sit arbejde frigiver den drejekorsets semafor, hvilket igen tillader arbejdet med at læse opgaver [34] .
Pseudokode for den universelle læse-skrive-låsealgoritmeInitialisering | Læseopgave | Skriveopgave |
---|---|---|
switch = Switch() skrivetilladelse = Semafor(1) drejekors = Semafor(1) | gribe (turstile) frigive (turstile) lås (switch, tilladelse-skrive) // Kritisk del af læseopgaven låse op (switch, permission-write) | gribe (turstile) capture (skrivetilladelse) // Kritisk del af skriveopgaven give slip (turstile) frigivelse (skrivetilladelse) |
På operativsystemniveau er der implementeringer af læse- og skrivesemaforer, som er modificeret på en særlig måde for at øge effektiviteten i massebrug [43] .
Et af de klassiske synkroniseringsproblemer er Dining Philosophers-problemet. Problemet omfatter 5 filosoffer, der spiser ved et rundt bord, 5 tallerkener, 5 gafler og en fælles pastaret midt på bordet. Der er en tallerken foran hver filosof, og en gaffel til højre og venstre, men hver gaffel er delt mellem to nabofilosoffer, og man kan kun spise pasta med to gafler ad gangen. Desuden kan hver af filosofferne enten tænke eller spise pasta [44] .
Filosoffer repræsenterer de tråde, der interagerer i programmet, og løsningen af problemet omfatter en række forhold [44] :
For at løse problemet kan hver gaffel tildeles en binær semafor. Da filosoffen forsøger at samle gaflen op, fanges semaforen, og så snart han er færdig med at spise, frigives gaflernes semaforer. Problemet er, at naboen allerede kunne tage gaflen, så er filosoffen blokeret, indtil hans nabo spiser. Hvis alle filosoffer begynder at spise på samme tid, er dødvande muligt [44] .
En løsning på dødvandet kunne være at begrænse antallet af filosoffer, der spiser på samme tid, til 4. I dette tilfælde vil mindst én filosof være i stand til at spise, mens de andre venter. Begrænsningen kan implementeres gennem en semafor med en begyndelsesværdi på 4. Hver af filosofferne vil fange denne semafor, før de tager gaflerne, og frigiver den efter at have spist. Også denne løsning garanterer, at filosoffer ikke sulter, for hvis en filosof venter på, at en nabo slipper gaflen, vil han slippe den før eller siden [44] .
Der er også en enklere løsning. Deadlock er muligt, hvis 5 filosoffer samtidig holder en gaffel i samme hånd, for eksempel hvis de alle er højrehåndede og tog den højre gaffel først. Hvis en af filosofferne er venstrehåndet og tager den venstre gaffel først, så er hverken dødvande eller sult mulig. Det er således tilstrækkeligt, at en af filosofferne først fanger semaforen på venstre gaffel, og derefter den højre, mens de andre filosoffer gør det modsatte [44] .
RutsjebaneEt andet klassisk problem er rutsjebaneproblemet , hvor et tog af vogne fyldes helt op med passagerer, derefter ruller dem rundt og kommer tilbage efter mere. Ifølge betingelserne for problemet overstiger antallet af villige passagerer antallet af sæder i toget, så de næste passagerer venter i kø, mens toget kører i en cirkel. Hvis toget har M-sæder, så skal toget først vente til M-passagerer sidder på deres pladser, derefter skal det give dem en tur, vente til de alle står af og igen vente på nye passagerer [45] .
Sammensætningen af vogne sammen med passagerer kan repræsenteres som interagerende opgaver. Hver passager bør blokeres, mens de venter på deres tur, og selve toget bør blokeres i stadierne med påfyldning og tømning af sæder. For at læsse og losse toget kan du bruge to semaforer med sporskifter, hver beskyttet af deres egen mutex, og for at spærre passagerer for af- og pålæsning kan du bruge to semaforer, der er ansvarlige for pladser i vognene. Ventende passagerer beslaglægger læssesemaforen, og toget med læssesemaforen giver M af dem besked om tilgængeligheden af sæder. Toget spærres herefter af et sporskifte, indtil den sidste påstigningspassager signalerer med passende semafor, hvorefter turen begynder. Før turen bliver passagerer blokeret af en semafor til aflæsning, som forhindrer dem i at forlade toget. Efter turen underretter toget M-passagerer med en aflæsningssemafor, så de kan stå af, og blokerer derefter sporskiftesemaforen til aflæsning, mens de venter, indtil alle passagerer er rejst. Så snart den sidste passager forlader toget, vil han signalere semaforen for det andet sporskifte og tillade toget at samle passagerer op igen [45] .
Konceptet med en semafor giver kun operationerne med at dekrementere og øge med 1. Samtidig kan en opgave, der dekrementerer en semafor, normalt ikke vide, om den vil blokere på den eller ej. Ved signalering er der ingen måde at vide, om der er opgaver blokeret af semaforen, og hvis en opgave signalerer en anden, blokeret semafor, så fortsætter begge opgaver med at arbejde parallelt, og der er ingen måde at vide, hvem af dem der vil modtage processortid næste [17] .
På trods af begrænsningerne af begrebet semaforer kan specifikke implementeringer af dem være blottet for visse begrænsninger. For eksempel er muligheden for at øge en semaforværdi med et vilkårligt tal tilvejebragt i Linux [46] , Windows [41] og System V (POSIX) [47] implementeringer . Og POSIX semaforer giver dig mulighed for at bestemme, om en semaforlås vil forekomme [48] .
Ud over begrænsningerne af selve begrebet semafor, er der også begrænsninger pålagt af operativsystemet eller en bestemt implementering af en semafor. Operativsystemets opgaveplanlægger er normalt ansvarlig for at allokere processortid mellem processer og tråde . Brugen af semaforer stiller en række krav til planlæggeren og semaforimplementeringen i sig selv for at forhindre ressourcesult, hvilket er uacceptabelt i multitasking-applikationer [49] .
De første to krav er nødvendige, så enhver opgave kan få processortid og ikke være i en uendelig klar tilstand, hvilket allerede giver dig mulighed for at skrive applikationer uden ressourcesult. Det tredje krav er nødvendigt for at forhindre ressourcesult i semaforbaseret gensidig udelukkelse. Hvis signalering kun vil øge semafortælleren, men ikke vil vække opgaven, der er blokeret på den, så er en situation mulig, når den samme opgave uendeligt frigiver og fanger semaforen, og andre blokerede opgaver ikke har tid til at gå ind i klartilstand, eller det gør de, men meget sjældnere. . Men selvom det tredje krav er opfyldt, er der i tilfælde af et stort antal blokerede opgaver mulighed for ressourcesult, hvis de samme opgaver låses op hver gang. Dette problem løses af det fjerde krav, som for eksempel observeres, hvis opgaver blokeret af semaforen vækkes på skift [49] .
Overholdelse af de første tre krav tillader implementering af de såkaldte svage semaforer og overholdelse af alle fire- stærke [49] .
Hvis semaforer bruges forkert, kan der opstå dødvande [50] - situationer hvor to eller flere parallelle opgaver bliver blokeret, mens de venter på en begivenhed fra hinanden [11] . I en sådan situation vil opgaver ikke kunne fortsætte deres udførelse normalt, og normalt skal en eller flere processer tvinges til at afslutte. Deadlocks kan være resultatet af simple semaforer eller andre synkroniseringsfejl eller raceforhold , som er sværere at fejlfinde.
En almindelig fejl er at kalde en underrutine inden for en kritisk sektion, der bruger den samme kritiske sektion [51] .
Et illustrativt eksempel på gensidig blokering kan være indlejrede optagelser af binære semaforer A og B, der beskytter forskellige ressourcer, forudsat at de fanges i omvendt rækkefølge i en af trådene, hvilket for eksempel kan skyldes stilforskelle i at skrive programmet kode. Fejlen ved en sådan implementering er en race-tilstand, som kan få programmet til at køre det meste af tiden, men i tilfælde af parallel ressourcegreb er chancerne for et dødvande høje [52] .
hovedstrøm | |
---|---|
| |
Stream 1 | Stream 2 |
|
|
Svarende til dødvande er problemet med ressourcesult, som måske ikke fører til et fuldstændigt ophør af arbejdet, men kan vise sig at være ekstremt negativt, når algoritmen implementeres. Essensen af problemet ligger i periodiske eller hyppige afslag på at skaffe en ressource på grund af dens indfangning af andre opgaver [12] .
Et typisk tilfælde for dette problem er en simpel implementering af læse-/ , som låser ressourcen til at skrive under læsning. Den periodiske fremkomst af nye læseopgaver kan føre til en ubegrænset skrivelås på ressourcen. Ved lav belastning af systemet kan problemet ikke vise sig i lang tid, men under høj belastning kan der opstå en situation, hvor der er mindst én læseopgave på et givet tidspunkt, hvilket vil gøre skrivelåsen permanent under tid med høj belastning [12] . Givet en semafor, der frigives, når køen af læsere er tom, ville en simpel løsning være at tilføje en binær semafor (eller mutex) for at beskytte skribenternes kode, mens den på samme tid fungerer som en drejekors for læsere. Forfattere vil gå ind i den kritiske sektion og få fat i en tom kø-semafor, der blokerer på to semaforer, så længe der er læsere. Læseropgaver vil blokere, når de går ind i drejekorset, hvis forfatteropgaven venter på, at læserne skal færdiggøre deres arbejde. Så snart den sidste læseopgave har afsluttet sit arbejde, frigiver den den tomme kø-semafor og ophæver blokeringen af den ventende skriveopgave [34] .
Gensidig udelukkelse kan også lide af ressourcesult, hvis implementeringen er baseret på svage semaforer, men der er algoritmer til at omgå begrænsningerne for svage semaforer i dette tilfælde [49] .
Et andet problem kan være den prioritetsinversion, der kan opstå, når semaforer bruges af realtidsprocesser. Realtidsprocesser kan kun afbrydes af operativsystemet til udførelse af processer med højere prioritet. I dette tilfælde kan processen blokere på semaforen og vente på, at den frigives af en proces med en lavere prioritet. Hvis der på dette tidspunkt kører en proces med en gennemsnitlig prioritet mellem to processer, så kan en proces med høj prioritet være blokeret i en ubegrænset periode [13] .
Problemet med prioritetsinversion løses ved at nedarve prioriteter [14] . Hvis det er muligt, kan semaforer erstattes af mutexes, da mutexes kan have præcedensarv forudbestemt. Når en mutex er fanget af en tråd med en højere prioritet, vil prioriteten af den opgave, der ejer mutex'en, blive forhøjet for at frigive den så hurtigt som muligt [30] .
Den allestedsnærværende arv af prioriteter er en ekstremt vanskelig opgave at implementere, så systemer, der understøtter det, kan kun have en delvis implementering. Prioritetsarv skaber også andre problemer, såsom manglende evne til at kombinere kode med prioriteret arv med kode uden nedarvning, når man bruger den samme kritiske sektion [54] .
Hvis det er nødvendigt at bruge semaforer, eller hvis der ikke er understøttelse for nedarvning af prioriteter, kan algoritmer modificeres til selvstændigt at øge prioriteterne efter opgaver [54] .
POSIX -standarderne på operativsystemniveau giver en C-sprog API til håndtering af semaforer både på trådniveau og på procesniveau via delt hukommelse . Standarderne definerer en semafordatatype og et sæt funktioner til at arbejde med den [55] . POSIX-semaforer er tilgængelige på Linux , macOS , FreeBSD og andre POSIX-kompatible operativsystemer. sem_t
Fungere | Beskrivelse |
---|---|
sem_init()[dok. en] | Initialisering af en semafor med en startværdi for tælleren og et brugsflag på procesniveau. |
sem_destroy()[dok. 2] | Slip semaforen. |
sem_open()[dok. 3] | Opret en ny eller åbn en eksisterende navngivet semafor. |
sem_close()[dok. fire] | Lukning af semaforen efter at have arbejdet med den. |
sem_unlink()[dok. 5] | Fjernelse af navnet fra en navngivet semafor (ødelægger den ikke). |
sem_wait()[dok. 6] | Sænk værdien af semaforen med 1. |
sem_timedwait()[dok. 7] | Reduktion af værdien af en semafor med 1 med en grænse for den maksimale blokeringstid, hvorefter en fejl returneres. |
sem_trywait()[dok. otte] | Forsøg på at dekrementere en semafor i ikke-blokerende tilstand returnerer en fejl, hvis dekrementering uden blokering ikke er mulig. |
sem_post()[dok. 9] | Forøg semaforværdien med 1. |
sem_getvalue()[dok. ti] | Få den aktuelle værdi af semaforen. |
En af ulemperne ved POSIX semaforer er den fejltilbøjelige funktionsspecifikation sem_timedwait(), der fungerer på realtidsur ( CLOCK_REALTIME) [56] i stedet for systemets oppetid ( CLOCK_MONOTONIC), hvilket kan få programmer til at gå ned, når systemtiden ændres og kan være kritisk for indlejret enheder [57] , men nogle real-time operativsystemer tilbyder analoger til denne funktion, der fungerer med systemets oppetid [58] . En anden ulempe er manglen på støtte til at vente på flere semaforer på samme tid eller på en semafor og en filbeskrivelse.
På Linux er POSIX semaforer implementeret i det futex-baserede Glibc- bibliotek [59] .
System V semaforerPOSIX-standarderne definerer også et sæt funktioner fra X/Open System Interfaces (XSI)-standarden for interprocess semafor-håndtering i operativsystemet [60] . I modsætning til almindelige semaforer kan XSI semaforer øges og reduceres med et vilkårligt antal, de er allokeret i arrays, og deres levetid strækker sig ikke til processer, men til operativsystemet. Så hvis du glemmer at lukke XSI-semaforen, når alle ansøgningsprocesser er afsluttet, vil den fortsætte med at eksistere i operativsystemet, hvilket kaldes en ressourcelæk. Sammenlignet med XSI semaforer er almindelige POSIX semaforer meget nemmere at bruge og kan være hurtigere [61] .
XSI semaforsæt i systemet identificeres med en numerisk nøgle af typen key_t, men det er muligt at oprette anonyme semaforsæt til brug i en applikation ved at angive en konstant IPC_PRIVATEi stedet for en numerisk nøgle [62] .
Fungere | Beskrivelse |
---|---|
semget()[dok. elleve] | Opretter eller får en semafor sæt identifikator med den givne numeriske nøgle [62] . |
semop()[dok. 12] | Udfører atomoperationer med dekrementering og inkrementering med det givne nummer af semafortælleren med dets nummer fra sættet med den givne identifikator, og tillader også blokering af at vente på nulværdien af semafortælleren, hvis 0 [47] er angivet som det givne tal . |
semctl()[dok. 13] | Giver dig mulighed for at styre en semafor efter dens nummer fra et sæt med en given identifikator, herunder at hente og indstille den aktuelle værdi af tælleren; også ansvarlig for at ødelægge semaforsættet [63] . |
Linux -operativsystemer understøtter POSIX semaforer, men tilbyder også et alternativ til semaforer i form af en tæller bundet til en filbeskrivelse via et systemkald eventfd()[dok. 14] med flag EFD_SEMAPHORE. Når en sådan tæller læses gennem en funktion read(), formindskes den med 1, hvis dens værdi ikke var nul. Hvis værdien var null, så sker blokering (hvis flaget ikke er angivet EFD_NONBLOCK), som det er tilfældet med almindelige semaforer. Funktionen write()øger tællerværdien med det tal, der er skrevet til filbeskrivelsen. Fordelen ved en sådan semafor er evnen til at vente på den signalerede tilstand af semaforen sammen med andre hændelser ved hjælp af systemkald select()eller poll()[46] .
Windows -kernen giver også en C API til at arbejde med semaforer. Tråde, der er blokeret på en semafor, er i kø FIFO , men kan gå til slutningen af køen, hvis tråden afbrydes for at behandle andre hændelser [19] .
Fungere | Beskrivelse |
---|---|
CreateSemaphoreA()[dok. femten] | Opret en semafor, med angivelse af startværdien af tælleren, den maksimale værdi og navnet på semaforen. |
OpenSemaphoreW()[dok. 16] | Adgang til en semafor ved dens navn, hvis den allerede eksisterer. |
CloseHandle()[dok. 17] | Lukning af semaforen efter at have arbejdet med den. |
WaitForSingleObject()[dok. 18] ellerWaitForMultipleObjects() [dok. 19] | Sænk semaforværdien med 1 med blokering i tilfælde af nul tællerværdi; giver dig mulighed for at begrænse den maksimale blokeringstid. |
ReleaseSemaphore()[dok. tyve] | Forøg værdien af semaforen med den angivne mængde. |
Windows semaforfunktioner inkluderer evnen til at øge en semafor med et vilkårligt tal [41] og evnen til at vente på dens signaltilstand sammen med blokerende ventetider for andre semaforer eller objekter [64] .
Semaforer understøttes normalt ikke eksplicit på programmeringssprogsniveau, men leveres ofte af indbyggede biblioteker eller tredjepartsbiblioteker. På nogle sprog, såsom Ada [65] og Go [66] , implementeres semaforer let i sproget.
Sprog | Modul eller bibliotek | Datatype |
---|---|---|
Xi | pthread,rt | sem_t[dok. 21] |
Ada | GNAT.Semaphores[dok. 22] | Counting_Semaphore,Binary_Semaphore |
C++ | Boost | boost::interprocess::interprocess_semaphore[dok. 23] |
C# | System.Threading[dok. 24] | Semaphore[dok. 25] |
D | core.sync.semaphore[dok. 26] | Semaphore[dok. 27] |
Gå | golang.org/x/sync/semaphore[dok. 28] | Weighted |
Java | java.util.concurrent[dok. 29] | java.util.concurrent.Semaphore[dok. tredive] |
Python | threading[dok. 31] ,asyncio [dok. 32] | threading.Semaphore[dok. 33] ,asyncio.Semaphore [dok. 34] |
Det enkleste eksempel på at bruge en semafor er den gensidige udelukkelse af muligheden for at udføre kritiske dele af kode for tråde eller processer. For at organisere gensidig udelukkelse kan en binær semafor og to funktioner tjene: at gå ind i den kritiske sektion og forlade den. For nemheds skyld inkluderer eksemplet ikke muligheden for at huske ID'et for den indfangende tråd og ID'et for den proces, som tråden tilhører. Det antages også, at den kritiske sektion har en begrænset, ikke særlig lang udførelsestid, så afbrydelser af semaforoptagelsesoperationen ( EINTR) ignoreres, og resultaterne af afbrydelsen kan behandles efter den kritiske sektion. Selve semaforen er abstraheret til en struktur for at forbedre kodens læsbarhed.
I eksemplet startes to tråde, hvoraf den ene øger tælleren, og den anden formindsker den. Da tælleren er en delt ressource, skal adgangen til den være gensidigt udelukkende, ellers kan en tråd overskrive resultaterne af en andens operationer, og den endelige resultatværdi kan være forkert. Derfor er tælleren beskyttet af en abstraheret binær semafor, der implementerer gensidig udelukkelse.
Eksempel på en simpel semaforbaseret kritisk sektionsimplementering i C (POSIX) #include <errno.h> #include <pthread.h> #include <semaphore.h> #include <stdbool.h> #include <stdio.h> #include <stdlib.h> #ifndef __STDC_LIB_EXT1__ typedef int errno_t ; #Afslut Hvis enum { EOK = 0 , }; // Forenklet mutex implementering struct guard_t { sem_t sem_guard ; }; typedef struct guard_t guard_t ; // Initialiser den forenklede mutex errno_t guard_init ( guard_t * guard , bool between_processes ) { int r ; r = sem_init ( & guard -> sem_guard , between_processes , 1 ); if ( r == -1 ) { returnere errno ; } returnere EOK ; } // Slip den forenklede mutex void guard_free ( guard_t * guard ) { sem_destroy ( & guard -> sem_guard ); } // Går ind i den kritiske sektion errno_t guard_enter ( guard_t * guard ) { int r ; gør { r = sem_wait ( & guard -> sem_guard ); } while (( r == -1 ) && ( fejl == EINTR )); if ( r == -1 ) { returnere errno ; } returnere EOK ; } // Afslut den kritiske sektion errno_t guard_leave ( guard_t * guard ) { int r ; r = sem_post ( & guard -> sem_guard ); if ( r == -1 ) { returnere errno ; } returnere EOK ; } // Tæller beskyttet af en forenklet mutex struct safe_counter_t { guard_t lås ; int tæller ; }; enum { // Antal ned-/forøgelsesoperationer OPERATIONS_COUNT = 100000 , }; // Tråd øger tælleren void * thread_inc_func ( void * thread_data ) { struct safe_counter_t * safe_counter = tråd_data ; for ( int i = 0 ; i < OPERATIONS_COUNT ; ++ i ) { guard_enter ( & safe_counter -> lås ); ++ safe_counter -> counter ; guard_leave ( & safe_counter -> lås ); } } // Tråd, der dekrementerer tælleren void * thread_dec_func ( void * thread_data ) { struct safe_counter_t * safe_counter = tråd_data ; for ( int i = 0 ; i < OPERATIONS_COUNT ; ++ i ) { guard_enter ( & safe_counter -> lås ); - sikker_tæller -> tæller ; guard_leave ( & safe_counter -> lås ); } } // Udskriv en fejlmeddelelse i henhold til dens kode void print_error ( errno_t errnum , const char * error_text ) { errno = errnum ; fejl ( fejltekst ); } int main ( int argc , char ** argv ) { errno_t errnum ; // initialisering struct safe_counter_t safe_counter ; safe_counter . tæller = 0 ; guard_t lås ; errnum = guard_init ( & safe_counter . lock , false ); if ( errnum ) { print_error ( errnum , "Fejl ved initialisering af mutex-lås" ); exit ( EXIT_FAILURE ); } // Start to tråde pthread_t thread_inc ; errnum = pthread_create ( & thread_inc , NULL , thread_inc_func , & safe_counter ); if ( errnum ) { print_error ( errnum , "Fejl ved oprettelse af thread_inc" ); exit ( EXIT_FAILURE ); } pthread_t thread_dec ; errnum = pthread_create ( & thread_dec , NULL , thread_dec_func , & safe_counter ); if ( errnum ) { print_error ( errnum , "Fejl ved oprettelse af thread_dec" ); exit ( EXIT_FAILURE ); } // Vent på, at tråde er færdige med at udføre errnum = pthread_join ( thread_inc , NULL ); if ( errnum ) { print_error ( errnum , "Fejl venter på thread_inc" ); exit ( EXIT_FAILURE ); } errnum = pthread_join ( thread_dec , NULL ); if ( errnum ) { print_error ( errnum , "Fejl venter på thread_dec" ); exit ( EXIT_FAILURE ); } // Frigiv data guard_free ( & lås ); // Udskriv resultatet af trådene, "0" printf ( "Tæller: %d \n " , sikker_tæller . tæller ); returner EXIT_SUCCESS ; }Synkronisering af ringbufferen er lidt mere kompliceret end at beskytte den kritiske sektion: der er allerede to semaforer, og yderligere variabler er føjet til dem . Eksemplet viser strukturen og de grundlæggende funktioner, der er nødvendige for at synkronisere en C -ringbuffer ved hjælp af POSIX -grænsefladen . Denne implementering tillader en tråd at skrive data til ringbufferen cyklisk og en anden tråd at læse fra den asynkront.
Eksempel på implementering af en synkroniseringsprimitiv for en cirkulær buffer ved hjælp af semaforer i C (POSIX) #include <errno.h> #include <semaphore.h> #include <stdio.h> #ifndef __STDC_LIB_EXT1__ typedef int errno_t ; #endif enum { EOK = 0 , }; struct ring_buffer_t { size_t længde ; størrelse_t w_indeks ; størrelse_t r_indeks ; sem_t sem_r ; sem_t sem_w ; }; errno_t ring_buffer_init ( struct ring_buffer_t * rbuf , size_t length ) { rbuf -> længde = længde ; rbuf -> r_index = 0 ; rbuf -> w_index = 0 ; int r ; r = sem_init ( & rbuf -> sem_r , 1 , 0 ); if ( r == -1 ) { returnere errno ; } errno_t errnum ; r = sem_init ( & rbuf -> sem_w , 1 , længde ); if ( r == -1 ) { errnum = errno ; goto abort_sem_r ; } returnere EOK ; abort_sem_r : sem_destroy ( & rbuf -> sem_r ); returnere errnum ; } void ring_buffer_free ( struct ring_buffer_t * rbuf ) { sem_destroy ( & rbuf -> sem_w ); sem_destroy ( & rbuf -> sem_r ); } errno_t ring_buffer_write_begin ( struct ring_buffer_t * rbuf ) { int r ; gør { r = sem_wait ( & rbuf -> sem_w ); } while (( r == -1 ) && ( fejl == EINTR )); if ( r == -1 ) { returnere errno ; } returnere EOK ; } errno_t ring_buffer_write_end ( struct ring_buffer_t * rbuf ) { ++ rbuf -> w_index ; if ( rbuf -> w_index >= rbuf -> længde ) { rbuf -> w_index = 0 ; } int r ; r = sem_post ( & rbuf -> sem_r ); if ( r == -1 ) { returnere errno ; } returnere EOK ; } errno_t ring_buffer_read_begin ( struct ring_buffer_t * rbuf ) { int r ; gør { r = sem_wait ( & rbuf -> sem_r ); } while (( r == -1 ) && ( fejl == EINTR )); if ( r == -1 ) { returnere errno ; } returnere EOK ; } errno_t ring_buffer_read_end ( struct ring_buffer_t * rbuf ) { ++ rbuf -> r_index ; if ( rbuf -> r_index >= rbuf -> længde ) { rbuf -> r_index = 0 ; } int r ; r = sem_post ( & rbuf -> sem_w ); if ( r == -1 ) { returnere errno ; } returnere EOK ; }Generelt udfører operativsystemer atomlæsning og skrivning af semafortællerværdien, men implementeringsdetaljer kan variere på forskellige arkitekturer. Ved anskaffelse af en semafor skal operativsystemet atomisk dekrementere tællerværdien, hvorefter processen kan fortsætte sit arbejde. Hvis værdien som følge af en dekrementering af tælleren kan blive negativ, så skal operativsystemet suspendere udførelsen af processen, indtil tællerværdien bliver sådan, at dekrementeringsoperationen fører til et ikke-negativt resultat [16] . I dette tilfælde kan der, afhængigt af arkitekturen på implementeringsniveauet, udføres både et forsøg på at reducere værdien af semaforen [67] og dens fald med et negativt resultat [68] . På applikationsgrænsefladeniveauet antages det almindeligvis, at minimumsværdien af en semafor er 0 [3] . Når værdien af den semafor, som processerne blev blokeret på, stiger, låses den næste proces op, og semaforværdien på applikationsniveauet forbliver lig med nul.
En lås på operativsystemniveau indebærer normalt ikke en fysisk ventetid på processoren, men overfører kontrol over processoren til en anden opgave, mens en semafor, der venter på frigivelse, kommer ind i køen af opgaver, der er blokeret af denne semafor [69] . Hvis antallet af opgaver, der er klar til udførelse, er mindre end antallet af processorer, kan operativsystemkernen skifte ledige processorer til strømbesparende tilstand, før der opstår hændelser.
For at synkronisere processorernes arbejde i multiprocessorsystemer er der specielle instruktioner, der giver dig mulighed for at beskytte adgangen til enhver celle. I x86 -arkitekturen giver Intel et præfiks for en række processorinstruktioner, LOCKder giver dig mulighed for at udføre atomoperationer på hukommelsesceller. Celleoperationer udført med præfikset LOCKblokerer andre processorer i at få adgang til cellen, hvilket på et primitivt niveau gør det muligt at organisere lette semaforer med en aktiv venteløkke [70] .
Atomisk dekrementering af en semaforværdi med 1 kan udføres med en instruktion DECLforan LOCK, som sætter fortegnsflaget, CShvis den resulterende værdi er mindre end nul. Et træk ved denne tilgang er, at semaforværdien kan være mindre end nul, så efter at have dekrementeret tælleren, kan flaget CSkontrolleres ved hjælp af instruktionen JNS, og hvis tegnet er negativt, kan operativsystemet blokere den aktuelle opgave [71] .
Instruktionen kan bruges til at øge værdien af en semafor med 1 atomisk LOCK INCL. Hvis den resulterende værdi er negativ eller lig med nul, betyder det, at der er ventende opgaver, i hvilket tilfælde operativsystemet kan låse op for den næste opgave. For at springe ophævelsesprocesser over, kan instruktionen bruges JG, som springer til etiketten, hvis nuloperationsresultatet ( ZF) og resultattegnet ( SF)-flag nulstilles til 0, det vil sige, hvis værdien er større end 0 [72] .
Under blokering, i de tilfælde, hvor der ikke er aktuelle opgaver, kan en instruktion bruges til at sætte HLTprocessoren i en laveffekttilstand, mens man venter på afbrydelser [73] , som først skal aktiveres ved hjælp af instruktionen STI. I moderne processorer kan det dog være mere optimalt at bruge instruktionerne MWAITog MONITOR. Instruktionen MWAITer ens HLT, men giver dig mulighed for at vække processoren ved at skrive til en hukommelsescelle på adressen specificeret i MONITOR. NWAITkan bruges til at overvåge semafor-slotændringer, men på multitasking-operativsystemer bruges denne instruktion til at overvåge et flag for at køre opgaveplanlæggeren på en given kerne [74] .
Reduktion af strømforbruget under den aktive dvalecyklus kan opnås ved hjælp af PAUSE[75] instruktionen .
I ARM-arkitekturenARMv7 - arkitekturen bruger såkaldte lokale og globale eksklusive skærme til at synkronisere hukommelse mellem processorer, som er statsmaskiner, der styrer atomadgang til hukommelsesceller [76] [77] . En atomlæsning af en hukommelsescelle kan udføres ved hjælp af instruktionen LDREX[78] , og en atomisk skrivning kan udføres gennem instruktionen STREX, som også returnerer succesflaget for operationen [79] .
For at mindske værdien af en semafor skal du vente, indtil dens tæller bliver større end nul. Venter kan implementeres på forskellige måder:
På niveauet af et multitasking-operativsystem kan en kombination af disse metoder bruges til at give maksimal processorudnyttelse med en overgang til strømbesparende tilstand under inaktive tider.
Forøgelse af værdien af en semafor kan være en cyklisk læsning af den aktuelle værdi af tælleren gennem instruktionen LDREX, derefter forøgelse af en kopi af værdien og forsøg på at skrive tilbage til tællerens placering ved hjælp af instruktionen STREX[84] . Efter en vellykket registrering af tælleren, hvis dens begyndelsesværdi var nul, er det nødvendigt at genoptage udførelsen af blokerede opgaver [84] , hvilket i tilfælde af et kontekstskift kan løses ved hjælp af operativsystemer [80] . Hvis processoren blev låst ved hjælp af instruktionen WFE, kan den låses op ved hjælp af instruktionen SEV, der giver besked om tilstedeværelsen af enhver hændelse [85] .
Efter at have reduceret eller øget værdien af semaforen, udføres instruktionen for at DMBsikre integriteten af hukommelsen for den ressource, der er beskyttet af semaforen [86] .
Inter-proces kommunikation | |
---|---|
Metoder | |
Udvalgte protokoller og standarder |
Datatyper | |
---|---|
Ufortolkelig | |
Numerisk | |
Tekst | |
Reference | |
Sammensatte | |
abstrakt | |
Andet | |
relaterede emner |